Lesson 1
Managing Product Reviews and Data Aggregation in PHP
Introduction

Hello, and welcome to today's lesson! Today, we are going to dive into the world of managing product reviews and applying data aggregation in practice. We will start with a relatively simple Starter Task to set up our base and then gradually build up to a more complex solution involving data aggregation. Let's jump in!

Starter Task: Methods and Their Definitions

For our starter task, we will lay the foundation by implementing basic operations for managing product reviews. We will be using classes to encapsulate the details of a product review. Our Review class will consist of the following properties:

  • $text — The textual content of the review.
  • $rating — The rating score of the review, ranging from 1 to 5.
  • $flagged — A boolean indicating whether the review is flagged as inappropriate.

These are the methods we will implement in a ReviewManager class:

  • addReview($productId, $reviewId, $reviewText, $rating) — Adds a review to the product specified by $productId. If a review with $reviewId already exists, it updates the existing review. Returns true if the review was added or updated successfully, false otherwise. For newly added items, the $flagged attribute is set to false initially.

  • getReview($productId, $reviewId) — Returns the review details ($text, $rating, and $flagged fields) for the review specified by $reviewId under the given $productId. If the review or product does not exist, returns null.

  • deleteReview($productId, $reviewId) — Deletes the review specified by $reviewId under the given $productId. Returns true if the review was deleted, false otherwise.

Starter Task Implementation

Let's look at the code that implements these functionalities in PHP:

php
1<?php 2 3class Review { 4 public $text; 5 public $rating; 6 public $flagged; 7 8 public function __construct($text, $rating, $flagged = false) { 9 $this->text = $text; 10 $this->rating = $rating; 11 $this->flagged = $flagged; 12 } 13} 14 15class ReviewManager { 16 private $products = []; 17 18 public function addReview($productId, $reviewId, $reviewText, $rating) { 19 if ($rating < 1 || $rating > 5) { 20 return false; // Invalid rating 21 } 22 $this->products[$productId][$reviewId] = new Review($reviewText, $rating); 23 return true; 24 } 25 26 public function getReview($productId, $reviewId) { 27 if (isset($this->products[$productId][$reviewId])) { 28 return $this->products[$productId][$reviewId]; 29 } 30 return null; 31 } 32 33 public function deleteReview($productId, $reviewId) { 34 if (isset($this->products[$productId][$reviewId])) { 35 unset($this->products[$productId][$reviewId]); 36 if (empty($this->products[$productId])) { 37 unset($this->products[$productId]); // Remove product if no reviews left 38 } 39 return true; 40 } 41 return false; 42 } 43} 44 45// Example usage 46$reviewManager = new ReviewManager(); 47 48// Adding some reviews 49$reviewManager->addReview("p1", "r1", "Great product!", 5); 50$reviewManager->addReview("p1", "r2", "Not bad", 3); 51 52// Testing getReview method 53$reviewR1 = $reviewManager->getReview("p1", "r1"); 54if ($reviewR1 !== null) { 55 echo "Review r1 for p1: {"; 56 echo "text: " . $reviewR1->text . ", "; 57 echo "rating: " . $reviewR1->rating . ", "; 58 echo "flagged: " . ($reviewR1->flagged ? "true" : "false") . "}\n"; 59} else { 60 echo "Review r1 for p1 not found.\n"; 61} 62 63$reviewR3 = $reviewManager->getReview("p1", "r3"); 64if ($reviewR3 !== null) { 65 echo "Review r3 for p1 found.\n"; 66} else { 67 echo "Review r3 for p1 not found.\n"; 68} 69 70// Testing deleteReview method 71if ($reviewManager->deleteReview("p1", "r2")) { 72 echo "Review r2 for p1 deleted successfully.\n"; 73} else { 74 echo "Failed to delete review r2 for p1.\n"; 75} 76 77$reviewR2 = $reviewManager->getReview("p1", "r2"); 78if ($reviewR2 !== null) { 79 echo "Review r2 for p1 found.\n"; 80} else { 81 echo "Review r2 for p1 not found.\n"; 82}

This code establishes the foundational methods needed for managing product reviews within a ReviewManager class. The addReview method allows for adding a new review or updating an existing one, ensuring each review contains valid rating values between 1 and 5. The getReview method retrieves the review details for a specific product, including the review text and rating, returning null if the product or review doesn't exist. The deleteReview method facilitates the removal of a specific review, and if no reviews are left for a product, the product itself is removed from the product list. Together, these methods form the basic operations required to manage a collection of product reviews efficiently.

Now, let's extend this with new features.

New Task: Advanced Functions and Data Aggregation

With our basic review management system in place, we will now introduce new methods to handle more complex operations, such as flagging inappropriate reviews and aggregating review data for a specific product.

Here are the new methods we will add:

  • flagReview($productId, $reviewId) — This method flags a specific review as inappropriate for a given product. Returns true if the review was successfully flagged, false otherwise.

  • aggregateReviews($productId) — This method aggregates review data for a given product, providing statistics such as the total number of reviews, the number of flagged reviews, average rating, and review texts excluding flagged ones. If the product does not have any reviews or does not exist, returns null.

Implementation, Step 1: Adding the 'flag_review' Method

First, let's add functionality to flag a review:

php
1<?php 2 3class Review { 4 public $text; 5 public $rating; 6 public $flagged; 7 8 public function __construct($text, $rating, $flagged = false) { 9 $this->text = $text; 10 $this->rating = $rating; 11 $this->flagged = $flagged; 12 } 13} 14 15class ReviewManager { 16 private $products = []; 17 18 public function addReview($productId, $reviewId, $reviewText, $rating) { 19 if ($rating < 1 || $rating > 5) { 20 return false; // Invalid rating 21 } 22 $this->products[$productId][$reviewId] = new Review($reviewText, $rating); 23 return true; 24 } 25 26 public function getReview($productId, $reviewId) { 27 if (isset($this->products[$productId][$reviewId])) { 28 return $this->products[$productId][$reviewId]; 29 } 30 return null; 31 } 32 33 public function deleteReview($productId, $reviewId) { 34 if (isset($this->products[$productId][$reviewId])) { 35 unset($this->products[$productId][$reviewId]); 36 if (empty($this->products[$productId])) { 37 unset($this->products[$productId]); // Remove product if no reviews left 38 } 39 return true; 40 } 41 return false; 42 } 43 44 public function flagReview($productId, $reviewId) { 45 if (isset($this->products[$productId][$reviewId])) { 46 $this->products[$productId][$reviewId]->flagged = true; 47 return true; 48 } 49 return false; 50 } 51} 52 53// Example usage 54$reviewManager = new ReviewManager(); 55 56// Adding some reviews 57$reviewManager->addReview("p1", "r1", "Great product!", 5); 58$reviewManager->addReview("p1", "r2", "Not bad", 3); 59$reviewManager->addReview("p1", "r3", "Terrible", 1); 60 61// Flagging a review 62$reviewManager->flagReview("p1", "r3"); 63 64// Testing flagReview method 65$reviewR3 = $reviewManager->getReview("p1", "r3"); 66if ($reviewR3 !== null) { 67 echo "Review r3 for p1: {"; 68 echo "text: " . $reviewR3->text . ", "; 69 echo "rating: " . $reviewR3->rating . ", "; 70 echo "flagged: " . ($reviewR3->flagged ? "true" : "false") . "}\n"; 71} else { 72 echo "Review r3 for p1 not found.\n"; 73}

In this step, we are adding the flagReview method to our ReviewManager class. This method enables users to mark a specific review as inappropriate. It checks whether the product and review exist in the dataset, and if they do, it sets the flagged attribute of the review to true. This flagging mechanism is crucial for maintaining the quality and appropriateness of the reviews in the system.

Step 2: Adding the 'aggregate_reviews' Method

Next, we will implement the method to aggregate reviews:

php
1<?php 2 3class Review { 4 public $text; 5 public $rating; 6 public $flagged; 7 8 public function __construct($text, $rating, $flagged = false) { 9 $this->text = $text; 10 $this->rating = $rating; 11 $this->flagged = $flagged; 12 } 13} 14 15class AggregatedData { 16 public $totalReviews; 17 public $flaggedReviews; 18 public $averageRating; 19 public $reviewTexts; 20 21 public function __construct($totalReviews, $flaggedReviews, $averageRating, $reviewTexts) { 22 $this->totalReviews = $totalReviews; 23 $this->flaggedReviews = $flaggedReviews; 24 $this->averageRating = $averageRating; 25 $this->reviewTexts = $reviewTexts; 26 } 27} 28 29class ReviewManager { 30 private $products = []; 31 32 public function addReview($productId, $reviewId, $reviewText, $rating) { 33 if ($rating < 1 || $rating > 5) { 34 return false; // Invalid rating 35 } 36 $this->products[$productId][$reviewId] = new Review($reviewText, $rating); 37 return true; 38 } 39 40 public function getReview($productId, $reviewId) { 41 if (isset($this->products[$productId][$reviewId])) { 42 return $this->products[$productId][$reviewId]; 43 } 44 return null; 45 } 46 47 public function deleteReview($productId, $reviewId) { 48 if (isset($this->products[$productId][$reviewId])) { 49 unset($this->products[$productId][$reviewId]); 50 if (empty($this->products[$productId])) { 51 unset($this->products[$productId]); // Remove product if no reviews left 52 } 53 return true; 54 } 55 return false; 56 } 57 58 public function flagReview($productId, $reviewId) { 59 if (isset($this->products[$productId][$reviewId])) { 60 $this->products[$productId][$reviewId]->flagged = true; 61 return true; 62 } 63 return false; 64 } 65 66 public function aggregateReviews($productId) { 67 if (!isset($this->products[$productId]) || empty($this->products[$productId])) { 68 return null; // No reviews or product doesn't exist 69 } 70 71 $totalReviews = count($this->products[$productId]); 72 $flaggedReviews = 0; 73 $totalRating = 0; 74 $reviewTexts = []; 75 76 foreach ($this->products[$productId] as $review) { 77 if ($review->flagged) { 78 $flaggedReviews++; 79 } else { 80 $totalRating += $review->rating; 81 $reviewTexts[] = $review->text; 82 } 83 } 84 85 $averageRating = ($totalReviews == $flaggedReviews) ? 0.0 : $totalRating / ($totalReviews - $flaggedReviews); 86 return new AggregatedData($totalReviews, $flaggedReviews, $averageRating, $reviewTexts); 87 } 88} 89 90// Example usage 91$reviewManager = new ReviewManager(); 92 93// Adding some reviews 94$reviewManager->addReview("p1", "r1", "Great product!", 5); 95$reviewManager->addReview("p1", "r2", "Not bad", 3); 96$reviewManager->addReview("p1", "r3", "Terrible", 1); 97 98// Flagging a review 99$reviewManager->flagReview("p1", "r3"); 100 101// Testing the aggregation method 102$aggregatedData = $reviewManager->aggregateReviews("p1"); 103if ($aggregatedData !== null) { 104 echo "Aggregated data for p1: {\n"; 105 echo "total_reviews: " . $aggregatedData->totalReviews . ",\n"; 106 echo "flagged_reviews: " . $aggregatedData->flaggedReviews . ",\n"; 107 echo "average_rating: " . $aggregatedData->averageRating . ",\n"; 108 echo "review_texts: ["; 109 foreach ($aggregatedData->reviewTexts as $text) { 110 echo "\"$text\", "; 111 } 112 echo "]\n}\n"; 113} else { 114 echo "No aggregated data for p1.\n"; 115}

In this step, the aggregateReviews method is added to the ReviewManager class, along with the definition of an AggregatedData class. The method aggregates review data for a given product by calculating various statistics, such as the total number of reviews, the number of flagged reviews, the average rating excluding flagged reviews, and a list of review texts that are not flagged. The method first ensures the product exists and contains reviews. It then iterates through the reviews to collect the necessary data, considering only non-flagged reviews for the average rating and review texts. If all reviews are flagged, the average rating defaults to zero. This aggregation provides a comprehensive overview of a product’s review status, useful for both users and administrators.

Conclusion

Great job! Today, you have learned how to manage product reviews and apply data aggregation in practice using PHP. We started with basic operations for adding, retrieving, and deleting reviews. Then, we extended our functionality to include flagging reviews and aggregating review data. This gradual build-up demonstrates how to enhance features incrementally and handle more complex data aggregation tasks.

Feel free to practice solving similar challenges to strengthen your skills further. Keep coding, and see you in the next lesson!

Enjoy this lesson? Now it's time to practice with Cosmo!
Practice is how you turn knowledge into actual skills.