diff --git a/java/integTest/src/test/java/glide/cluster/ClientTests.java b/java/integTest/src/test/java/glide/cluster/ClientTests.java index 3924747391..347faa885b 100644 --- a/java/integTest/src/test/java/glide/cluster/ClientTests.java +++ b/java/integTest/src/test/java/glide/cluster/ClientTests.java @@ -2,47 +2,284 @@ package glide.cluster; import static glide.TestConfiguration.CLUSTER_PORTS; +import static glide.api.BaseClient.OK; +import static glide.api.models.commands.InfoOptions.Section.SERVER; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; import static org.junit.jupiter.api.Assertions.assertThrows; import static org.junit.jupiter.api.Assertions.assertTrue; +import static org.junit.jupiter.api.Assumptions.assumeTrue; import glide.api.RedisClusterClient; +import glide.api.models.ClusterValue; +import glide.api.models.commands.InfoOptions; import glide.api.models.configuration.NodeAddress; import glide.api.models.configuration.RedisClusterClientConfiguration; +import glide.api.models.configuration.RedisCredentials; import glide.api.models.exceptions.ClosingException; +import glide.api.models.exceptions.RequestException; +import java.lang.module.ModuleDescriptor; +import java.security.SecureRandom; +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.CompletableFuture; import java.util.concurrent.ExecutionException; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.stream.Collectors; import lombok.SneakyThrows; import org.junit.jupiter.api.Test; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.ValueSource; public class ClientTests { + private RedisClusterClientConfiguration.RedisClusterClientConfigurationBuilder + commonClusterClientConfig() { + return RedisClusterClientConfiguration.builder() + .address(NodeAddress.builder().port(CLUSTER_PORTS[0]).build()); + } + + private static Map parseInfoResponse(ClusterValue infoResponse) { + String value; + if (infoResponse.hasSingleData()) { + value = infoResponse.getSingleValue(); + } else { + value = infoResponse.getMultiValue().entrySet().iterator().next().getValue(); + } + + return value + .lines() + .filter(line -> line.contains(":")) + .map(line -> line.split(":", 2)) + .collect( + Collectors.toMap( + parts -> parts[0], + parts -> parts[1], + (existingValue, newValue) -> newValue, + HashMap::new)); + } + + private static String getRandomString(int length) { + String asciiLetters = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"; + SecureRandom random = new SecureRandom(); + StringBuilder sb = new StringBuilder(length); + + for (int i = 0; i < length; i++) { + int index = random.nextInt(asciiLetters.length()); + char randomChar = asciiLetters.charAt(index); + sb.append(randomChar); + } + + return sb.toString(); + } + + @SneakyThrows + private Boolean check_if_server_version_gte(RedisClusterClient client, String minVersion) { + ClusterValue infoStr = client.info(InfoOptions.builder().section(SERVER).build()).get(); + String redisVersion = parseInfoResponse(infoStr).get("redis_version"); + assertNotNull(redisVersion); + return ModuleDescriptor.Version.parse(redisVersion) + .compareTo(ModuleDescriptor.Version.parse(minVersion)) + >= 0; + } + + @SneakyThrows @Test + public void register_client_name_and_version() { + RedisClusterClient client = + RedisClusterClient.CreateClient(commonClusterClientConfig().build()).get(); + + String minVersion = "7.2.0"; + assumeTrue( + check_if_server_version_gte(client, minVersion), "Redis version required >= " + minVersion); + String info = + (String) client.customCommand(new String[] {"CLIENT", "INFO"}).get().getSingleValue(); + assertTrue(info.contains("lib-name=GlideJava")); + assertTrue(info.contains("lib-ver=unknown")); + + client.close(); + } + @SneakyThrows - public void custom_command_info() { + @Test + public void send_and_receive_large_values() { RedisClusterClient client = - RedisClusterClient.CreateClient( - RedisClusterClientConfiguration.builder() - .address(NodeAddress.builder().port(CLUSTER_PORTS[0]).build()) - .clientName("TEST_CLIENT_NAME") - .build()) - .get(); + RedisClusterClient.CreateClient(commonClusterClientConfig().build()).get(); - String clientInfo = - (String) client.customCommand(new String[] {"CLIENT", "INFO"}).get().getSingleValue(); - assertTrue(clientInfo.contains("name=TEST_CLIENT_NAME")); + int length = 2 ^ 16; + String key = getRandomString(length); + String value = getRandomString(length); + + assertEquals(length, key.length()); + assertEquals(length, value.length()); + assertEquals(OK, client.set(key, value).get()); + assertEquals(value, client.get(key).get()); + + client.close(); + } + + @SneakyThrows + @Test + public void send_and_receive_non_ascii_unicode() { + RedisClusterClient client = + RedisClusterClient.CreateClient(commonClusterClientConfig().build()).get(); + + String key = "foo"; + String value = "שלום hello 汉字"; + + assertEquals(OK, client.set(key, value).get()); + assertEquals(value, client.get(key).get()); + + client.close(); + } + + @SneakyThrows + @ParameterizedTest + @ValueSource(ints = {100, 2 ^ 16}) + public void client_can_handle_concurrent_workload(int valueSize) { + ExecutorService executorService = Executors.newFixedThreadPool(8); + RedisClusterClient client = + RedisClusterClient.CreateClient(commonClusterClientConfig().build()).get(); + CompletableFuture[] futures = new CompletableFuture[100]; + + for (int i = 0; i < 100; i++) { + futures[i] = + CompletableFuture.runAsync( + () -> { + String key = getRandomString(valueSize); + String value = getRandomString(valueSize); + try { + assertEquals(OK, client.set(key, value).get()); + assertEquals(value, client.get(key).get()); + } catch (InterruptedException | ExecutionException e) { + throw new RuntimeException(e); + } + }, + executorService); + } + + CompletableFuture.allOf(futures).join(); + + client.close(); + executorService.shutdown(); + } + + @SneakyThrows + @Test + public void can_connect_with_auth_requirepass() { + RedisClusterClient client = + RedisClusterClient.CreateClient(commonClusterClientConfig().build()).get(); + + String password = "TEST_AUTH"; + client.customCommand(new String[] {"CONFIG", "SET", "requirepass", password}).get(); + + // Creation of a new client without a password should fail + ExecutionException exception = + assertThrows( + ExecutionException.class, + () -> RedisClusterClient.CreateClient(commonClusterClientConfig().build()).get()); + assertTrue(exception.getCause() instanceof ClosingException); + + // Creation of a new client with credentials + RedisClusterClient auth_client = + RedisClusterClient.CreateClient( + commonClusterClientConfig() + .credentials(RedisCredentials.builder().password(password).build()) + .build()) + .get(); + + String key = getRandomString(10); + String value = getRandomString(10); + + assertEquals(OK, auth_client.set(key, value).get()); + assertEquals(value, auth_client.get(key).get()); + + // Reset password + client.customCommand(new String[] {"CONFIG", "SET", "requirepass", ""}).get(); + + auth_client.close(); + client.close(); + } + + @SneakyThrows + @Test + public void can_connect_with_auth_acl() { + RedisClusterClient client = + RedisClusterClient.CreateClient(commonClusterClientConfig().build()).get(); + + String username = "testuser"; + String password = "TEST_AUTH"; + assertEquals( + OK, + client + .customCommand( + new String[] { + "ACL", + "SETUSER", + username, + "on", + "allkeys", + "+get", + "+cluster", + "+ping", + "+info", + "+client", + ">" + password, + }) + .get() + .getSingleValue()); + + String key = getRandomString(10); + String value = getRandomString(10); + + assertEquals(OK, client.set(key, value).get()); + + // Creation of a new cluster client with credentials + RedisClusterClient testUserClient = + RedisClusterClient.CreateClient( + commonClusterClientConfig() + .credentials( + RedisCredentials.builder().username(username).password(password).build()) + .build()) + .get(); + + assertEquals(value, testUserClient.get(key).get()); + + ExecutionException executionException = + assertThrows(ExecutionException.class, () -> testUserClient.set("foo", "bar").get()); + assertTrue(executionException.getCause() instanceof RequestException); + + client.customCommand(new String[] {"ACL", "DELUSER", username}).get(); + + testUserClient.close(); + client.close(); } @Test @SneakyThrows public void close_client_throws_ExecutionException_with_ClosingException_cause() { RedisClusterClient client = - RedisClusterClient.CreateClient( - RedisClusterClientConfiguration.builder() - .address(NodeAddress.builder().port(CLUSTER_PORTS[0]).build()) - .build()) - .get(); + RedisClusterClient.CreateClient(commonClusterClientConfig().build()) + .get(); client.close(); ExecutionException executionException = - assertThrows(ExecutionException.class, () -> client.set("foo", "bar").get()); + assertThrows(ExecutionException.class, () -> client.set("foo", "bar").get()); assertTrue(executionException.getCause() instanceof ClosingException); } + + @SneakyThrows + @Test + public void custom_command_info() { + RedisClusterClient client = + RedisClusterClient.CreateClient( + commonClusterClientConfig().clientName("TEST_CLIENT_NAME").build()) + .get(); + + String clientInfo = + (String) client.customCommand(new String[] {"CLIENT", "INFO"}).get().getSingleValue(); + assertTrue(clientInfo.contains("name=TEST_CLIENT_NAME")); + + client.close(); + } } diff --git a/java/integTest/src/test/java/glide/standalone/ClientTests.java b/java/integTest/src/test/java/glide/standalone/ClientTests.java index 960696b06e..32b06f9d43 100644 --- a/java/integTest/src/test/java/glide/standalone/ClientTests.java +++ b/java/integTest/src/test/java/glide/standalone/ClientTests.java @@ -2,46 +2,274 @@ package glide.standalone; import static glide.TestConfiguration.STANDALONE_PORTS; +import static glide.api.BaseClient.OK; +import static glide.api.models.commands.InfoOptions.Section.SERVER; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; import static org.junit.jupiter.api.Assertions.assertThrows; import static org.junit.jupiter.api.Assertions.assertTrue; +import static org.junit.jupiter.api.Assumptions.assumeTrue; import glide.api.RedisClient; +import glide.api.models.commands.InfoOptions; import glide.api.models.configuration.NodeAddress; import glide.api.models.configuration.RedisClientConfiguration; +import glide.api.models.configuration.RedisCredentials; import glide.api.models.exceptions.ClosingException; +import glide.api.models.exceptions.RequestException; +import java.lang.module.ModuleDescriptor.Version; +import java.security.SecureRandom; +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.CompletableFuture; import java.util.concurrent.ExecutionException; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.stream.Collectors; import lombok.SneakyThrows; import org.junit.jupiter.api.Test; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.ValueSource; public class ClientTests { + + private RedisClientConfiguration.RedisClientConfigurationBuilder commonClientConfig() { + return RedisClientConfiguration.builder() + .address(NodeAddress.builder().port(STANDALONE_PORTS[0]).build()); + } + + private static Map parseInfoResponse(String serverInfo) { + return serverInfo + .lines() + .filter(line -> line.contains(":")) + .map(line -> line.split(":", 2)) + .collect( + Collectors.toMap( + parts -> parts[0], + parts -> parts[1], + (existingValue, newValue) -> newValue, + HashMap::new)); + } + + private static String getRandomString(int length) { + String asciiLetters = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"; + SecureRandom random = new SecureRandom(); + StringBuilder sb = new StringBuilder(length); + + for (int i = 0; i < length; i++) { + int index = random.nextInt(asciiLetters.length()); + char randomChar = asciiLetters.charAt(index); + sb.append(randomChar); + } + + return sb.toString(); + } + + @SneakyThrows + private Boolean check_if_server_version_gte(RedisClient client, String minVersion) { + String infoStr = client.info(InfoOptions.builder().section(SERVER).build()).get(); + String redisVersion = parseInfoResponse(infoStr).get("redis_version"); + assertNotNull(redisVersion); + return Version.parse(redisVersion).compareTo(Version.parse(minVersion)) >= 0; + } + + @SneakyThrows @Test + public void register_client_name_and_version() { + RedisClient client = RedisClient.CreateClient(commonClientConfig().build()).get(); + + String minVersion = "7.2.0"; + assumeTrue( + check_if_server_version_gte(client, minVersion), "Redis version required >= " + minVersion); + String info = (String) client.customCommand(new String[] {"CLIENT", "INFO"}).get(); + assertTrue(info.contains("lib-name=GlideJava")); + assertTrue(info.contains("lib-ver=unknown")); + + client.close(); + } + @SneakyThrows - public void custom_command_info() { - RedisClient client = - RedisClient.CreateClient( - RedisClientConfiguration.builder() - .address(NodeAddress.builder().port(STANDALONE_PORTS[0]).build()) - .clientName("TEST_CLIENT_NAME") - .build()) - .get(); + @Test + public void send_and_receive_large_values() { + RedisClient client = RedisClient.CreateClient(commonClientConfig().build()).get(); + + int length = 2 ^ 16; + String key = getRandomString(length); + String value = getRandomString(length); + + assertEquals(length, key.length()); + assertEquals(length, value.length()); + assertEquals(OK, client.set(key, value).get()); + assertEquals(value, client.get(key).get()); + + client.close(); + } + + @SneakyThrows + @Test + public void send_and_receive_non_ascii_unicode() { + RedisClient client = RedisClient.CreateClient(commonClientConfig().build()).get(); + + String key = "foo"; + String value = "שלום hello 汉字"; + + assertEquals(OK, client.set(key, value).get()); + assertEquals(value, client.get(key).get()); + + client.close(); + } + + @SneakyThrows + @ParameterizedTest + @ValueSource(ints = {100, 2 ^ 16}) + public void client_can_handle_concurrent_workload(int valueSize) { + ExecutorService executorService = Executors.newFixedThreadPool(8); + RedisClient client = RedisClient.CreateClient(commonClientConfig().build()).get(); + CompletableFuture[] futures = new CompletableFuture[100]; + + for (int i = 0; i < 100; i++) { + futures[i] = + CompletableFuture.runAsync( + () -> { + String key = getRandomString(valueSize); + String value = getRandomString(valueSize); + try { + assertEquals(OK, client.set(key, value).get()); + assertEquals(value, client.get(key).get()); + } catch (InterruptedException | ExecutionException e) { + throw new RuntimeException(e); + } + }, + executorService); + } + + CompletableFuture.allOf(futures).join(); + + client.close(); + executorService.shutdown(); + } + + @SneakyThrows + @Test + public void can_connect_with_auth_require_pass() { + RedisClient client = RedisClient.CreateClient(commonClientConfig().build()).get(); + + String password = "TEST_AUTH"; + client.customCommand(new String[] {"CONFIG", "SET", "requirepass", password}).get(); + + // Creation of a new client without a password should fail + ExecutionException exception = + assertThrows( + ExecutionException.class, + () -> RedisClient.CreateClient(commonClientConfig().build()).get()); + assertTrue(exception.getCause() instanceof ClosingException); + + // Creation of a new client with credentials + RedisClient auth_client = + RedisClient.CreateClient( + commonClientConfig() + .credentials(RedisCredentials.builder().password(password).build()) + .build()) + .get(); + + String key = getRandomString(10); + String value = getRandomString(10); + + assertEquals(OK, auth_client.set(key, value).get()); + assertEquals(value, auth_client.get(key).get()); + + // Reset password + client.customCommand(new String[] {"CONFIG", "SET", "requirepass", ""}).get(); + + auth_client.close(); + client.close(); + } + + @SneakyThrows + @Test + public void can_connect_with_auth_acl() { + RedisClient client = RedisClient.CreateClient(commonClientConfig().build()).get(); + + String username = "testuser"; + String password = "TEST_AUTH"; + assertEquals( + OK, + client + .customCommand( + new String[] { + "ACL", + "SETUSER", + username, + "on", + "allkeys", + "+get", + "+cluster", + "+ping", + "+info", + "+client", + ">" + password, + }) + .get()); + + String key = getRandomString(10); + String value = getRandomString(10); + + assertEquals(OK, client.set(key, value).get()); + + // Creation of a new client with credentials + RedisClient testUserClient = + RedisClient.CreateClient( + commonClientConfig() + .credentials( + RedisCredentials.builder().username(username).password(password).build()) + .build()) + .get(); + + assertEquals(value, testUserClient.get(key).get()); + ExecutionException executionException = + assertThrows(ExecutionException.class, () -> testUserClient.set("foo", "bar").get()); + assertTrue(executionException.getCause() instanceof RequestException); + + client.customCommand(new String[] {"ACL", "DELUSER", username}).get(); + + testUserClient.close(); + client.close(); + } + + @SneakyThrows + @Test + public void select_standalone_database_id() { + RedisClient client = RedisClient.CreateClient(commonClientConfig().databaseId(4).build()).get(); String clientInfo = (String) client.customCommand(new String[] {"CLIENT", "INFO"}).get(); - assertTrue(clientInfo.contains("name=TEST_CLIENT_NAME")); + assertTrue(clientInfo.contains("db=4")); + + client.close(); } @Test @SneakyThrows public void close_client_throws_ExecutionException_with_ClosingException_cause() { RedisClient client = - RedisClient.CreateClient( - RedisClientConfiguration.builder() - .address(NodeAddress.builder().port(STANDALONE_PORTS[0]).build()) - .build()) - .get(); + RedisClient.CreateClient( + commonClientConfig().build()) + .get(); client.close(); ExecutionException executionException = - assertThrows(ExecutionException.class, () -> client.set("key", "value").get()); + assertThrows(ExecutionException.class, () -> client.set("key", "value").get()); assertTrue(executionException.getCause() instanceof ClosingException); } + + @SneakyThrows + @Test + public void custom_command_info() { + RedisClient client = + RedisClient.CreateClient(commonClientConfig().clientName("TEST_CLIENT_NAME").build()).get(); + + String clientInfo = (String) client.customCommand(new String[] {"CLIENT", "INFO"}).get(); + assertTrue(clientInfo.contains("name=TEST_CLIENT_NAME")); + + client.close(); + } }