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

org.redisson.RedissonSemaphore Maven / Gradle / Ivy

Go to download

Easy Redis Java client and Real-Time Data Platform. Valkey compatible. Sync/Async/RxJava3/Reactive API. Client side caching. Over 50 Redis based Java objects and services: JCache API, Apache Tomcat, Hibernate, Spring, Set, Multimap, SortedSet, Map, List, Queue, Deque, Semaphore, Lock, AtomicLong, Map Reduce, Bloom filter, Scheduler, RPC

There is a newer version: 3.40.2
Show newest version
/**
 * Copyright (c) 2013-2021 Nikita Koksharov
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.redisson;

import io.netty.util.Timeout;
import io.netty.util.TimerTask;
import org.redisson.api.RFuture;
import org.redisson.api.RSemaphore;
import org.redisson.client.codec.LongCodec;
import org.redisson.client.codec.StringCodec;
import org.redisson.client.protocol.RedisCommands;
import org.redisson.command.CommandAsyncExecutor;
import org.redisson.misc.RPromise;
import org.redisson.misc.RedissonPromise;
import org.redisson.pubsub.SemaphorePubSub;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.Arrays;
import java.util.Collections;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference;

/**
 * Distributed and concurrent implementation of {@link java.util.concurrent.Semaphore}.
 * 

* Works in non-fair mode. Therefore order of acquiring is unpredictable. * * @author Nikita Koksharov * */ public class RedissonSemaphore extends RedissonExpirable implements RSemaphore { private static final Logger log = LoggerFactory.getLogger(RSemaphore.class); private final SemaphorePubSub semaphorePubSub; final CommandAsyncExecutor commandExecutor; public RedissonSemaphore(CommandAsyncExecutor commandExecutor, String name) { super(commandExecutor, name); this.commandExecutor = commandExecutor; this.semaphorePubSub = commandExecutor.getConnectionManager().getSubscribeService().getSemaphorePubSub(); } String getChannelName() { return getChannelName(getRawName()); } public static String getChannelName(String name) { if (name.contains("{")) { return "redisson_sc:" + name; } return "redisson_sc:{" + name + "}"; } @Override public void acquire() throws InterruptedException { acquire(1); } @Override public void acquire(int permits) throws InterruptedException { if (tryAcquire(permits)) { return; } CompletableFuture future = subscribe(); commandExecutor.syncSubscriptionInterrupted(future); try { while (true) { if (tryAcquire(permits)) { return; } commandExecutor.getNow(future).getLatch().acquire(); } } finally { unsubscribe(commandExecutor.getNow(future)); } // get(acquireAsync(permits)); } @Override public RFuture acquireAsync() { return acquireAsync(1); } @Override public RFuture acquireAsync(int permits) { RPromise result = new RedissonPromise(); RFuture tryAcquireFuture = tryAcquireAsync(permits); tryAcquireFuture.onComplete((res, e) -> { if (e != null) { result.tryFailure(e); return; } if (res) { if (!result.trySuccess(null)) { releaseAsync(permits); } return; } CompletableFuture subscribeFuture = subscribe(); subscribeFuture.whenComplete((r, e1) -> { if (e1 != null) { result.tryFailure(e1); return; } acquireAsync(permits, r, result); }); }); return result; } private void tryAcquireAsync(AtomicLong time, int permits, RedissonLockEntry entry, RPromise result) { if (result.isDone()) { unsubscribe(entry); return; } if (time.get() <= 0) { unsubscribe(entry); result.trySuccess(false); return; } long curr = System.currentTimeMillis(); RFuture tryAcquireFuture = tryAcquireAsync(permits); tryAcquireFuture.onComplete((res, e) -> { if (e != null) { unsubscribe(entry); result.tryFailure(e); return; } if (res) { unsubscribe(entry); if (!result.trySuccess(true)) { releaseAsync(permits); } return; } long el = System.currentTimeMillis() - curr; time.addAndGet(-el); if (time.get() <= 0) { unsubscribe(entry); result.trySuccess(false); return; } // waiting for message long current = System.currentTimeMillis(); if (entry.getLatch().tryAcquire()) { tryAcquireAsync(time, permits, entry, result); } else { AtomicBoolean executed = new AtomicBoolean(); AtomicReference futureRef = new AtomicReference(); Runnable listener = () -> { executed.set(true); if (futureRef.get() != null && !futureRef.get().cancel()) { entry.getLatch().release(); return; } long elapsed = System.currentTimeMillis() - current; time.addAndGet(-elapsed); tryAcquireAsync(time, permits, entry, result); }; entry.addListener(listener); long t = time.get(); if (!executed.get()) { Timeout scheduledFuture = commandExecutor.getConnectionManager().newTimeout(new TimerTask() { @Override public void run(Timeout timeout) throws Exception { if (entry.removeListener(listener)) { long elapsed = System.currentTimeMillis() - current; time.addAndGet(-elapsed); tryAcquireAsync(time, permits, entry, result); } } }, t, TimeUnit.MILLISECONDS); futureRef.set(scheduledFuture); } } }); } private void acquireAsync(int permits, RedissonLockEntry entry, RPromise result) { if (result.isDone()) { unsubscribe(entry); return; } RFuture tryAcquireFuture = tryAcquireAsync(permits); tryAcquireFuture.onComplete((res, e) -> { if (e != null) { unsubscribe(entry); result.tryFailure(e); return; } if (res) { unsubscribe(entry); if (!result.trySuccess(null)) { releaseAsync(permits); } return; } if (entry.getLatch().tryAcquire()) { acquireAsync(permits, entry, result); } else { entry.addListener(() -> { acquireAsync(permits, entry, result); }); } }); } @Override public boolean tryAcquire() { return tryAcquire(1); } @Override public boolean tryAcquire(int permits) { return get(tryAcquireAsync(permits)); } @Override public RFuture tryAcquireAsync() { return tryAcquireAsync(1); } @Override public RFuture tryAcquireAsync(int permits) { if (permits < 0) { throw new IllegalArgumentException("Permits amount can't be negative"); } if (permits == 0) { return RedissonPromise.newSucceededFuture(true); } return commandExecutor.evalWriteAsync(getRawName(), LongCodec.INSTANCE, RedisCommands.EVAL_BOOLEAN, "local value = redis.call('get', KEYS[1]); " + "if (value ~= false and tonumber(value) >= tonumber(ARGV[1])) then " + "local val = redis.call('decrby', KEYS[1], ARGV[1]); " + "return 1; " + "end; " + "return 0;", Collections.singletonList(getRawName()), permits); } @Override public RFuture tryAcquireAsync(long waitTime, TimeUnit unit) { return tryAcquireAsync(1, waitTime, unit); } @Override public boolean tryAcquire(int permits, long waitTime, TimeUnit unit) throws InterruptedException { log.debug("trying to acquire, permits: {}, waitTime: {}, unit: {}, name: {}", permits, waitTime, unit, getName()); long time = unit.toMillis(waitTime); long current = System.currentTimeMillis(); if (tryAcquire(permits)) { log.debug("acquired, permits: {}, waitTime: {}, unit: {}, name: {}", permits, waitTime, unit, getName()); return true; } time -= System.currentTimeMillis() - current; if (time <= 0) { log.debug("unable to acquire, permits: {}, name: {}", permits, getName()); return false; } current = System.currentTimeMillis(); CompletableFuture future = subscribe(); RedissonLockEntry entry; try { entry = future.get(time, TimeUnit.MILLISECONDS); } catch (ExecutionException | CancellationException | TimeoutException e) { log.debug("unable to subscribe for permits acquisition, permits: {}, name: {}", permits, getName()); return false; } try { time -= System.currentTimeMillis() - current; if (time <= 0) { log.debug("unable to acquire, permits: {}, name: {}", permits, getName()); return false; } while (true) { current = System.currentTimeMillis(); if (tryAcquire(permits)) { log.debug("acquired, permits: {}, wait-time: {}, unit: {}, name: {}", permits, waitTime, unit, getName()); return true; } time -= System.currentTimeMillis() - current; if (time <= 0) { log.debug("unable to acquire, permits: {}, name: {}", permits, getName()); return false; } // waiting for message current = System.currentTimeMillis(); log.debug("wait for acquisition, permits: {}, wait-time(ms): {}, name: {}", permits, time, getName()); entry.getLatch().tryAcquire(time, TimeUnit.MILLISECONDS); time -= System.currentTimeMillis() - current; if (time <= 0) { log.debug("unable to acquire, permits: {}, name: {}", permits, getName()); return false; } } } finally { unsubscribe(entry); } // return get(tryAcquireAsync(permits, waitTime, unit)); } @Override public RFuture tryAcquireAsync(int permits, long waitTime, TimeUnit unit) { RPromise result = new RedissonPromise(); AtomicLong time = new AtomicLong(unit.toMillis(waitTime)); long curr = System.currentTimeMillis(); RFuture tryAcquireFuture = tryAcquireAsync(permits); tryAcquireFuture.onComplete((res, e) -> { if (e != null) { result.tryFailure(e); return; } if (res) { if (!result.trySuccess(true)) { releaseAsync(permits); } return; } long elap = System.currentTimeMillis() - curr; time.addAndGet(-elap); if (time.get() <= 0) { result.trySuccess(false); return; } long current = System.currentTimeMillis(); AtomicReference futureRef = new AtomicReference(); CompletableFuture subscribeFuture = subscribe(); subscribeFuture.whenComplete((r, ex) -> { if (ex != null) { result.tryFailure(ex); return; } if (futureRef.get() != null) { futureRef.get().cancel(); } long elapsed = System.currentTimeMillis() - current; time.addAndGet(-elapsed); if (time.get() < 0) { unsubscribe(r); result.trySuccess(false); return; } tryAcquireAsync(time, permits, r, result); }); if (!subscribeFuture.isDone()) { Timeout scheduledFuture = commandExecutor.getConnectionManager().newTimeout(new TimerTask() { @Override public void run(Timeout timeout) throws Exception { if (!subscribeFuture.isDone()) { result.trySuccess(false); } } }, time.get(), TimeUnit.MILLISECONDS); futureRef.set(scheduledFuture); } }); return result; } private CompletableFuture subscribe() { return semaphorePubSub.subscribe(getRawName(), getChannelName()); } private void unsubscribe(RedissonLockEntry entry) { semaphorePubSub.unsubscribe(entry, getRawName(), getChannelName()); } @Override public boolean tryAcquire(long time, TimeUnit unit) throws InterruptedException { return tryAcquire(1, time, unit); } @Override public void release() { release(1); } @Override public void release(int permits) { get(releaseAsync(permits)); } @Override public RFuture releaseAsync() { return releaseAsync(1); } @Override public RFuture releaseAsync(int permits) { if (permits < 0) { throw new IllegalArgumentException("Permits amount can't be negative"); } if (permits == 0) { return RedissonPromise.newSucceededFuture(null); } RFuture future = commandExecutor.evalWriteAsync(getRawName(), StringCodec.INSTANCE, RedisCommands.EVAL_VOID, "local value = redis.call('incrby', KEYS[1], ARGV[1]); " + "redis.call('publish', KEYS[2], value); ", Arrays.asList(getRawName(), getChannelName()), permits); if (log.isDebugEnabled()) { future.onComplete((o, e) -> { if (e == null) { log.debug("released, permits: {}, name: {}", permits, getName()); } }); } return future; } @Override public int drainPermits() { return get(drainPermitsAsync()); } @Override public RFuture drainPermitsAsync() { return commandExecutor.evalWriteAsync(getRawName(), LongCodec.INSTANCE, RedisCommands.EVAL_INTEGER, "local value = redis.call('get', KEYS[1]); " + "if (value == false) then " + "return 0; " + "end; " + "redis.call('set', KEYS[1], 0); " + "return value;", Collections.singletonList(getRawName())); } @Override public int availablePermits() { return get(availablePermitsAsync()); } @Override public RFuture availablePermitsAsync() { return commandExecutor.writeAsync(getRawName(), LongCodec.INSTANCE, RedisCommands.GET_INTEGER, getRawName()); } @Override public boolean trySetPermits(int permits) { return get(trySetPermitsAsync(permits)); } @Override public RFuture trySetPermitsAsync(int permits) { RFuture future = commandExecutor.evalWriteAsync(getRawName(), LongCodec.INSTANCE, RedisCommands.EVAL_BOOLEAN, "local value = redis.call('get', KEYS[1]); " + "if (value == false) then " + "redis.call('set', KEYS[1], ARGV[1]); " + "redis.call('publish', KEYS[2], ARGV[1]); " + "return 1;" + "end;" + "return 0;", Arrays.asList(getRawName(), getChannelName()), permits); if (log.isDebugEnabled()) { future.onComplete((r, e) -> { if (r) { log.debug("permits set, permits: {}, name: {}", permits, getName()); } else { log.debug("unable to set permits, permits: {}, name: {}", permits, getName()); } }); } return future; } @Override public void addPermits(int permits) { get(addPermitsAsync(permits)); } @Override public RFuture addPermitsAsync(int permits) { return commandExecutor.evalWriteAsync(getRawName(), LongCodec.INSTANCE, RedisCommands.EVAL_VOID, "local value = redis.call('get', KEYS[1]); " + "if (value == false) then " + "value = 0;" + "end;" + "redis.call('set', KEYS[1], value + ARGV[1]); " + "redis.call('publish', KEYS[2], value + ARGV[1]); ", Arrays.asList(getRawName(), getChannelName()), permits); } }