From 7ae3daf1874068f39d204271970d7e4a0cac5d98 Mon Sep 17 00:00:00 2001 From: Michael Rittmeister Date: Sun, 3 Sep 2023 12:06:34 -0400 Subject: [PATCH] Don't use value class Reset with AtomicRef (#863) The use of value classes for the Reset class has already caused issues implementing #855. However, it was isolated to native platforms. Unfortunately, this behavior will become a compiler error soon, so we need to change it. This can be achieved by simply using the backing Instant instead of Reset with AtomicRef. Related Issues: #855 #69 Kotlin/kotlinx-atomicfu#291 https://youtrack.jetbrains.com/issue/KT-61584 Co-authored-by: lukellmann --- .../kotlin/ratelimit/AbstractRateLimiter.kt | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/rest/src/commonMain/kotlin/ratelimit/AbstractRateLimiter.kt b/rest/src/commonMain/kotlin/ratelimit/AbstractRateLimiter.kt index b7803871eb8..874360ba85d 100644 --- a/rest/src/commonMain/kotlin/ratelimit/AbstractRateLimiter.kt +++ b/rest/src/commonMain/kotlin/ratelimit/AbstractRateLimiter.kt @@ -6,11 +6,11 @@ import dev.kord.rest.request.Request import dev.kord.rest.request.RequestIdentifier import dev.kord.rest.request.identifier import kotlinx.atomicfu.atomic -import kotlinx.atomicfu.update import kotlinx.coroutines.CompletableDeferred import kotlinx.coroutines.delay import kotlinx.coroutines.sync.Mutex import kotlinx.datetime.Clock +import kotlinx.datetime.Instant import mu.KLogger import kotlin.time.Duration.Companion.minutes @@ -18,7 +18,7 @@ public abstract class AbstractRateLimiter internal constructor(public val clock: internal abstract val logger: KLogger private val autoBanRateLimiter = IntervalRateLimiter(limit = 25000, interval = 10.minutes) - private val globalSuspensionPoint = atomic(Reset(clock.now())) + private val globalSuspensionPoint = atomic(clock.now()) internal val buckets = ConcurrentHashMap() private val routeBuckets = ConcurrentHashMap>() @@ -26,8 +26,8 @@ public abstract class AbstractRateLimiter internal constructor(public val clock: private val Request<*, *>.buckets get() = routeBuckets[identifier].orEmpty().map { it.bucket } internal fun RequestIdentifier.addBucket(id: BucketKey) = routeBuckets.getOrPut(this) { mutableSetOf() }.add(id) - internal suspend fun Reset.await() { - val duration = value - clock.now() + private suspend fun Instant.await() { + val duration = this - clock.now() if (duration.isNegative()) return delay(duration) } @@ -67,7 +67,7 @@ public abstract class AbstractRateLimiter internal constructor(public val clock: when (response) { is RequestResponse.GlobalRateLimit -> { logger.trace { "[RATE LIMIT]:[GLOBAL]:exhausted until ${response.reset.value}" } - globalSuspensionPoint.update { response.reset } + globalSuspensionPoint.value = response.reset.value } is RequestResponse.BucketRateLimit -> { logger.trace { "[RATE LIMIT]:[BUCKET]:Bucket ${response.bucketKey.value} was exhausted until ${response.reset.value}" } @@ -83,7 +83,7 @@ public abstract class AbstractRateLimiter internal constructor(public val clock: } internal inner class Bucket(val id: BucketKey) { - private val reset = atomic(Reset(clock.now())) + private val reset = atomic(clock.now()) private val mutex = Mutex() suspend fun awaitAndLock() { @@ -93,7 +93,7 @@ public abstract class AbstractRateLimiter internal constructor(public val clock: } fun updateReset(newValue: Reset) { - reset.update { newValue } + reset.value = newValue.value } fun unlock() = mutex.unlock()