You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
iceraven-browser/app/src/main/java/org/mozilla/fenix/shopping/middleware/ReviewQualityCheckService.kt

236 lines
8.2 KiB
Kotlin

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
package org.mozilla.fenix.shopping.middleware
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.withContext
import mozilla.components.browser.state.selector.selectedTab
import mozilla.components.browser.state.store.BrowserStore
import mozilla.components.concept.engine.shopping.ProductAnalysis
import mozilla.components.concept.engine.shopping.ProductRecommendation
import mozilla.components.support.base.log.logger.Logger
import org.mozilla.fenix.GleanMetrics.Shopping
import kotlin.coroutines.resume
import kotlin.coroutines.suspendCoroutine
/**
* Service that handles the network requests for the review quality check feature.
*/
interface ReviewQualityCheckService {
/**
* Fetches the product review for the current tab.
*
* @return [ProductAnalysis] if the request succeeds, null otherwise.
*/
suspend fun fetchProductReview(): ProductAnalysis?
/**
* Triggers a reanalysis of the product review for the current tab.
*
* @return [AnalysisStatusDto] if the request succeeds, null otherwise.
*/
suspend fun reanalyzeProduct(): AnalysisStatusDto?
/**
* Fetches the status of the product review for the current tab.
*
* @return [AnalysisStatusProgressDto] if the request succeeds, null otherwise.
*/
suspend fun analysisStatus(): AnalysisStatusProgressDto?
/**
* Fetches product recommendations related to the product user is browsing in the current tab.
*
* @return [ProductRecommendation] if request succeeds, null otherwise.
*/
suspend fun productRecommendation(shouldRecordAvailableTelemetry: Boolean): ProductRecommendation?
/**
* Reports that a product is back in stock.
*
* @return [ReportBackInStockStatusDto] if the request succeeds, null otherwise.
*/
suspend fun reportBackInStock(): ReportBackInStockStatusDto?
}
/**
* Service that handles the network requests for the review quality check feature.
*
* @param browserStore Reference to the application's [BrowserStore] to access state.
*/
class DefaultReviewQualityCheckService(
private val browserStore: BrowserStore,
) : ReviewQualityCheckService {
private val recommendationsCache: MutableMap<String, ProductRecommendation> = mutableMapOf()
private val logger = Logger("DefaultReviewQualityCheckService")
override suspend fun fetchProductReview(): ProductAnalysis? = withContext(Dispatchers.Main) {
suspendCoroutine { continuation ->
browserStore.state.selectedTab?.let { tab ->
tab.engineState.engineSession?.requestProductAnalysis(
url = tab.content.url,
onResult = {
continuation.resume(it)
},
onException = {
logger.error("Error fetching product review", it)
continuation.resume(null)
},
)
}
}
}
override suspend fun reanalyzeProduct(): AnalysisStatusDto? = withContext(Dispatchers.Main) {
suspendCoroutine { continuation ->
browserStore.state.selectedTab?.let { tab ->
tab.engineState.engineSession?.reanalyzeProduct(
url = tab.content.url,
onResult = {
continuation.resume(it.asEnumOrDefault(AnalysisStatusDto.OTHER))
},
onException = {
logger.error("Error starting reanalysis", it)
continuation.resume(null)
},
)
}
}
}
override suspend fun analysisStatus(): AnalysisStatusProgressDto? = withContext(Dispatchers.Main) {
suspendCoroutine { continuation ->
browserStore.state.selectedTab?.let { tab ->
tab.engineState.engineSession?.requestAnalysisStatus(
url = tab.content.url,
onResult = {
continuation.resume(
AnalysisStatusProgressDto(
status = it.status.asEnumOrDefault(AnalysisStatusDto.OTHER)!!,
progress = it.progress,
),
)
},
onException = {
logger.error("Error fetching analysis status", it)
continuation.resume(null)
},
)
}
}
}
override suspend fun productRecommendation(shouldRecordAvailableTelemetry: Boolean): ProductRecommendation? =
withContext(Dispatchers.Main) {
suspendCoroutine { continuation ->
browserStore.state.selectedTab?.let { tab ->
if (recommendationsCache.containsKey(tab.content.url)) {
continuation.resume(recommendationsCache[tab.content.url])
} else {
tab.engineState.engineSession?.requestProductRecommendations(
url = tab.content.url,
onResult = {
if (it.isEmpty()) {
if (shouldRecordAvailableTelemetry) {
Shopping.surfaceNoAdsAvailable.record()
}
} else {
Shopping.adsExposure.record()
}
// Return the first available recommendation since ui requires only
// one recommendation.
continuation.resume(
it.firstOrNull()?.also { recommendation ->
recommendationsCache[tab.content.url] = recommendation
},
)
},
onException = {
logger.error("Error fetching product recommendation", it)
continuation.resume(null)
},
)
}
}
}
}
override suspend fun reportBackInStock(): ReportBackInStockStatusDto? = withContext(Dispatchers.Main) {
suspendCoroutine { continuation ->
browserStore.state.selectedTab?.let { tab ->
tab.engineState.engineSession?.reportBackInStock(
url = tab.content.url,
onResult = {
continuation.resume(it.asEnumOrDefault<ReportBackInStockStatusDto>())
},
onException = {
logger.error("Error reporting product back in stock", it)
continuation.resume(null)
},
)
}
}
}
}
/**
* Enum that represents the status of the product review analysis.
*/
enum class AnalysisStatusDto {
/**
* Analysis is waiting to be picked up.
*/
PENDING,
/**
* Analysis is in progress.
*/
IN_PROGRESS,
/**
* Analysis is completed.
*/
COMPLETED,
/**
* Any other status.
*/
OTHER,
}
/**
* Class that represents the analysis status response of the product review analysis.
*
* @property status Enum indicating the current status of the analysis
* @property progress Number indicating the progress of the analysis
*/
data class AnalysisStatusProgressDto(
val status: AnalysisStatusDto,
val progress: Double,
)
/**
* Enum that represents the status returned from reporting a product is back in stock.
*/
enum class ReportBackInStockStatusDto {
/**
* Report created.
*/
REPORT_CREATED,
/**
* Product is already reported to be back in stock.
*/
ALREADY_REPORTED,
/**
* Product was not actually marked as deleted.
*/
NOT_DELETED,
}