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:
anotherJJz 2021-06-09 18:54:42 +08:00 committed by Norman Maurer
parent 998a576f65
commit 17fd44a5d4
2 changed files with 153 additions and 45 deletions

View File

@ -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);

View File

@ -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);
}
}