cohort.CohortStorage.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of evaluation-proxy-core Show documentation
Show all versions of evaluation-proxy-core Show documentation
Core package for Amplitude's evaluation proxy.
package com.amplitude.cohort
import com.amplitude.RedisConfiguration
import com.amplitude.util.RedisConnection
import com.amplitude.util.RedisKey
import com.amplitude.util.json
import kotlinx.coroutines.sync.Mutex
import kotlinx.coroutines.sync.withLock
import kotlinx.serialization.encodeToString
import kotlin.time.Duration
internal interface CohortStorage {
suspend fun getCohortDescription(cohortId: String): CohortDescription?
suspend fun getCohortDescriptions(): Map
suspend fun getCohortMembers(cohortDescription: CohortDescription): Set?
suspend fun getCohortMembershipsForUser(userId: String, cohortIds: Set? = null): Set
suspend fun getCohortMembershipsForGroup(
groupType: String,
groupName: String,
cohortIds: Set? = null
): Set
suspend fun putCohort(description: CohortDescription, members: Set)
suspend fun removeCohort(cohortDescription: CohortDescription)
}
internal fun getCohortStorage(projectId: String, redisConfiguration: RedisConfiguration?, ttl: Duration): CohortStorage {
val uri = redisConfiguration?.uri
return if (uri == null) {
InMemoryCohortStorage()
} else {
val redis = RedisConnection(uri)
val readOnlyRedis = if (redisConfiguration.readOnlyUri != null) {
RedisConnection(redisConfiguration.readOnlyUri)
} else {
redis
}
RedisCohortStorage(projectId, ttl, redisConfiguration.prefix, redis, readOnlyRedis)
}
}
internal class InMemoryCohortStorage : CohortStorage {
private class Cohort(
val description: CohortDescription,
val members: Set
)
private val lock = Mutex()
private val cohorts = mutableMapOf()
override suspend fun getCohortDescription(cohortId: String): CohortDescription? {
return lock.withLock { cohorts[cohortId]?.description }
}
override suspend fun getCohortDescriptions(): Map {
return lock.withLock { cohorts.mapValues { it.value.description } }
}
override suspend fun getCohortMembers(cohortDescription: CohortDescription): Set? {
return lock.withLock { cohorts[cohortDescription.id]?.members }
}
override suspend fun putCohort(description: CohortDescription, members: Set) {
return lock.withLock { cohorts[description.id] = Cohort(description, members) }
}
override suspend fun removeCohort(cohortDescription: CohortDescription) {
lock.withLock { cohorts.remove(cohortDescription.id) }
}
override suspend fun getCohortMembershipsForUser(userId: String, cohortIds: Set?): Set {
return lock.withLock {
(cohortIds ?: cohorts.keys).mapNotNull { id ->
when (cohorts[id]?.members?.contains(userId)) {
true -> id
else -> null
}
}.toSet()
}
}
override suspend fun getCohortMembershipsForGroup(
groupType: String,
groupName: String,
cohortIds: Set?
): Set {
return lock.withLock {
(cohortIds ?: cohorts.keys).mapNotNull { id ->
val cohort = cohorts[id]
if (cohort?.description?.groupType != groupType) {
null
} else {
when (cohort.members.contains(groupName)) {
true -> id
else -> null
}
}
}.toSet()
}
}
}
internal class RedisCohortStorage(
private val projectId: String,
private val ttl: Duration,
private val prefix: String,
private val redis: RedisConnection,
private val readOnlyRedis: RedisConnection
) : CohortStorage {
override suspend fun getCohortDescription(cohortId: String): CohortDescription? {
val jsonEncodedDescription = redis.hget(RedisKey.CohortDescriptions(prefix, projectId), cohortId) ?: return null
return json.decodeFromString(jsonEncodedDescription)
}
override suspend fun getCohortDescriptions(): Map {
val jsonEncodedDescriptions = redis.hgetall(RedisKey.CohortDescriptions(prefix, projectId))
return jsonEncodedDescriptions?.mapValues { json.decodeFromString(it.value) } ?: mapOf()
}
override suspend fun getCohortMembers(cohortDescription: CohortDescription): Set? {
return redis.smembers(RedisKey.CohortMembers(prefix, projectId, cohortDescription))
}
override suspend fun getCohortMembershipsForUser(userId: String, cohortIds: Set?): Set {
val descriptions = getCohortDescriptions()
val memberships = mutableSetOf()
for (description in descriptions.values) {
if (cohortIds != null && !cohortIds.contains(description.id)) {
continue
}
// High volume, use read connection
val isMember = readOnlyRedis.sismember(RedisKey.CohortMembers(prefix, projectId, description), userId)
if (isMember) {
memberships += description.id
}
}
return memberships
}
override suspend fun getCohortMembershipsForGroup(
groupType: String,
groupName: String,
cohortIds: Set?
): Set {
val descriptions = getCohortDescriptions()
val memberships = mutableSetOf()
for (description in descriptions.values) {
if (cohortIds != null && !cohortIds.contains(description.id)) {
continue
}
if (description.groupType != groupType) {
continue
}
// High volume, use read connection
val isMember = readOnlyRedis.sismember(RedisKey.CohortMembers(prefix, projectId, description), groupName)
if (isMember) {
memberships += description.id
}
}
return memberships
}
override suspend fun putCohort(description: CohortDescription, members: Set) {
val jsonEncodedDescription = json.encodeToString(description)
val existingDescription = getCohortDescription(description.id)
if ((existingDescription?.lastComputed ?: 0L) < description.lastComputed) {
redis.sadd(RedisKey.CohortMembers(prefix, projectId, description), members)
redis.hset(RedisKey.CohortDescriptions(prefix, projectId), mapOf(description.id to jsonEncodedDescription))
if (existingDescription != null) {
redis.expire(RedisKey.CohortMembers(prefix, projectId, existingDescription), ttl)
}
}
}
override suspend fun removeCohort(cohortDescription: CohortDescription) {
redis.hdel(RedisKey.CohortDescriptions(prefix, projectId), cohortDescription.id)
redis.del(RedisKey.CohortMembers(prefix, projectId, cohortDescription))
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy