Using Reviews to Rate Products and Channels

Rating value

Let's assume the rating interface is using a system with 5 stars. The rating field of a review is a number between -100 and 100.
We first have to know how to encode the number of stars in the rating field.

Number of starsRating value
★★★★★5
★★★★☆4
★★★☆☆3
★★☆☆☆2
★☆☆☆☆1
☆☆☆☆☆0

Here, we use the rating field to represent the number of stars.
We could also use this field to store a percentage (ex: 95 would mean 95 %), or to store a like/dislike information (1 would mean like, -1 would mean dislike).

Review approval process

If you do not need any approval process, skip this part

If we have an approval process for a review to be used for a product or a channel, we model the approval process with a state machine.

First of all, we create the approved state:

POST /<project-key>/states with:

final StateDraftDsl approvedStateDraft = StateDraftBuilder.of("approved", StateType.REVIEW_STATE)
.roles(asSet(StateRole.REVIEW_INCLUDED_IN_STATISTICS))
.build();
return client.execute(StateCreateCommand.of(approvedStateDraft));

Then we create the initial to-approve state, which has a possible transition to the approved state:

POST /<project-key>/states with:

final StateDraftDsl stateDraft = StateDraftBuilder.of("to-approve", StateType.REVIEW_STATE)
.initial(true)
.transitions(asSet(approvedState.toReference()))
.build();
return client.execute(StateCreateCommand.of(stateDraft));

Creating Reviews

Now we can create a review in the initial state to-approve:

POST /<project-key>/reviews with:

final ResourceIdentifier<State> stateResourceIdentifier = ResourceIdentifier.ofKey(state.getKey());
final ResourceIdentifier<Product> target = ResourceIdentifier.ofId(product.getId(), "product");
final ReviewDraftDsl reviewDraft = ReviewDraftBuilder.ofRating(4)
.key("review-1")
.state(stateResourceIdentifier)
.target(target)
.build();
return client.execute(ReviewCreateCommand.of(reviewDraft));

remove the field state if you have not created any state machine first

Query which Reviews should be approved

skip this part if you do not have any approval process

We can query which reviews should be approved with a where predicate

GET /<project-key>/reviews with query parameters:

  • where=state(id in ("<id-of-to-approve-state>"))
final ReviewQuery reviewQuery = ReviewQuery.of()
.withPredicates(m -> m.state().id().is(state.getId()));
return client.execute(reviewQuery);

Approving a Review

skip this part if you do not have any approval process

We can now approve the review review-1:

POST /<project-key>/reviews/key=review-1 with:

final ReviewUpdateCommand reviewUpdateCommand = ReviewUpdateCommand.of(review, TransitionState.of(approvedState));
return client.execute(reviewUpdateCommand);

Displaying Products

We can display all products:

  • that have at least 3 stars (average rating superior to 3)
  • with facets about the number of products rated with an average in the different ranges 0 to 1 star, 1 to 2 stars, 2 to 3 stars, 3 to 4 stars and 4 to 5 stars.
  • sorted by average ratings

For that, we use the search endpoint:

GET /<project-key>/product-projections/search with query parameters:

  • filter = reviewRatingStatistics.averageRating:range (3 to *)
  • facet = reviewRatingStatistics.averageRating:range (0 to 1), (1 to 2), (2 to 3), (3 to 4), (4 to 5)
  • sort = reviewRatingStatistics.averageRating desc
final ProductProjectionSearch productSearchWithFilter = ProductProjectionSearch.ofCurrent()
.plusQueryFilters(m -> m.reviewRatingStatistics().averageRating().isGreaterThanOrEqualTo(BigDecimal.valueOf(3)))
.plusFacets(m -> m.reviewRatingStatistics().averageRating().allRanges())
.plusSort(m -> m.reviewRatingStatistics().averageRating().desc());
return client.execute(productSearchWithFilter);

The response gives us the following information:

{
"offset": 0,
"count": <number of products to display>,
"total": <total number of products with an average rating superior to 3>,
"results": [
{
"id": "<product-1-id>",
[...]
"reviewRatingStatistics": {
"averageRating": 4.07037,
"highestRating": 5,
"lowestRating": 3,
"count": 1009,
"ratingsDistribution": {
"5": 254,
"4": 572,
"3": 183
}
}
},
{
"id": "<product-2-id>",
[...]
"reviewRatingStatistics": {
"averageRating": 2.97677,
"highestRating": 1,
"lowestRating": 0.2,
"count": 3875,
"ratingsDistribution": {
"4": 145,
"3": 3495,
"2": 235
}
}
},
[...]
],
"facets": {
"reviewRatingStatistics.averageRating": {
"type": "range",
"dataType": "number",
"ranges": [
{
"from": 0.0,
"to": 1.0,
"count": 0
},
{
"from": 1.0,
"to": 2.0,
"count": 15
},
{
"from": 2.0,
"to": 3.0,
"count": 78
},
{
"from": 3.0,
"to": 4.0,
"count": 242
},
{
"from": 4.0,
"to": 5.0,
"count": 145
}
]
}
}
}

Displaying one Product

The following information can be found in the JSON data of one product:

  • average rating: reviewRatingStatistics.averageRating.
    The value is already rounded to 5 decimals. Depending on your need, you may have to round it more.
  • number of reviews: reviewRatingStatistics.count.
  • distribution of ratings: reviewRatingStatistics.ratingsDistribution:
    ★★★★★: reviewRatingStatistics.ratingsDistribution.5
    ★★★★☆: reviewRatingStatistics.ratingsDistribution.4
    ★★★☆☆: reviewRatingStatistics.ratingsDistribution.3
    ★★☆☆☆: reviewRatingStatistics.ratingsDistribution.2
    ★☆☆☆☆: reviewRatingStatistics.ratingsDistribution.1
    ☆☆☆☆☆: reviewRatingStatistics.ratingsDistribution.0
    If one field for one rating does not exist, it means that there are no reviews with that rating value.

Displaying all Reviews of one Product

To retrieve all reviews for one product, sorted by rating:

GET /<project-key>/reviews with query parameter:

  • where = target(typeId = "product" and id = "<product-id>")
  • sort = rating desc
final ReviewQuery reviewQuery = ReviewQuery.of()
.withPredicates(m -> m.target()
.typeId().is("product")
.and(m.target().id().is(product.getId()))
)
.plusSort(m -> m.rating().sort().desc());
return client.execute(reviewQuery);

To query only the reviews used for the rating statistics of the product:

GET <project-key>/reviews with query parameters:

  • where = target(typeId = "product" and id = "<product-id>") and includedInStatistics = true
  • sort = rating desc
final ReviewQuery reviewQuery = ReviewQuery.of()
.withPredicates(m -> m.target()
.typeId().is("product")
.and(m.target().id().is(product.getId()))
.and(m.includedInStatistics().is(true)))
.plusSort(m -> m.rating().sort().desc());
return client.execute(reviewQuery);