
com.lambdaworks.redis.StatefulRedisConnectionImpl Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of lettuce Show documentation
Show all versions of lettuce Show documentation
Advanced and thread-safe Java Redis client for synchronous, asynchronous, and
reactive usage. Supports Cluster, Sentinel, Pipelining, Auto-Reconnect, Codecs
and much more.
The newest version!
/*
* Copyright 2011-2016 the original author or authors.
*
* 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 com.lambdaworks.redis;
import static com.lambdaworks.redis.protocol.CommandType.AUTH;
import static com.lambdaworks.redis.protocol.CommandType.DISCARD;
import static com.lambdaworks.redis.protocol.CommandType.EXEC;
import static com.lambdaworks.redis.protocol.CommandType.MULTI;
import static com.lambdaworks.redis.protocol.CommandType.READONLY;
import static com.lambdaworks.redis.protocol.CommandType.READWRITE;
import static com.lambdaworks.redis.protocol.CommandType.SELECT;
import java.util.concurrent.TimeUnit;
import java.util.function.Consumer;
import com.lambdaworks.redis.api.StatefulRedisConnection;
import com.lambdaworks.redis.api.async.RedisAsyncCommands;
import com.lambdaworks.redis.api.reactive.RedisReactiveCommands;
import com.lambdaworks.redis.api.sync.RedisCommands;
import com.lambdaworks.redis.cluster.api.sync.RedisClusterCommands;
import com.lambdaworks.redis.codec.RedisCodec;
import com.lambdaworks.redis.output.MultiOutput;
import com.lambdaworks.redis.protocol.CompleteableCommand;
import com.lambdaworks.redis.protocol.ConnectionWatchdog;
import com.lambdaworks.redis.protocol.RedisCommand;
import com.lambdaworks.redis.protocol.TransactionalCommand;
import io.netty.channel.ChannelHandler;
/**
* A thread-safe connection to a Redis server. Multiple threads may share one {@link StatefulRedisConnectionImpl}
*
* A {@link ConnectionWatchdog} monitors each connection and reconnects automatically until {@link #close} is called. All
* pending commands will be (re)sent after successful reconnection.
*
* @param Key type.
* @param Value type.
* @author Mark Paluch
*/
public class StatefulRedisConnectionImpl extends RedisChannelHandler implements StatefulRedisConnection {
protected final RedisCodec codec;
protected final RedisCommands sync;
protected final RedisAsyncCommandsImpl async;
protected final RedisReactiveCommandsImpl reactive;
protected MultiOutput multi;
private char[] password;
private int db;
private boolean readOnly;
/**
* Initialize a new connection.
*
* @param writer the channel writer
* @param codec Codec used to encode/decode keys and values.
* @param timeout Maximum time to wait for a response.
* @param unit Unit of time for the timeout.
*/
public StatefulRedisConnectionImpl(RedisChannelWriter writer, RedisCodec codec, long timeout, TimeUnit unit) {
super(writer, timeout, unit);
this.codec = codec;
this.async = newRedisAsyncCommandsImpl();
this.sync = newRedisSyncCommandsImpl();
this.reactive = newRedisReactiveCommandsImpl();
}
@Override
public RedisAsyncCommands async() {
return async;
}
/**
* Create a new instance of {@link RedisCommands}. Can be overriden to extend.
*
* @return a new instance
*/
protected RedisCommands newRedisSyncCommandsImpl() {
return syncHandler(async(), RedisCommands.class, RedisClusterCommands.class);
}
/**
* Create a new instance of {@link RedisAsyncCommandsImpl}. Can be overriden to extend.
*
* @return a new instance
*/
protected RedisAsyncCommandsImpl newRedisAsyncCommandsImpl() {
return new RedisAsyncCommandsImpl<>(this, codec);
}
@Override
public RedisReactiveCommands reactive() {
return reactive;
}
/**
* Create a new instance of {@link RedisReactiveCommandsImpl}. Can be overriden to extend.
*
* @return a new instance
*/
protected RedisReactiveCommandsImpl newRedisReactiveCommandsImpl() {
return new RedisReactiveCommandsImpl<>(this, codec);
}
@Override
public RedisCommands sync() {
return sync;
}
@Override
public boolean isMulti() {
return multi != null;
}
@Override
public void activated() {
super.activated();
// do not block in here, since the channel flow will be interrupted.
if (password != null) {
async.authAsync(new String(password));
}
if (db != 0) {
async.selectAsync(db);
}
if (readOnly) {
async.readOnly();
}
}
@Override
public > C dispatch(C cmd) {
RedisCommand local = cmd;
if (local.getType().name().equals(AUTH.name())) {
local = attachOnComplete(local, status -> {
if ("OK".equals(status) && cmd.getArgs().getFirstString() != null) {
this.password = cmd.getArgs().getFirstString().toCharArray();
}
});
}
if (local.getType().name().equals(SELECT.name())) {
local = attachOnComplete(local, status -> {
if ("OK".equals(status) && cmd.getArgs().getFirstInteger() != null) {
this.db = cmd.getArgs().getFirstInteger().intValue();
}
});
}
if (local.getType().name().equals(READONLY.name())) {
local = attachOnComplete(local, status -> {
if ("OK".equals(status)) {
this.readOnly = true;
}
});
}
if (local.getType().name().equals(READWRITE.name())) {
local = attachOnComplete(local, status -> {
if ("OK".equals(status)) {
this.readOnly = false;
}
});
}
if (local.getType().name().equals(DISCARD.name())) {
if (multi != null) {
multi.cancel();
multi = null;
}
}
if (local.getType().name().equals(EXEC.name())) {
MultiOutput multiOutput = this.multi;
this.multi = null;
if (multiOutput == null) {
multiOutput = new MultiOutput<>(codec);
}
local.setOutput((MultiOutput) multiOutput);
}
if (multi != null) {
local = new TransactionalCommand<>(local);
multi.add(local);
}
try {
return (C) super.dispatch(local);
} finally {
if (cmd.getType().name().equals(MULTI.name())) {
multi = (multi == null ? new MultiOutput<>(codec) : multi);
}
}
}
private RedisCommand attachOnComplete(RedisCommand command, Consumer consumer) {
if (command instanceof CompleteableCommand) {
CompleteableCommand completeable = (CompleteableCommand) command;
completeable.onComplete(consumer);
}
return command;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy