Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.redisson.RedissonTransferQueue 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
/**
* 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.buffer.ByteBuf;
import io.netty.util.Timeout;
import io.netty.util.concurrent.ImmediateEventExecutor;
import org.redisson.api.RFuture;
import org.redisson.api.RRemoteService;
import org.redisson.api.RTransferQueue;
import org.redisson.api.RemoteInvocationOptions;
import org.redisson.api.annotation.RRemoteAsync;
import org.redisson.client.codec.Codec;
import org.redisson.client.protocol.RedisCommand;
import org.redisson.client.protocol.RedisStrictCommand;
import org.redisson.client.protocol.convertor.Convertor;
import org.redisson.client.protocol.decoder.ObjectListReplayDecoder;
import org.redisson.command.CommandAsyncExecutor;
import org.redisson.connection.decoder.ListDrainToDecoder;
import org.redisson.executor.RemotePromise;
import org.redisson.iterator.RedissonListIterator;
import org.redisson.misc.CompletableFutureWrapper;
import org.redisson.misc.RPromise;
import org.redisson.misc.RedissonPromise;
import org.redisson.remote.RemoteServiceRequest;
import java.util.*;
import java.util.concurrent.*;
import java.util.function.Consumer;
/**
*
* @author Nikita Koksharov
*
*/
public class RedissonTransferQueue extends RedissonExpirable implements RTransferQueue {
public interface TransferQueueService {
void invoke(V value);
}
@RRemoteAsync(TransferQueueService.class)
public interface TransferQueueServiceAsync {
RFuture invoke(V value);
}
public static class TransferQueueServiceImpl implements TransferQueueService {
private Object result;
@Override
public void invoke(V value) {
result = value;
}
public Object getResult() {
return result;
}
}
private static final Convertor CONVERTER = obj -> {
if (obj != null) {
RemoteServiceRequest request = (RemoteServiceRequest) obj;
return request.getArgs()[0];
}
return null;
};
private static final RedisStrictCommand EVAL_REQUEST = new RedisStrictCommand<>("EVAL", CONVERTER);
private static final RedisCommand EVAL_LIST = new RedisCommand("EVAL", new ObjectListReplayDecoder<>(), CONVERTER);
private final String queueName;
private final String mapName;
private final TransferQueueServiceAsync service;
private final RRemoteService remoteService;
public RedissonTransferQueue(Codec codec, CommandAsyncExecutor commandExecutor, String name, RRemoteService remoteService) {
super(codec, commandExecutor, name);
service = remoteService.get(TransferQueueServiceAsync.class, RemoteInvocationOptions.defaults().noAck());
this.remoteService = remoteService;
queueName = ((RedissonRemoteService) remoteService).getRequestQueueName(TransferQueueService.class);
mapName = ((RedissonRemoteService) remoteService).getRequestTasksMapName(TransferQueueService.class);
}
public RedissonTransferQueue(CommandAsyncExecutor commandExecutor, String name, RRemoteService remoteService) {
super(commandExecutor, name);
service = remoteService.get(TransferQueueServiceAsync.class, RemoteInvocationOptions.defaults().noAck());
this.remoteService = remoteService;
queueName = ((RedissonRemoteService) remoteService).getRequestQueueName(TransferQueueService.class);
mapName = ((RedissonRemoteService) remoteService).getRequestTasksMapName(TransferQueueService.class);
}
@Override
public boolean tryTransfer(V v) {
RemotePromise future = (RemotePromise) service.invoke(v).toCompletableFuture();
boolean added = commandExecutor.get(future.getAddFuture());
if (added && !future.cancel(false)) {
commandExecutor.get(future);
return true;
}
return false;
}
public RFuture tryTransferAsync(V v) {
RemotePromise future = (RemotePromise) service.invoke(v).toCompletableFuture();
CompletableFuture result = future.getAddFuture().thenCompose(added -> {
if (!added) {
return CompletableFuture.completedFuture(false);
}
return future.cancelAsync(false).thenCompose(canceled -> {
if (canceled) {
return CompletableFuture.completedFuture(false);
} else {
return future.thenApply(res -> true);
}
});
});
return new CompletableFutureWrapper<>(result);
}
@Override
public void transfer(V v) throws InterruptedException {
RFuture future = service.invoke(v);
commandExecutor.getInterrupted(future);
}
@Override
public RFuture transferAsync(V v) {
return service.invoke(v);
}
@Override
public boolean tryTransfer(V v, long timeout, TimeUnit unit) throws InterruptedException {
RemotePromise future = (RemotePromise) service.invoke(v).toCompletableFuture();
long remainTime = unit.toMillis(timeout);
long startTime = System.currentTimeMillis();
CompletableFuture addFuture = future.getAddFuture().toCompletableFuture();
try {
addFuture.get(remainTime, TimeUnit.MILLISECONDS);
} catch (ExecutionException | TimeoutException e) {
if (!addFuture.cancel(false)) {
if (!future.cancel(false)) {
commandExecutor.getInterrupted(future);
return true;
}
}
return false;
}
remainTime -= System.currentTimeMillis() - startTime;
try {
future.toCompletableFuture().get(remainTime, TimeUnit.MILLISECONDS);
} catch (ExecutionException | TimeoutException e) {
if (!future.cancel(false)) {
commandExecutor.getInterrupted(future);
return true;
}
return false;
}
commandExecutor.getInterrupted(future);
return true;
}
public RFuture tryTransferAsync(V v, long timeout, TimeUnit unit) {
RPromise result = new RedissonPromise<>();
result.setUncancellable();
RemotePromise future = (RemotePromise) service.invoke(v).toCompletableFuture();
long remainTime = unit.toMillis(timeout);
long startTime = System.currentTimeMillis();
Timeout timeoutFuture = commandExecutor.getConnectionManager().newTimeout(tt -> {
if (!future.getAddFuture().cancel(false)) {
future.cancelAsync(false);
}
}, remainTime, TimeUnit.MILLISECONDS);
future.whenComplete((res, exc) -> {
if (future.isCancelled()) {
result.trySuccess(false);
return;
}
timeoutFuture.cancel();
if (exc != null) {
result.tryFailure(exc);
return;
}
result.trySuccess(true);
});
future.getAddFuture().whenComplete((added, e) -> {
if (future.getAddFuture().isCancelled()) {
result.trySuccess(false);
return;
}
if (e != null) {
timeoutFuture.cancel();
result.tryFailure(e);
return;
}
if (!added) {
timeoutFuture.cancel();
result.trySuccess(false);
return;
}
Runnable task = () -> {
future.cancelAsync(false).whenComplete((canceled, ex) -> {
if (ex != null) {
timeoutFuture.cancel();
result.tryFailure(ex);
return;
}
if (canceled) {
timeoutFuture.cancel();
result.trySuccess(false);
}
});
};
long time = remainTime - (System.currentTimeMillis() - startTime);
if (time > 0) {
commandExecutor.getConnectionManager().newTimeout(tt -> {
task.run();
}, time, TimeUnit.MILLISECONDS);
} else {
task.run();
}
});
return result;
}
@Override
public boolean hasWaitingConsumer() {
throw new UnsupportedOperationException();
}
@Override
public int getWaitingConsumerCount() {
throw new UnsupportedOperationException();
}
@Override
public boolean add(V v) {
RemotePromise future = (RemotePromise) service.invoke(v).toCompletableFuture();
return commandExecutor.get(future.getAddFuture());
}
public RFuture addAsync(V v) {
RemotePromise future = (RemotePromise) service.invoke(v).toCompletableFuture();
return new CompletableFutureWrapper<>(future.getAddFuture());
}
@Override
public boolean offer(V v) {
return add(v);
}
@Override
public V remove() {
V value = poll();
if (value == null) {
throw new NoSuchElementException();
}
return value;
}
@Override
public V poll() {
TransferQueueServiceImpl s = new TransferQueueServiceImpl();
RFuture r = remoteService.tryExecuteAsync(TransferQueueService.class, s, ImmediateEventExecutor.INSTANCE, -1, null);
get(r);
return (V) s.getResult();
}
public RFuture pollAsync() {
TransferQueueServiceImpl s = new TransferQueueServiceImpl();
RFuture future = remoteService.tryExecuteAsync(TransferQueueService.class, s, ImmediateEventExecutor.INSTANCE, -1, null);
CompletionStage f = future.thenApply(r -> (V) s.getResult());
return new CompletableFutureWrapper<>(f);
}
@Override
public V element() {
V value = peek();
if (value == null) {
throw new NoSuchElementException();
}
return value;
}
@Override
public V peek() {
return get(peekAsync());
}
public RFuture peekAsync() {
return commandExecutor.evalReadAsync(queueName, codec, EVAL_REQUEST,
"local id = redis.call('lindex', KEYS[1], 0); "
+ "if id ~= false then "
+ "return redis.call('hget', KEYS[2], id); "
+ "end "
+ "return nil;",
Arrays.asList(queueName, mapName));
}
@Override
public void put(V v) throws InterruptedException {
add(v);
}
@Override
public boolean offer(V v, long timeout, TimeUnit unit) throws InterruptedException {
return add(v);
}
@Override
public V take() throws InterruptedException {
return poll(0, TimeUnit.MILLISECONDS);
}
public RFuture takeAsync() {
return pollAsync(0, TimeUnit.MILLISECONDS);
}
@Override
public V poll(long timeout, TimeUnit unit) throws InterruptedException {
TransferQueueServiceImpl s = new TransferQueueServiceImpl();
remoteService.tryExecute(TransferQueueService.class, s, ImmediateEventExecutor.INSTANCE, timeout, unit);
return (V) s.getResult();
}
public RFuture pollAsync(long timeout, TimeUnit unit) {
TransferQueueServiceImpl s = new TransferQueueServiceImpl();
RFuture future = remoteService.tryExecuteAsync(TransferQueueService.class, s, ImmediateEventExecutor.INSTANCE, timeout, unit);
CompletionStage f = future.thenApply(r -> (V) s.getResult());
return new CompletableFutureWrapper<>(f);
}
@Override
public int remainingCapacity() {
return Integer.MAX_VALUE;
}
@Override
public boolean remove(Object o) {
throw new UnsupportedOperationException();
}
@Override
public boolean containsAll(Collection> c) {
if (c.isEmpty()) {
return true;
}
boolean all = true;
for (Object obj : c) {
all &= contains(obj);
}
return all;
}
@Override
public boolean addAll(Collection extends V> c) {
if (c.isEmpty()) {
return false;
}
boolean added = false;
for (V obj : c) {
added |= add(obj);
}
return added;
}
@Override
public boolean removeAll(Collection> c) {
throw new UnsupportedOperationException();
}
@Override
public boolean retainAll(Collection> c) {
throw new UnsupportedOperationException();
}
@Override
public void clear() {
RedissonKeys keys = new RedissonKeys(commandExecutor);
keys.delete(queueName, mapName);
}
public RFuture clearAsync() {
RedissonKeys keys = new RedissonKeys(commandExecutor);
CompletionStage f = keys.deleteAsync(queueName, mapName).thenApply(r -> null);
return new CompletableFutureWrapper<>(f);
}
@Override
public int size() {
return remoteService.getPendingInvocations(TransferQueueService.class);
}
public RFuture sizeAsync() {
return remoteService.getPendingInvocationsAsync(TransferQueueService.class);
}
@Override
public boolean isEmpty() {
return size() == 0;
}
@Override
public boolean contains(Object o) {
ByteBuf encodedObject = encode(o);
boolean result = stream().anyMatch(v -> {
ByteBuf encodedValue = encode(v);
boolean res = encodedValue.equals(encodedObject);
encodedValue.release();
return res;
});
encodedObject.release();
return result;
}
public RFuture getValueAsync(int index) {
return commandExecutor.evalReadAsync(queueName, codec, EVAL_REQUEST,
"local id = redis.call('lindex', KEYS[1], ARGV[1]); "
+ "if id ~= false then "
+ "return redis.call('hget', KEYS[2], id); "
+ "end "
+ "return nil;",
Arrays.asList(queueName, mapName), index);
}
@Override
public Iterator iterator() {
return new RedissonListIterator(0) {
@Override
public V getValue(int index) {
RFuture future = getValueAsync(index);
return get(future);
}
@Override
public V remove(int index) {
if (index == 0) {
RFuture future = commandExecutor.evalWriteNoRetryAsync(queueName, codec, EVAL_REQUEST,
"local id = redis.call('lpop', KEYS[1]); "
+ "if id ~= false then "
+ "return redis.call('hget', KEYS[2], id); "
+ "end "
+ "return nil;",
Arrays.asList(queueName, mapName));
return get(future);
}
RFuture future = commandExecutor.evalWriteAsync(queueName, codec, EVAL_REQUEST,
"local id = redis.call('lindex', KEYS[1], ARGV[1]); " +
"if id ~= false then " +
"redis.call('lset', KEYS[1], ARGV[1], 'DELETED_BY_REDISSON');" +
"redis.call('lrem', KEYS[1], 1, 'DELETED_BY_REDISSON');" +
"local val = redis.call('hget', KEYS[2], id); " +
"redis.call('hdel', KEYS[2], id); " +
"return val; " +
"end; " +
"return nil;",
Arrays.asList(queueName, mapName), index);
return get(future);
}
@Override
public void fastSet(int index, V value) {
throw new UnsupportedOperationException();
}
@Override
public void add(int index, V value) {
throw new UnsupportedOperationException();
}
};
}
@Override
public Object[] toArray() {
List list = readAll();
return list.toArray();
}
@Override
public T[] toArray(T[] a) {
List list = readAll();
return list.toArray(a);
}
@Override
public int drainTo(Collection super V> c) {
return get(drainToAsync(c));
}
public RFuture drainToAsync(Collection super V> c) {
if (c == null) {
throw new NullPointerException();
}
return commandExecutor.evalWriteAsync(getRawName(), codec, new RedisCommand("EVAL", new ListDrainToDecoder(c), CONVERTER),
"local ids = redis.call('lrange', KEYS[1], 0, -1); " +
"local result = {};"
+ "for i=1, #ids, 5000 do "
+ "local vals = redis.call('hmget', KEYS[2], unpack(ids, i, math.min(i+4999, #ids))); "
+ "for k,v in ipairs(vals) do "
+ "table.insert(result, v); "
+ "end; "
+ "end; " +
"redis.call('del', KEYS[1], KEYS[2]); " +
"return result",
Arrays.asList(queueName, mapName));
}
@Override
public int drainTo(Collection super V> c, int maxElements) {
if (maxElements <= 0) {
return 0;
}
return get(drainToAsync(c, maxElements));
}
public RFuture drainToAsync(Collection super V> c, int maxElements) {
if (c == null) {
throw new NullPointerException();
}
return commandExecutor.evalWriteAsync(getRawName(), codec, new RedisCommand("EVAL", new ListDrainToDecoder(c), CONVERTER),
"local elemNum = math.min(ARGV[1], redis.call('llen', KEYS[1])) - 1;" +
"local ids = redis.call('lrange', KEYS[1], 0, elemNum); " +
"redis.call('ltrim', KEYS[1], elemNum + 1, -1); " +
"local result = {};"
+ "for i=1, #ids, 5000 do "
+ "local vals = redis.call('hmget', KEYS[2], unpack(ids, i, math.min(i+4999, #ids))); "
+ "redis.call('hdel', KEYS[2], unpack(ids, i, math.min(i+4999, #ids)));"
+ "for k,v in ipairs(vals) do "
+ "table.insert(result, v); "
+ "end; "
+ "end; " +
"return result",
Arrays.asList(queueName, mapName), maxElements);
}
@Override
public List readAll() {
return get(readAllAsync());
}
public RFuture> readAllAsync() {
return commandExecutor.evalReadAsync(getRawName(), codec, EVAL_LIST,
"local ids = redis.call('lrange', KEYS[1], 0, -1); " +
"local result = {};"
+ "for i=1, #ids, 5000 do "
+ "local vals = redis.call('hmget', KEYS[2], unpack(ids, i, math.min(i+4999, #ids))); "
+ "for k,v in ipairs(vals) do "
+ "table.insert(result, v); "
+ "end; "
+ "end; " +
"return result;",
Arrays.asList(queueName, mapName));
}
@Override
public V pollFromAny(long timeout, TimeUnit unit, String... queueNames) throws InterruptedException {
throw new UnsupportedOperationException();
}
@Override
public V pollLastAndOfferFirstTo(String queueName, long timeout, TimeUnit unit) throws InterruptedException {
throw new UnsupportedOperationException();
}
@Override
public V takeLastAndOfferFirstTo(String queueName) throws InterruptedException {
throw new UnsupportedOperationException();
}
@Override
public int subscribeOnElements(Consumer consumer) {
return commandExecutor.getConnectionManager().getElementsSubscribeService().subscribeOnElements(this::takeAsync, consumer);
}
@Override
public void unsubscribe(int listenerId) {
commandExecutor.getConnectionManager().getElementsSubscribeService().unsubscribe(listenerId);
}
@Override
public V pollLastAndOfferFirstTo(String queueName) {
throw new UnsupportedOperationException();
}
@Override
public List poll(int limit) {
throw new UnsupportedOperationException();
}
@Override
public RFuture pollFromAnyAsync(long timeout, TimeUnit unit, String... queueNames) {
throw new UnsupportedOperationException();
}
@Override
public RFuture pollLastAndOfferFirstToAsync(String queueName, long timeout, TimeUnit unit) {
throw new UnsupportedOperationException();
}
@Override
public RFuture takeLastAndOfferFirstToAsync(String queueName) {
throw new UnsupportedOperationException();
}
@Override
public RFuture putAsync(V value) {
RemotePromise future = (RemotePromise) service.invoke(value).toCompletableFuture();
CompletableFuture f = future.getAddFuture().thenApply(r -> null);
return new CompletableFutureWrapper<>(f);
}
@Override
public RFuture offerAsync(V e) {
return addAsync(e);
}
@Override
public RFuture pollLastAndOfferFirstToAsync(String queueName) {
throw new UnsupportedOperationException();
}
@Override
public RFuture> pollAsync(int limit) {
throw new UnsupportedOperationException();
}
@Override
public RFuture retainAllAsync(Collection> c) {
throw new UnsupportedOperationException();
}
@Override
public RFuture removeAllAsync(Collection> c) {
throw new UnsupportedOperationException();
}
@Override
public RFuture containsAsync(Object o) {
throw new UnsupportedOperationException();
}
@Override
public RFuture containsAllAsync(Collection> c) {
throw new UnsupportedOperationException();
}
@Override
public RFuture removeAsync(Object o) {
throw new UnsupportedOperationException();
}
@Override
public RFuture addAllAsync(Collection extends V> c) {
throw new UnsupportedOperationException();
}
}