|
@@ -8,10 +8,17 @@ import java.io.IOException
|
|
import java.util.ArrayDeque
|
|
import java.util.ArrayDeque
|
|
import java.util.concurrent.Semaphore
|
|
import java.util.concurrent.Semaphore
|
|
import java.util.concurrent.TimeUnit
|
|
import java.util.concurrent.TimeUnit
|
|
|
|
+import kotlin.time.Duration
|
|
|
|
+import kotlin.time.Duration.Companion.seconds
|
|
|
|
+import kotlin.time.toDuration
|
|
|
|
+import kotlin.time.toDurationUnit
|
|
|
|
|
|
/**
|
|
/**
|
|
* An OkHttp interceptor that handles rate limiting.
|
|
* An OkHttp interceptor that handles rate limiting.
|
|
*
|
|
*
|
|
|
|
+ * This uses `java.time` APIs and is the legacy method, kept
|
|
|
|
+ * for compatibility reasons with existing extensions.
|
|
|
|
+ *
|
|
* Examples:
|
|
* Examples:
|
|
*
|
|
*
|
|
* permits = 5, period = 1, unit = seconds => 5 requests per second
|
|
* permits = 5, period = 1, unit = seconds => 5 requests per second
|
|
@@ -19,27 +26,43 @@ import java.util.concurrent.TimeUnit
|
|
*
|
|
*
|
|
* @since extension-lib 1.3
|
|
* @since extension-lib 1.3
|
|
*
|
|
*
|
|
- * @param permits {Int} Number of requests allowed within a period of units.
|
|
|
|
- * @param period {Long} The limiting duration. Defaults to 1.
|
|
|
|
- * @param unit {TimeUnit} The unit of time for the period. Defaults to seconds.
|
|
|
|
|
|
+ * @param permits [Int] Number of requests allowed within a period of units.
|
|
|
|
+ * @param period [Long] The limiting duration. Defaults to 1.
|
|
|
|
+ * @param unit [TimeUnit] The unit of time for the period. Defaults to seconds.
|
|
*/
|
|
*/
|
|
|
|
+@Deprecated("Use the version with kotlin.time APIs instead.")
|
|
fun OkHttpClient.Builder.rateLimit(
|
|
fun OkHttpClient.Builder.rateLimit(
|
|
permits: Int,
|
|
permits: Int,
|
|
period: Long = 1,
|
|
period: Long = 1,
|
|
unit: TimeUnit = TimeUnit.SECONDS,
|
|
unit: TimeUnit = TimeUnit.SECONDS,
|
|
-) = addInterceptor(RateLimitInterceptor(null, permits, period, unit))
|
|
|
|
|
|
+) = addInterceptor(RateLimitInterceptor(null, permits, period.toDuration(unit.toDurationUnit())))
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * An OkHttp interceptor that handles rate limiting.
|
|
|
|
+ *
|
|
|
|
+ * Examples:
|
|
|
|
+ *
|
|
|
|
+ * permits = 5, period = 1.seconds => 5 requests per second
|
|
|
|
+ * permits = 10, period = 2.minutes => 10 requests per 2 minutes
|
|
|
|
+ *
|
|
|
|
+ * @since extension-lib 1.5
|
|
|
|
+ *
|
|
|
|
+ * @param permits [Int] Number of requests allowed within a period of units.
|
|
|
|
+ * @param period [Duration] The limiting duration. Defaults to 1.seconds.
|
|
|
|
+ */
|
|
|
|
+fun OkHttpClient.Builder.rateLimit(permits: Int, period: Duration = 1.seconds) =
|
|
|
|
+ addInterceptor(RateLimitInterceptor(null, permits, period))
|
|
|
|
|
|
/** We can probably accept domains or wildcards by comparing with [endsWith], etc. */
|
|
/** We can probably accept domains or wildcards by comparing with [endsWith], etc. */
|
|
@Suppress("PLATFORM_CLASS_MAPPED_TO_KOTLIN")
|
|
@Suppress("PLATFORM_CLASS_MAPPED_TO_KOTLIN")
|
|
internal class RateLimitInterceptor(
|
|
internal class RateLimitInterceptor(
|
|
private val host: String?,
|
|
private val host: String?,
|
|
private val permits: Int,
|
|
private val permits: Int,
|
|
- period: Long,
|
|
|
|
- unit: TimeUnit,
|
|
|
|
|
|
+ period: Duration,
|
|
) : Interceptor {
|
|
) : Interceptor {
|
|
|
|
|
|
private val requestQueue = ArrayDeque<Long>(permits)
|
|
private val requestQueue = ArrayDeque<Long>(permits)
|
|
- private val rateLimitMillis = unit.toMillis(period)
|
|
|
|
|
|
+ private val rateLimitMillis = period.inWholeMilliseconds
|
|
private val fairLock = Semaphore(1, true)
|
|
private val fairLock = Semaphore(1, true)
|
|
|
|
|
|
override fun intercept(chain: Interceptor.Chain): Response {
|
|
override fun intercept(chain: Interceptor.Chain): Response {
|