From ce92e6a84f0853a36e36e54c0df453594aaf7b2e Mon Sep 17 00:00:00 2001 From: Tihomir Krasimirov Mateev Date: Mon, 30 Dec 2024 18:20:21 +0100 Subject: [PATCH] Handle UTF-8 characters in command arguments (#3075) --- .../io/lettuce/core/protocol/CommandArgs.java | 27 ++++++------------- .../core/AuthenticationIntegrationTests.java | 15 +++++++++++ .../core/RedisCommandBuilderUnitTests.java | 11 ++++++++ .../commands/AclCommandIntegrationTests.java | 4 +-- 4 files changed, 36 insertions(+), 21 deletions(-) diff --git a/src/main/java/io/lettuce/core/protocol/CommandArgs.java b/src/main/java/io/lettuce/core/protocol/CommandArgs.java index da5f897852..addc78a8db 100644 --- a/src/main/java/io/lettuce/core/protocol/CommandArgs.java +++ b/src/main/java/io/lettuce/core/protocol/CommandArgs.java @@ -20,6 +20,7 @@ package io.lettuce.core.protocol; import java.nio.ByteBuffer; +import java.nio.CharBuffer; import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.Base64; @@ -53,7 +54,7 @@ */ public class CommandArgs { - static final byte[] CRLF = "\r\n".getBytes(StandardCharsets.US_ASCII); + static final byte[] CRLF = "\r\n".getBytes(StandardCharsets.UTF_8); protected final RedisCodec codec; @@ -609,16 +610,9 @@ void encode(ByteBuf target) { } static void writeString(ByteBuf target, String value) { + byte[] output = value.getBytes(StandardCharsets.UTF_8); - target.writeByte('$'); - - IntegerArgument.writeInteger(target, value.length()); - target.writeBytes(CRLF); - - for (int i = 0; i < value.length(); i++) { - target.writeByte((byte) value.charAt(i)); - } - target.writeBytes(CRLF); + BytesArgument.writeBytes(target, output); } @Override @@ -646,16 +640,11 @@ void encode(ByteBuf target) { } static void writeString(ByteBuf target, char[] value) { + final ByteBuffer byteBuffer = StandardCharsets.UTF_8.encode(CharBuffer.wrap(value)); + final byte[] output = new byte[byteBuffer.remaining()]; + byteBuffer.get(output); - target.writeByte('$'); - - IntegerArgument.writeInteger(target, value.length); - target.writeBytes(CRLF); - - for (char c : value) { - target.writeByte((byte) c); - } - target.writeBytes(CRLF); + BytesArgument.writeBytes(target, output); } @Override diff --git a/src/test/java/io/lettuce/core/AuthenticationIntegrationTests.java b/src/test/java/io/lettuce/core/AuthenticationIntegrationTests.java index 9914d21896..91ec54ca28 100644 --- a/src/test/java/io/lettuce/core/AuthenticationIntegrationTests.java +++ b/src/test/java/io/lettuce/core/AuthenticationIntegrationTests.java @@ -54,6 +54,9 @@ void setUp(StatefulRedisConnection connection) { connection.sync().dispatch(CommandType.ACL, new StatusOutput<>(StringCodec.UTF8), new CommandArgs<>(StringCodec.UTF8).add("SETUSER").add("john").add("on").add(">foobared").add("-@all")); + connection.sync().dispatch(CommandType.ACL, new StatusOutput<>(StringCodec.UTF8), + new CommandArgs<>(StringCodec.UTF8).add("SETUSER").add("日本語").add("on").add(">日本語").add("+@all")); + connection.sync().dispatch(CommandType.ACL, new StatusOutput<>(StringCodec.UTF8), new CommandArgs<>(StringCodec.UTF8).add("SETUSER").add("steave").add("on").add(">foobared").add("+@all")); } @@ -90,6 +93,18 @@ void ownCredentialProvider(RedisClient client) { }); } + @Test + @Inject + void authAsJapanese(RedisClient client) { + + RedisURI uri = RedisURI.builder().withHost(TestSettings.host()).withPort(TestSettings.port()) + .withAuthentication("日本語", "日本語".toCharArray()).build(); + + StatefulRedisConnection connection = client.connect(uri); + assertThat(connection.sync().ping()).isEqualTo("PONG"); + connection.close(); + } + // Simulate test user credential rotation, and verify that re-authentication is successful @Test @Inject diff --git a/src/test/java/io/lettuce/core/RedisCommandBuilderUnitTests.java b/src/test/java/io/lettuce/core/RedisCommandBuilderUnitTests.java index d8e71cffb3..2d4f1cf405 100644 --- a/src/test/java/io/lettuce/core/RedisCommandBuilderUnitTests.java +++ b/src/test/java/io/lettuce/core/RedisCommandBuilderUnitTests.java @@ -32,6 +32,17 @@ class RedisCommandBuilderUnitTests { RedisCommandBuilder sut = new RedisCommandBuilder<>(StringCodec.UTF8); + @Test() + void shouldCorrectlyConstructHello() { + + Command command = sut.hello(3, "日本語", "日本語".toCharArray(), null); + ByteBuf buf = Unpooled.directBuffer(); + command.encode(buf); + + assertThat(buf.toString(StandardCharsets.UTF_8)).isEqualTo("*5\r\n" + "$5\r\n" + "HELLO\r\n" + "$1\r\n" + "3\r\n" + + "$4\r\n" + "AUTH\r\n" + "$9\r\n" + "日本語\r\n" + "$9\r\n" + "日本語\r\n"); + } + @Test void shouldCorrectlyConstructXreadgroup() { diff --git a/src/test/java/io/lettuce/core/commands/AclCommandIntegrationTests.java b/src/test/java/io/lettuce/core/commands/AclCommandIntegrationTests.java index 7fbeb7cf9f..4d154dc0f7 100644 --- a/src/test/java/io/lettuce/core/commands/AclCommandIntegrationTests.java +++ b/src/test/java/io/lettuce/core/commands/AclCommandIntegrationTests.java @@ -131,7 +131,7 @@ void aclLog() { @Test void aclList() { - assertThat(redis.aclList()).hasSize(1).first().asString().contains("user default"); + assertThat(redis.aclList()).hasSize(2).first().asString().contains("user default"); } @Test @@ -161,7 +161,7 @@ void aclSetuserWithCategories() { @Test void aclUsers() { - assertThat(redis.aclUsers()).hasSize(1).first().isEqualTo("default"); + assertThat(redis.aclUsers()).hasSize(2).first().isEqualTo("default"); } @Test