Различные исключения подключения в Redis

Я использую Redis в режиме сервера, а не в режиме кластера. У меня есть 3 запущенных экземпляра Redis Server, 3 подчиненных устройства, соответствующих этим трем мастерам. У меня также есть 3 Sentinel, каждый из которых следит за всеми тремя мастерами.

Чтобы подключиться к моим Sentinels и создать пул, я использую ShardedJedisSentinelPool. Теперь мой класс для подключения к этому пулу выглядит так

public class RedisServerConnectionDAL 
{
    private RedisServerConnectionDAL()
    {

    }

    private static ShardedJedisSentinelPool pool;
    private static ShardedJedis jedis;
    private static ShardedJedisPipeline pipeline;
    private static Set<String> redissentinels;
    private static List<String> redismasters;
    private static int redistimeout;
    private static RedisServerConnectionDAL redisdal;
    private static Map<String,String> redisconfmap;
    public static int WAIT_IF_FAIL;
    public static int RETRIES;

    public synchronized static RedisServerConnectionDAL getInstance()
    {
         if(redisdal==null)
         {
             Properties redisprops = new Properties();

             redisprops = ConfigReader.ReadConfig(Constant.REDIS_CONNECTION_CONFIG_FILE);

             GenericObjectPoolConfig config = new GenericObjectPoolConfig();

             redisconfmap = new HashMap(redisprops);

             redisdal = new RedisServerConnectionDAL();

             String redisNodesString = redisconfmap.get(Constant.REDIS_SENTINELS);

             redissentinels = new HashSet<String>(Arrays.asList(redisNodesString.split(Constant.CONFIGURATION_FILE_MULTIVALUE_SEPERATOR)));

             String redisMastersString = redisconfmap.get(Constant.REDIS_MASTERS);

             redismasters = new ArrayList<String>();

             redismasters.addAll(Arrays.asList(redisMastersString.split(Constant.CONFIGURATION_FILE_MULTIVALUE_SEPERATOR)));

             redistimeout = Integer.parseInt(redisconfmap.get(Constant.REDIS_TIMEOUT));

             WAIT_IF_FAIL = Integer.parseInt(redisconfmap.get(Constant.REDIS_WAIT_IF_FAIL));

             RETRIES = Integer.parseInt(redisconfmap.get(Constant.REDIS_RETRIES));

             pool = new ShardedJedisSentinelPool(redismasters, redissentinels, config, redistimeout);

             jedis = pool.getResource();

             pipeline = jedis.pipelined();

         }
         return redisdal;
    }

    public ShardedJedis getJedisResource()
    {
        jedis = pool.getResource();
        return jedis;
    }

    public ShardedJedisPipeline getPipelineResource()
    {
        pipeline = jedis.pipelined();
        return pipeline;
    }

    public ShardedJedisPipeline getPipeline()
    {
        return pipeline;
    }

    public ShardedJedis getJedis()
    {
        return jedis;
    }

    public static void resetRedisServerConnectionDal()
    {
        pipeline = null;
        jedis = null;
        redisdal = null;
    }
} 

Итак, теперь, когда я хочу взаимодействовать с Redis, я делаю что-то вроде этого

Jedis jedis = RedisServerConnectionDAL.getInstance.getJedis();

