All Downloads are FREE. Search and download functionalities are using the official Maven repository.

es.moki.ratelimitj.redis.request.RedisSlidingWindowRequestRateLimiter Maven / Gradle / Ivy

package es.moki.ratelimitj.redis.request;


import com.lambdaworks.redis.api.StatefulRedisConnection;
import es.moki.ratelimitj.core.limiter.request.AsyncRequestRateLimiter;
import es.moki.ratelimitj.core.limiter.request.ReactiveRequestRateLimiter;
import es.moki.ratelimitj.core.limiter.request.RequestLimitRule;
import es.moki.ratelimitj.core.limiter.request.RequestRateLimiter;
import es.moki.ratelimitj.core.time.SystemTimeSupplier;
import es.moki.ratelimitj.core.time.TimeSupplier;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import reactor.core.publisher.Mono;

import javax.annotation.concurrent.ThreadSafe;
import java.util.Set;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.TimeUnit;

import static com.lambdaworks.redis.ScriptOutputType.VALUE;
import static java.util.Objects.requireNonNull;

@ThreadSafe
public class RedisSlidingWindowRequestRateLimiter implements RequestRateLimiter, AsyncRequestRateLimiter, ReactiveRequestRateLimiter {

    private static final Logger LOG = LoggerFactory.getLogger(RedisSlidingWindowRequestRateLimiter.class);

    private final LimitRuleJsonSerialiser serialiser = new LimitRuleJsonSerialiser();

    private final StatefulRedisConnection connection;
    private final RedisScriptLoader scriptLoader;
    private final String rulesJson;
    private final TimeSupplier timeSupplier;

    public RedisSlidingWindowRequestRateLimiter(StatefulRedisConnection connection, Set rules) {
        this(connection, rules, new SystemTimeSupplier());
    }

    public RedisSlidingWindowRequestRateLimiter(StatefulRedisConnection connection, Set rules, TimeSupplier timeSupplier) {
        this.connection = connection;
        scriptLoader = new RedisScriptLoader(connection, "sliding-window-ratelimit.lua");
        rulesJson = serialiserLimitRules(rules);
        this.timeSupplier = timeSupplier;
    }

    private String serialiserLimitRules(Set rules) {
        return serialiser.encode(rules);
    }

    // TODO support multi keys

    public CompletionStage overLimitAsync(String key) {
        return overLimitAsync(key, 1);
    }

    public CompletionStage overLimitAsync(String key, int weight) {
        return eqOrGeLimitAsync(key, weight, true);
    }

    @Override
    public CompletionStage resetLimitAsync(String key) {
        return connection.async().del(key).thenApply(result -> 1 == result);
    }

    @Override
    public boolean overLimitWhenIncremented(String key) {
        return overLimitWhenIncremented(key, 1);
    }

    @Override
    public boolean overLimitWhenIncremented(String key, int weight) {
        return toBlocking(eqOrGeLimitAsync(key, weight, true));
    }

    @Override
    public boolean geLimitWhenIncremented(String key) {
        return geLimitWhenIncremented(key, 1);
    }

    @Override
    public boolean geLimitWhenIncremented(String key, int weight) {
        return toBlocking(eqOrGeLimitAsync(key, weight, false));
    }

//    @Override
//    public boolean isOverLimit(String key) {
//        return overLimitWhenIncremented(key, 0);
//    }
//
//    @Override
//    public boolean isGeLimit(String key) {
//        return geLimitWhenIncremented(key, 0);
//    }

    @Override
    public boolean resetLimit(String key) {
        return toBlocking(resetLimitAsync(key));
    }

    @Override
    public Mono overLimitWhenIncrementedReactive(String key) {
        return Mono.fromFuture(overLimitAsync(key).toCompletableFuture());
    }

    @Override
    public Mono overLimitWhenIncrementedReactive(String key, int weight) {
        return Mono.fromFuture(overLimitAsync(key, weight).toCompletableFuture());
    }

    @Override
    public  Mono geLimitWhenIncrementedReactive(String key) {
        return Mono.fromFuture(eqOrGeLimitAsync(key, 1, false).toCompletableFuture());
    }

    @Override
    public  Mono geLimitWhenIncrementedReactive(String key, int weight) {
        return Mono.fromFuture(eqOrGeLimitAsync(key, weight, false).toCompletableFuture());
    }

    @Override
    public Mono resetLimitReactive(String key) {
        return Mono.fromFuture(resetLimitAsync(key).toCompletableFuture());
    }

    private CompletionStage eqOrGeLimitAsync(String key, int weight, boolean strictlyGreater) {
        requireNonNull(key);

        String sha = scriptLoader.scriptSha();

        return timeSupplier.getAsync().thenCompose(time ->
                connection.async().evalsha(sha, VALUE, new String[]{key}, rulesJson, Long.toString(time), Integer.toString(weight), strictlyGreater ? "1" : "0")
        ).thenApply(result -> {
            boolean overLimit = "1".equals(result);
            if (overLimit) {
                LOG.debug("Requests matched by key '{}' incremented by weight {} are greater than {} the limit", key, weight, strictlyGreater ? "" : "or equal to ");
            }
            return overLimit;
        });

        // TODO handle scenario where script is not loaded, flush scripts and test scenario
    }

    private boolean toBlocking(CompletionStage completionStage) {
        try {
            return completionStage.toCompletableFuture().get(10, TimeUnit.SECONDS);
        } catch (Exception e) {
            throw new RuntimeException("Failed to complete operation", e);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy