cohort.CohortStorage.kt Maven / Gradle / Ivy
package com.amplitude.experiment.cohort
import com.amplitude.experiment.EvaluationProxyConfig
import com.amplitude.experiment.LocalEvaluationMetrics
import com.amplitude.experiment.util.Cache
import com.amplitude.experiment.util.LocalEvaluationMetricsWrapper
import com.amplitude.experiment.util.Logger
import com.amplitude.experiment.util.USER_GROUP_TYPE
import com.amplitude.experiment.util.wrapMetrics
import java.util.concurrent.locks.ReentrantLock
import java.util.concurrent.locks.ReentrantReadWriteLock
import kotlin.concurrent.read
import kotlin.concurrent.withLock
import kotlin.concurrent.write
internal interface CohortStorage {
fun getCohort(cohortId: String): Cohort?
fun getCohorts(): Map
fun getCohortsForUser(userId: String, cohortIds: Set): Set
fun getCohortsForGroup(groupType: String, groupName: String, cohortIds: Set): Set
fun putCohort(cohort: Cohort)
fun deleteCohort(cohortId: String)
}
internal class ProxyCohortStorage(
private val proxyConfig: EvaluationProxyConfig,
private val membershipApi: CohortMembershipApi,
private val metrics: LocalEvaluationMetrics = LocalEvaluationMetricsWrapper()
) : CohortStorage {
private val storage: CohortStorage = InMemoryCohortStorage()
private val membershipCacheLock = ReentrantLock()
private val membershipCache = mutableMapOf>>()
override fun getCohort(cohortId: String): Cohort? {
return storage.getCohort(cohortId)
}
override fun getCohorts(): Map {
return storage.getCohorts()
}
override fun getCohortsForUser(userId: String, cohortIds: Set): Set {
return getCohortsForGroup(USER_GROUP_TYPE, userId, cohortIds)
}
override fun getCohortsForGroup(groupType: String, groupName: String, cohortIds: Set): Set {
val localCohortIds = storage.getCohorts().keys
return if (localCohortIds.containsAll(cohortIds)) {
storage.getCohortsForGroup(groupType, groupName, cohortIds)
} else {
getMembershipCache(groupType)[groupName]
?: try {
wrapMetrics(
metric = metrics::onProxyCohortMembership,
failure = metrics::onProxyCohortMembershipFailure
) {
membershipApi.getCohortMemberships(groupType, groupName).also { proxyCohortMemberships ->
getMembershipCache(groupType)[groupName] = proxyCohortMemberships
}
}
} catch (e: Exception) {
Logger.e("Failed to get cohort membership from proxy.", e)
// Fall back on in memory storage in the case of proxy failure.
storage.getCohortsForGroup(groupType, groupName, cohortIds)
}
}
}
override fun putCohort(cohort: Cohort) {
storage.putCohort(cohort)
}
override fun deleteCohort(cohortId: String) {
storage.deleteCohort(cohortId)
}
private fun getMembershipCache(groupType: String): Cache> {
return membershipCacheLock.withLock {
membershipCache.getOrPut(groupType) {
Cache(
proxyConfig.cohortCacheCapacity,
proxyConfig.cohortCacheTtlMillis
)
}
}
}
}
internal class InMemoryCohortStorage : CohortStorage {
private val lock = ReentrantReadWriteLock()
private val cohortStore = mutableMapOf()
override fun getCohort(cohortId: String): Cohort? {
lock.read {
return cohortStore[cohortId]
}
}
override fun getCohorts(): Map {
lock.read {
return cohortStore.toMap()
}
}
override fun getCohortsForUser(userId: String, cohortIds: Set): Set {
return getCohortsForGroup(USER_GROUP_TYPE, userId, cohortIds)
}
override fun getCohortsForGroup(groupType: String, groupName: String, cohortIds: Set): Set {
val result = mutableSetOf()
lock.read {
for (cohortId in cohortIds) {
val cohort = cohortStore[cohortId]
if (cohort == null) {
Logger.w("Targeted $groupType cohort $cohortId not found in storage.")
continue
}
if (cohort.groupType != groupType) {
continue
}
if (cohort.members.contains(groupName)) {
result.add(cohortId)
}
}
}
return result
}
override fun putCohort(cohort: Cohort) {
lock.write {
cohortStore[cohort.id] = cohort
}
}
override fun deleteCohort(cohortId: String) {
lock.write {
cohortStore.remove(cohortId)
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy