Modify List to Map of pooled redis message in FixedRedisMessagePool (#11300)
Motivation: To simplify retrieving pooled message messages, add enums that can be used as key. Modifications: - Modify pooled collections from List to Map in FixedRedisMessagePool - Allow to use enum as the key to easy get pooled message. - Add unit tests Result: Users can get pooled message by enum instead of the whole string Co-authored-by: Norman Maurer <norman_maurer@apple.com>
This commit is contained in:
parent
998a576f65
commit
17fd44a5d4
@ -31,33 +31,42 @@ import java.util.Map;
|
|||||||
@UnstableApi
|
@UnstableApi
|
||||||
public final class FixedRedisMessagePool implements RedisMessagePool {
|
public final class FixedRedisMessagePool implements RedisMessagePool {
|
||||||
|
|
||||||
private static final String[] DEFAULT_SIMPLE_STRINGS = {
|
public enum RedisReplyKey {
|
||||||
"OK",
|
OK, PONG, QUEUED
|
||||||
"PONG",
|
}
|
||||||
"QUEUED",
|
|
||||||
};
|
|
||||||
|
|
||||||
private static final String[] DEFAULT_ERRORS = {
|
public enum RedisErrorKey {
|
||||||
"ERR",
|
ERR("ERR"),
|
||||||
"ERR index out of range",
|
ERR_IDX("ERR index out of range"),
|
||||||
"ERR no such key",
|
ERR_NOKEY("ERR no such key"),
|
||||||
"ERR source and destination objects are the same",
|
ERR_SAMEOBJ("ERR source and destination objects are the same"),
|
||||||
"ERR syntax error",
|
ERR_SYNTAX("ERR syntax error"),
|
||||||
"BUSY Redis is busy running a script. You can only call SCRIPT KILL or SHUTDOWN NOSAVE.",
|
BUSY("BUSY Redis is busy running a script. You can only call SCRIPT KILL or SHUTDOWN NOSAVE."),
|
||||||
"BUSYKEY Target key name already exists.",
|
BUSYKEY("BUSYKEY Target key name already exists."),
|
||||||
"EXECABORT Transaction discarded because of previous errors.",
|
EXECABORT("EXECABORT Transaction discarded because of previous errors."),
|
||||||
"LOADING Redis is loading the dataset in memory",
|
LOADING("LOADING Redis is loading the dataset in memory"),
|
||||||
"MASTERDOWN Link with MASTER is down and slave-serve-stale-data is set to 'no'.",
|
MASTERDOWN("MASTERDOWN Link with MASTER is down and slave-serve-stale-data is set to 'no'."),
|
||||||
"MISCONF Redis is configured to save RDB snapshots, but is currently not able to persist on disk. " +
|
MISCONF("MISCONF Redis is configured to save RDB snapshots, but is currently not able to persist on disk. " +
|
||||||
"Commands that may modify the data set are disabled. Please check Redis logs for details " +
|
"Commands that may modify the data set are disabled. Please check Redis logs for details " +
|
||||||
"about the error.",
|
"about the error."),
|
||||||
"NOAUTH Authentication required.",
|
NOREPLICAS("NOREPLICAS Not enough good slaves to write."),
|
||||||
"NOREPLICAS Not enough good slaves to write.",
|
NOSCRIPT("NOSCRIPT No matching script. Please use EVAL."),
|
||||||
"NOSCRIPT No matching script. Please use EVAL.",
|
OOM("OOM command not allowed when used memory > 'maxmemory'."),
|
||||||
"OOM command not allowed when used memory > 'maxmemory'.",
|
READONLY("READONLY You can't write against a read only slave."),
|
||||||
"READONLY You can't write against a read only slave.",
|
WRONGTYPE("WRONGTYPE Operation against a key holding the wrong kind of value"),
|
||||||
"WRONGTYPE Operation against a key holding the wrong kind of value",
|
NOT_AUTH("NOAUTH Authentication required.");
|
||||||
};
|
|
||||||
|
private final String msg;
|
||||||
|
|
||||||
|
RedisErrorKey(String msg) {
|
||||||
|
this.msg = msg;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return msg;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private static final long MIN_CACHED_INTEGER_NUMBER = RedisConstants.NULL_VALUE; // inclusive
|
private static final long MIN_CACHED_INTEGER_NUMBER = RedisConstants.NULL_VALUE; // inclusive
|
||||||
private static final long MAX_CACHED_INTEGER_NUMBER = 128; // exclusive
|
private static final long MAX_CACHED_INTEGER_NUMBER = 128; // exclusive
|
||||||
@ -73,8 +82,10 @@ public final class FixedRedisMessagePool implements RedisMessagePool {
|
|||||||
// internal caches.
|
// internal caches.
|
||||||
private final Map<ByteBuf, SimpleStringRedisMessage> byteBufToSimpleStrings;
|
private final Map<ByteBuf, SimpleStringRedisMessage> byteBufToSimpleStrings;
|
||||||
private final Map<String, SimpleStringRedisMessage> stringToSimpleStrings;
|
private final Map<String, SimpleStringRedisMessage> stringToSimpleStrings;
|
||||||
|
private final Map<RedisReplyKey, SimpleStringRedisMessage> keyToSimpleStrings;
|
||||||
private final Map<ByteBuf, ErrorRedisMessage> byteBufToErrors;
|
private final Map<ByteBuf, ErrorRedisMessage> byteBufToErrors;
|
||||||
private final Map<String, ErrorRedisMessage> stringToErrors;
|
private final Map<String, ErrorRedisMessage> stringToErrors;
|
||||||
|
private final Map<RedisErrorKey, ErrorRedisMessage> keyToErrors;
|
||||||
private final Map<ByteBuf, IntegerRedisMessage> byteBufToIntegers;
|
private final Map<ByteBuf, IntegerRedisMessage> byteBufToIntegers;
|
||||||
private final LongObjectMap<IntegerRedisMessage> longToIntegers;
|
private final LongObjectMap<IntegerRedisMessage> longToIntegers;
|
||||||
private final LongObjectMap<byte[]> longToByteBufs;
|
private final LongObjectMap<byte[]> longToByteBufs;
|
||||||
@ -83,33 +94,38 @@ public final class FixedRedisMessagePool implements RedisMessagePool {
|
|||||||
* Creates a {@link FixedRedisMessagePool} instance.
|
* Creates a {@link FixedRedisMessagePool} instance.
|
||||||
*/
|
*/
|
||||||
private FixedRedisMessagePool() {
|
private FixedRedisMessagePool() {
|
||||||
byteBufToSimpleStrings = new HashMap<>(DEFAULT_SIMPLE_STRINGS.length, 1.0f);
|
keyToSimpleStrings = new HashMap<>(RedisReplyKey.values().length, 1.0f);
|
||||||
stringToSimpleStrings = new HashMap<>(DEFAULT_SIMPLE_STRINGS.length, 1.0f);
|
stringToSimpleStrings = new HashMap<>(RedisReplyKey.values().length, 1.0f);
|
||||||
for (String message : DEFAULT_SIMPLE_STRINGS) {
|
byteBufToSimpleStrings = new HashMap<>(RedisReplyKey.values().length, 1.0f);
|
||||||
ByteBuf key = Unpooled.unmodifiableBuffer(
|
for (RedisReplyKey value : RedisReplyKey.values()) {
|
||||||
Unpooled.unreleasableBuffer(Unpooled.wrappedBuffer(message.getBytes(CharsetUtil.UTF_8))));
|
ByteBuf key = Unpooled.unreleasableBuffer(Unpooled.wrappedBuffer(
|
||||||
SimpleStringRedisMessage cached = new SimpleStringRedisMessage(message);
|
value.name().getBytes(CharsetUtil.UTF_8))).asReadOnly();
|
||||||
byteBufToSimpleStrings.put(key, cached);
|
SimpleStringRedisMessage message = new SimpleStringRedisMessage(new String(Unpooled.unreleasableBuffer(
|
||||||
stringToSimpleStrings.put(message, cached);
|
Unpooled.wrappedBuffer(value.name().getBytes(CharsetUtil.UTF_8))).array()));
|
||||||
|
stringToSimpleStrings.put(value.name(), message);
|
||||||
|
keyToSimpleStrings.put(value, message);
|
||||||
|
byteBufToSimpleStrings.put(key, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
byteBufToErrors = new HashMap<>(DEFAULT_ERRORS.length, 1.0f);
|
keyToErrors = new HashMap<>(RedisErrorKey.values().length, 1.0f);
|
||||||
stringToErrors = new HashMap<>(DEFAULT_ERRORS.length, 1.0f);
|
stringToErrors = new HashMap<>(RedisErrorKey.values().length, 1.0f);
|
||||||
for (String message : DEFAULT_ERRORS) {
|
byteBufToErrors = new HashMap<>(RedisErrorKey.values().length, 1.0f);
|
||||||
ByteBuf key = Unpooled.unmodifiableBuffer(
|
for (RedisErrorKey value : RedisErrorKey.values()) {
|
||||||
Unpooled.unreleasableBuffer(Unpooled.wrappedBuffer(message.getBytes(CharsetUtil.UTF_8))));
|
ByteBuf key = Unpooled.unreleasableBuffer(Unpooled.wrappedBuffer(
|
||||||
ErrorRedisMessage cached = new ErrorRedisMessage(message);
|
value.toString().getBytes(CharsetUtil.UTF_8))).asReadOnly();
|
||||||
byteBufToErrors.put(key, cached);
|
ErrorRedisMessage message = new ErrorRedisMessage(new String(Unpooled.unreleasableBuffer(
|
||||||
stringToErrors.put(message, cached);
|
Unpooled.wrappedBuffer(value.toString().getBytes(CharsetUtil.UTF_8))).array()));
|
||||||
|
stringToErrors.put(value.toString(), message);
|
||||||
|
keyToErrors.put(value, message);
|
||||||
|
byteBufToErrors.put(key, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
byteBufToIntegers = new HashMap<>(SIZE_CACHED_INTEGER_NUMBER, 1.0f);
|
byteBufToIntegers = new HashMap<>(SIZE_CACHED_INTEGER_NUMBER, 1.0f);
|
||||||
longToIntegers = new LongObjectHashMap<IntegerRedisMessage>(SIZE_CACHED_INTEGER_NUMBER, 1.0f);
|
longToIntegers = new LongObjectHashMap<>(SIZE_CACHED_INTEGER_NUMBER, 1.0f);
|
||||||
longToByteBufs = new LongObjectHashMap<byte[]>(SIZE_CACHED_INTEGER_NUMBER, 1.0f);
|
longToByteBufs = new LongObjectHashMap<>(SIZE_CACHED_INTEGER_NUMBER, 1.0f);
|
||||||
for (long value = MIN_CACHED_INTEGER_NUMBER; value < MAX_CACHED_INTEGER_NUMBER; value++) {
|
for (long value = MIN_CACHED_INTEGER_NUMBER; value < MAX_CACHED_INTEGER_NUMBER; value++) {
|
||||||
byte[] keyBytes = RedisCodecUtil.longToAsciiBytes(value);
|
byte[] keyBytes = RedisCodecUtil.longToAsciiBytes(value);
|
||||||
ByteBuf keyByteBuf = Unpooled.unmodifiableBuffer(Unpooled.unreleasableBuffer(
|
ByteBuf keyByteBuf = Unpooled.unreleasableBuffer(Unpooled.wrappedBuffer(keyBytes)).asReadOnly();
|
||||||
Unpooled.wrappedBuffer(keyBytes)));
|
|
||||||
IntegerRedisMessage cached = new IntegerRedisMessage(value);
|
IntegerRedisMessage cached = new IntegerRedisMessage(value);
|
||||||
byteBufToIntegers.put(keyByteBuf, cached);
|
byteBufToIntegers.put(keyByteBuf, cached);
|
||||||
longToIntegers.put(value, cached);
|
longToIntegers.put(value, cached);
|
||||||
@ -122,6 +138,14 @@ public final class FixedRedisMessagePool implements RedisMessagePool {
|
|||||||
return stringToSimpleStrings.get(content);
|
return stringToSimpleStrings.get(content);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns {@link SimpleStringRedisMessage} for the given {@link RedisReplyKey}
|
||||||
|
* or {@code null} if it does not exist.
|
||||||
|
*/
|
||||||
|
public SimpleStringRedisMessage getSimpleString(RedisReplyKey key) {
|
||||||
|
return keyToSimpleStrings.get(key);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public SimpleStringRedisMessage getSimpleString(ByteBuf content) {
|
public SimpleStringRedisMessage getSimpleString(ByteBuf content) {
|
||||||
return byteBufToSimpleStrings.get(content);
|
return byteBufToSimpleStrings.get(content);
|
||||||
@ -132,6 +156,14 @@ public final class FixedRedisMessagePool implements RedisMessagePool {
|
|||||||
return stringToErrors.get(content);
|
return stringToErrors.get(content);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns {@link ErrorRedisMessage} for the given {@link RedisErrorKey}
|
||||||
|
* or {@code null} if it does not exist.
|
||||||
|
*/
|
||||||
|
public ErrorRedisMessage getError(RedisErrorKey key) {
|
||||||
|
return keyToErrors.get(key);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ErrorRedisMessage getError(ByteBuf content) {
|
public ErrorRedisMessage getError(ByteBuf content) {
|
||||||
return byteBufToErrors.get(content);
|
return byteBufToErrors.get(content);
|
||||||
|
@ -0,0 +1,76 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 The Netty Project
|
||||||
|
*
|
||||||
|
* The Netty Project licenses this file to you 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:
|
||||||
|
*
|
||||||
|
* https://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 io.netty.handler.codec.redis;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import static io.netty.handler.codec.redis.RedisCodecTestUtil.byteBufOf;
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Verifies the correct functionality of the {@link FixedRedisMessagePool}.
|
||||||
|
*/
|
||||||
|
public class FixedRedisMessagePoolTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void shouldGetSameMessageObject() {
|
||||||
|
FixedRedisMessagePool pool = FixedRedisMessagePool.INSTANCE;
|
||||||
|
|
||||||
|
SimpleStringRedisMessage fromStr = pool.getSimpleString("OK");
|
||||||
|
SimpleStringRedisMessage fromEnum = pool.getSimpleString(FixedRedisMessagePool.RedisReplyKey.OK);
|
||||||
|
SimpleStringRedisMessage fromByteBuf = pool.getSimpleString(byteBufOf("OK"));
|
||||||
|
|
||||||
|
assertEquals(fromStr.content(), "OK");
|
||||||
|
assertEquals(fromStr, fromEnum);
|
||||||
|
assertEquals(fromStr, fromByteBuf);
|
||||||
|
|
||||||
|
ErrorRedisMessage errorFromStr = pool.getError("NOAUTH Authentication required.");
|
||||||
|
ErrorRedisMessage errorFromEnum = pool.getError(FixedRedisMessagePool.RedisErrorKey.NOT_AUTH);
|
||||||
|
ErrorRedisMessage errorFromByteBuf = pool.getError(byteBufOf("NOAUTH Authentication required."));
|
||||||
|
|
||||||
|
assertEquals(errorFromStr.content(), "NOAUTH Authentication required.");
|
||||||
|
assertEquals(errorFromStr, errorFromEnum);
|
||||||
|
assertEquals(errorFromStr, errorFromByteBuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void shouldReturnNullByNotExistKey() {
|
||||||
|
FixedRedisMessagePool pool = FixedRedisMessagePool.INSTANCE;
|
||||||
|
|
||||||
|
SimpleStringRedisMessage message1 = pool.getSimpleString("Not exist");
|
||||||
|
SimpleStringRedisMessage message2 = pool.getSimpleString(byteBufOf("Not exist"));
|
||||||
|
|
||||||
|
assertNull(message1);
|
||||||
|
assertNull(message2);
|
||||||
|
|
||||||
|
ErrorRedisMessage error1 = pool.getError("Not exist");
|
||||||
|
ErrorRedisMessage error2 = pool.getError(byteBufOf("Not exist"));
|
||||||
|
|
||||||
|
assertNull(error1);
|
||||||
|
assertNull(error2);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void shouldReturnDifferentMessage() {
|
||||||
|
FixedRedisMessagePool pool = FixedRedisMessagePool.INSTANCE;
|
||||||
|
|
||||||
|
SimpleStringRedisMessage okMessage = pool.getSimpleString(FixedRedisMessagePool.RedisReplyKey.OK);
|
||||||
|
SimpleStringRedisMessage pongMessage = pool.getSimpleString(FixedRedisMessagePool.RedisReplyKey.PONG);
|
||||||
|
|
||||||
|
assertNotEquals(okMessage, pongMessage);
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user