try
{
    //code.....
}
catch(JedisConnectionException jce)
{
        log.error(jce.getMessage());
        StringWriter sw = new StringWriter();
        PrintWriter pw = new PrintWriter(sw);
        jce.printStackTrace(pw);
        log.error("Error:" + sw.toString());
        if(retrycount < RedisServerConnectionDAL.RETRIES)
        {
            retrycount++;
            log.error("Waiting for " + RedisServerConnectionDAL.WAIT_IF_FAIL + ", after which it will reconnect to Redis.");
            try 
            {
                Thread.sleep(RedisServerConnectionDAL.WAIT_IF_FAIL);
            }
            catch (InterruptedException e) 
            {
                log.error(e.getMessage());
                e.printStackTrace(pw);
                log.error("Error:" + sw.toString());
                log.error("Error while Sleeing!!!");
            }
            RedisServerConnectionDAL.resetRedisServerConnectionDal();

            jedis = RedisServerConnectionDAL.getInstance();.getJedisResource();
            pipeline = redisserverconnectiondal.getPipelineResource();
        }
        else
        {
            log.error("Redis retries exhausted");
            log.error(jce.getMessage());
            StringWriter sw1 = new StringWriter();
            PrintWriter pw1 = new PrintWriter(sw1);
            jce.printStackTrace(pw1);
            log.error("Error:" + sw1.toString());
            log.error("Error while executing pipeline in Redis!");
            retrycount = 0;;
        }
    }
    catch(Exception e)
    {
        log.error(e.getMessage());
        StringWriter sw = new StringWriter();
        PrintWriter pw = new PrintWriter(sw);
        e.printStackTrace(pw);
        log.error("Error:" + sw.toString());
        log.error("Error while executing pipeline in Redis!");
    }

Но я постоянно получаю исключения, такие как исключение отказа в соединении, исключение закрытия соединения.

Иногда я также получаю исключение нулевого указателя в блоке catch в строке

jedis = RedisServerConnectionDAL.getInstance();.getJedisResource();

Типичный журнал для исключения выглядит так

ERROR  com.cleartrail.entityprofiling.rediscachecomponent.InMemoryDataAccessLayer -     Error:redis.clients.jedis.exceptions.JedisConnectionException: java.net.SocketException: Connection reset
    at redis.clients.jedis.Protocol.sendCommand(Protocol.java:94)
    at redis.clients.jedis.Protocol.sendCommand(Protocol.java:74)
    at redis.clients.jedis.Connection.sendCommand(Connection.java:78)
    at redis.clients.jedis.BinaryClient.hexists(BinaryClient.java:257)
    at redis.clients.jedis.Client.hexists(Client.java:174)
    at redis.clients.jedis.Jedis.hexists(Jedis.java:705)
    at redis.clients.jedis.ShardedJedis.hexists(ShardedJedis.java:213)
    at com.cleartrail.entityprofiling.rediscachecomponent.InMemoryDataAccessLayer.checkForNewEntity(InMemoryDataAccessLayer.java:339)
    at com.cleartrail.entityprofiling.rediscachecomponent.InMemoryDataAccessLayer.writeInMemoryData(InMemoryDataAccessLayer.java:215)
    at com.cleartrail.entityprofiling.rediscachecomponent.InMemoryDataAccessLayer.run(InMemoryDataAccessLayer.java:715)
    at java.lang.Thread.run(Thread.java:745)
Caused by: java.net.SocketException: Connection reset
    at java.net.SocketOutputStream.socketWrite(SocketOutputStream.java:113)
    at java.net.SocketOutputStream.write(SocketOutputStream.java:153)
    at redis.clients.util.RedisOutputStream.flushBuffer(RedisOutputStream.java:31)
    at redis.clients.util.RedisOutputStream.writeIntCrLf(RedisOutputStream.java:186)
    at redis.clients.jedis.Protocol.sendCommand(Protocol.java:81)
    ... 10 more

Я знаю, что неправильно использую пул соединений. Пожалуйста, предложите правильный способ сделать это. Кроме того, я использую многопоточную среду jedis, в которой два потока будут вызывать метод RedisServerConnectionDAL.getInstance().getJedisResource() и взаимодействовать с Redis. Потокобезопасен ли Jedis?


person abi_pat    schedule 21.08.2015    source источник


Ответы (1)


Экземпляры Jedis не являются потокобезопасными, но вы можете использовать JedisPool, чтобы получить соединение для каждого потока и убедиться, что вы получаете рабочее соединение из пула.

клиент lettuce является потокобезопасным и управляет автоматическим повторным подключением. Вы не увидите никаких исключений в своем коде из-за переподключения/сброса соединения. салат также работает с Redis Sentinel.

person mp911de    schedule 21.08.2015