diff --git a/glide-core/src/client/value_conversion.rs b/glide-core/src/client/value_conversion.rs index 347fa11a54..22bd4f7c46 100644 --- a/glide-core/src/client/value_conversion.rs +++ b/glide-core/src/client/value_conversion.rs @@ -683,12 +683,14 @@ pub(crate) fn expected_type_for_cmd(cmd: &Cmd) -> Option { // TODO use enum to avoid mistakes match command.as_slice() { - b"HGETALL" | b"CONFIG GET" | b"FT.CONFIG GET" | b"HELLO" | b"XRANGE" => { - Some(ExpectedReturnType::Map { - key_type: &None, - value_type: &None, - }) - } + b"HGETALL" | b"CONFIG GET" | b"FT.CONFIG GET" | b"HELLO" => Some(ExpectedReturnType::Map { + key_type: &None, + value_type: &None, + }), + b"XRANGE" | b"XREVRANGE" => Some(ExpectedReturnType::Map { + key_type: &Some(ExpectedReturnType::BulkString), + value_type: &Some(ExpectedReturnType::ArrayOfPairs), + }), b"XREAD" => Some(ExpectedReturnType::Map { key_type: &Some(ExpectedReturnType::BulkString), value_type: &Some(ExpectedReturnType::Map { @@ -988,6 +990,24 @@ mod tests { assert!(converted_4.is_err()); } + #[test] + fn convert_xrange_xrevrange() { + assert!(matches!( + expected_type_for_cmd(redis::cmd("XRANGE").arg("key").arg("start").arg("end")), + Some(ExpectedReturnType::Map { + key_type: &Some(ExpectedReturnType::BulkString), + value_type: &Some(ExpectedReturnType::ArrayOfPairs), + }) + )); + assert!(matches!( + expected_type_for_cmd(redis::cmd("XREVRANGE").arg("key").arg("end").arg("start")), + Some(ExpectedReturnType::Map { + key_type: &Some(ExpectedReturnType::BulkString), + value_type: &Some(ExpectedReturnType::ArrayOfPairs), + }) + )); + } + #[test] fn convert_xread() { assert!(matches!( diff --git a/glide-core/src/protobuf/redis_request.proto b/glide-core/src/protobuf/redis_request.proto index 7a8f3a2746..8b629ec619 100644 --- a/glide-core/src/protobuf/redis_request.proto +++ b/glide-core/src/protobuf/redis_request.proto @@ -210,6 +210,7 @@ enum RequestType { BitFieldReadOnly = 173; Move = 174; SInterCard = 175; + XRevRange = 176; Copy = 178; } diff --git a/glide-core/src/request_type.rs b/glide-core/src/request_type.rs index 8e20449c1d..176a461fe0 100644 --- a/glide-core/src/request_type.rs +++ b/glide-core/src/request_type.rs @@ -180,6 +180,7 @@ pub enum RequestType { BitFieldReadOnly = 173, Move = 174, SInterCard = 175, + XRevRange = 176, Copy = 178, } @@ -365,6 +366,7 @@ impl From<::protobuf::EnumOrUnknown> for RequestType { ProtobufRequestType::SInterCard => RequestType::SInterCard, ProtobufRequestType::Copy => RequestType::Copy, ProtobufRequestType::Sort => RequestType::Sort, + ProtobufRequestType::XRevRange => RequestType::XRevRange, } } } @@ -545,6 +547,7 @@ impl RequestType { RequestType::SInterCard => Some(cmd("SINTERCARD")), RequestType::Copy => Some(cmd("COPY")), RequestType::Sort => Some(cmd("SORT")), + RequestType::XRevRange => Some(cmd("XREVRANGE")), } } } diff --git a/java/client/src/main/java/glide/api/BaseClient.java b/java/client/src/main/java/glide/api/BaseClient.java index db038b9373..fc7fb6889b 100644 --- a/java/client/src/main/java/glide/api/BaseClient.java +++ b/java/client/src/main/java/glide/api/BaseClient.java @@ -118,6 +118,7 @@ import static redis_request.RedisRequestOuterClass.RequestType.XLen; import static redis_request.RedisRequestOuterClass.RequestType.XRange; import static redis_request.RedisRequestOuterClass.RequestType.XRead; +import static redis_request.RedisRequestOuterClass.RequestType.XRevRange; import static redis_request.RedisRequestOuterClass.RequestType.XTrim; import static redis_request.RedisRequestOuterClass.RequestType.ZAdd; import static redis_request.RedisRequestOuterClass.RequestType.ZCard; @@ -1322,19 +1323,39 @@ public CompletableFuture xdel(@NonNull String key, @NonNull String[] ids) } @Override - public CompletableFuture> xrange( + public CompletableFuture> xrange( @NonNull String key, @NonNull StreamRange start, @NonNull StreamRange end) { String[] arguments = ArrayUtils.addFirst(StreamRange.toArgs(start, end), key); return commandManager.submitNewCommand( - XRange, arguments, response -> castMapOfArrays(handleMapResponse(response), String.class)); + XRange, arguments, response -> castMapOf2DArray(handleMapResponse(response), String.class)); } @Override - public CompletableFuture> xrange( + public CompletableFuture> xrange( @NonNull String key, @NonNull StreamRange start, @NonNull StreamRange end, long count) { String[] arguments = ArrayUtils.addFirst(StreamRange.toArgs(start, end, count), key); return commandManager.submitNewCommand( - XRange, arguments, response -> castMapOfArrays(handleMapResponse(response), String.class)); + XRange, arguments, response -> castMapOf2DArray(handleMapResponse(response), String.class)); + } + + @Override + public CompletableFuture> xrevrange( + @NonNull String key, @NonNull StreamRange end, @NonNull StreamRange start) { + String[] arguments = ArrayUtils.addFirst(StreamRange.toArgs(end, start), key); + return commandManager.submitNewCommand( + XRevRange, + arguments, + response -> castMapOf2DArray(handleMapResponse(response), String.class)); + } + + @Override + public CompletableFuture> xrevrange( + @NonNull String key, @NonNull StreamRange end, @NonNull StreamRange start, long count) { + String[] arguments = ArrayUtils.addFirst(StreamRange.toArgs(end, start, count), key); + return commandManager.submitNewCommand( + XRevRange, + arguments, + response -> castMapOf2DArray(handleMapResponse(response), String.class)); } @Override diff --git a/java/client/src/main/java/glide/api/commands/StreamBaseCommands.java b/java/client/src/main/java/glide/api/commands/StreamBaseCommands.java index bdb9f000e1..dbe98bc6b3 100644 --- a/java/client/src/main/java/glide/api/commands/StreamBaseCommands.java +++ b/java/client/src/main/java/glide/api/commands/StreamBaseCommands.java @@ -184,24 +184,23 @@ CompletableFuture>> xread( *
  • Use {@link InfRangeBound#MAX} to end with the maximum available ID. * * - * @return @return A Map of key to stream entry data, where entry data is an array of - * item pairings. + * @return A Map of key to stream entry data, where entry data is an array of pairings with format [[field, entry], [field, entry], ...]. * @example *
    {@code
          * // Retrieve all stream entries
    -     * Map result = client.xrange("key", InfRangeBound.MIN, InfRangeBound.MAX).get();
    +     * Map result = client.xrange("key", InfRangeBound.MIN, InfRangeBound.MAX).get();
          * result.forEach((k, v) -> {
          *     System.out.println("Stream ID: " + k);
    -     *     for (int i = 0; i < v.length;) {
    -     *         System.out.println(v[i++] + ": " + v[i++]);
    +     *     for (int i = 0; i < v.length; i++) {
    +     *         System.out.println(v[i][0] + ": " + v[i][1]);
          *     }
          * });
          * // Retrieve exactly one stream entry by id
    -     * Map result = client.xrange("key", IdBound.of(streamId), IdBound.of(streamId)).get();
    +     * Map result = client.xrange("key", IdBound.of(streamId), IdBound.of(streamId)).get();
          * System.out.println("Stream ID: " + streamid + " -> " + Arrays.toString(result.get(streamid)));
          * }
    */ - CompletableFuture> xrange(String key, StreamRange start, StreamRange end); + CompletableFuture> xrange(String key, StreamRange start, StreamRange end); /** * Returns stream entries matching a given range of IDs. @@ -222,20 +221,95 @@ CompletableFuture>> xread( * * * @param count Maximum count of stream entries to return. - * @return A Map of key to stream entry data, where entry data is an array of item - * pairings. + * @return A Map of key to stream entry data, where entry data is an array of pairings with format [[field, entry], [field, entry], ...]. * @example *
    {@code
          * // Retrieve the first 2 stream entries
    -     * Map result = client.xrange("key", InfRangeBound.MIN, InfRangeBound.MAX, 2).get();
    +     * Map result = client.xrange("key", InfRangeBound.MIN, InfRangeBound.MAX, 2).get();
          * result.forEach((k, v) -> {
          *     System.out.println("Stream ID: " + k);
    -     *     for (int i = 0; i < v.length;) {
    -     *         System.out.println(v[i++] + ": " + v[i++]);
    +     *     for (int i = 0; i < v.length; i++) {
    +     *         System.out.println(v[i][0] + ": " + v[i][1]);
          *     }
          * });
          * }
    */ - CompletableFuture> xrange( + CompletableFuture> xrange( String key, StreamRange start, StreamRange end, long count); + + /** + * Returns stream entries matching a given range of IDs in reverse order.
    + * Equivalent to {@link #xrange(String, StreamRange, StreamRange)} but returns the entries in + * reverse order. + * + * @param key The key of the stream. + * @param end Ending stream ID bound for range. + *
      + *
    • Use {@link IdBound#of} to specify a stream ID. + *
    • Use {@link IdBound#ofExclusive} to specify an exclusive bounded stream ID. + *
    • Use {@link InfRangeBound#MAX} to end with the maximum available ID. + *
    + * + * @param start Starting stream ID bound for range. + *
      + *
    • Use {@link IdBound#of} to specify a stream ID. + *
    • Use {@link IdBound#ofExclusive} to specify an exclusive bounded stream ID. + *
    • Use {@link InfRangeBound#MIN} to start with the minimum available ID. + *
    + * + * @return A Map of key to stream entry data, where entry data is an array of pairings with format [[field, entry], [field, entry], ...]. + * @example + *
    {@code
    +     * // Retrieve all stream entries
    +     * Map result = client.xrevrange("key", InfRangeBound.MAX, InfRangeBound.MIN).get();
    +     * result.forEach((k, v) -> {
    +     *     System.out.println("Stream ID: " + k);
    +     *     for (int i = 0; i < v.length; i++) {
    +     *         System.out.println(v[i][0] + ": " + v[i][1]);
    +     *     }
    +     * });
    +     * // Retrieve exactly one stream entry by id
    +     * Map result = client.xrevrange("key", IdBound.of(streamId), IdBound.of(streamId)).get();
    +     * System.out.println("Stream ID: " + streamid + " -> " + Arrays.toString(result.get(streamid)));
    +     * }
    + */ + CompletableFuture> xrevrange( + String key, StreamRange end, StreamRange start); + + /** + * Returns stream entries matching a given range of IDs in reverse order.
    + * Equivalent to {@link #xrange(String, StreamRange, StreamRange, long)} but returns the entries + * in reverse order. + * + * @param key The key of the stream. + * @param end Ending stream ID bound for range. + *
      + *
    • Use {@link IdBound#of} to specify a stream ID. + *
    • Use {@link IdBound#ofExclusive} to specify an exclusive bounded stream ID. + *
    • Use {@link InfRangeBound#MAX} to end with the maximum available ID. + *
    + * + * @param start Starting stream ID bound for range. + *
      + *
    • Use {@link IdBound#of} to specify a stream ID. + *
    • Use {@link IdBound#ofExclusive} to specify an exclusive bounded stream ID. + *
    • Use {@link InfRangeBound#MIN} to start with the minimum available ID. + *
    + * + * @param count Maximum count of stream entries to return. + * @return A Map of key to stream entry data, where entry data is an array of pairings with format [[field, entry], [field, entry], ...]. + * @example + *
    {@code
    +     * // Retrieve the first 2 stream entries
    +     * Map result = client.xrange("key", InfRangeBound.MAX, InfRangeBound.MIN, 2).get();
    +     * result.forEach((k, v) -> {
    +     *     System.out.println("Stream ID: " + k);
    +     *     for (int i = 0; i < v.length; i++) {
    +     *         System.out.println(v[i][0] + ": " + v[i][1]);
    +     *     }
    +     * });
    +     * }
    + */ + CompletableFuture> xrevrange( + String key, StreamRange end, StreamRange start, long count); } diff --git a/java/client/src/main/java/glide/api/models/BaseTransaction.java b/java/client/src/main/java/glide/api/models/BaseTransaction.java index e61e5a146c..98a89acce9 100644 --- a/java/client/src/main/java/glide/api/models/BaseTransaction.java +++ b/java/client/src/main/java/glide/api/models/BaseTransaction.java @@ -143,6 +143,7 @@ import static redis_request.RedisRequestOuterClass.RequestType.XLen; import static redis_request.RedisRequestOuterClass.RequestType.XRange; import static redis_request.RedisRequestOuterClass.RequestType.XRead; +import static redis_request.RedisRequestOuterClass.RequestType.XRevRange; import static redis_request.RedisRequestOuterClass.RequestType.XTrim; import static redis_request.RedisRequestOuterClass.RequestType.ZAdd; import static redis_request.RedisRequestOuterClass.RequestType.ZCard; @@ -2742,8 +2743,7 @@ public T xdel(@NonNull String key, @NonNull String[] ids) { *
  • Use {@link StreamRange.InfRangeBound#MAX} to end with the maximum available ID. * * - * @return Command Response - A Map of key to stream entry data, where entry data is - * an array of item pairings. + * @return Command Response - A Map of key to stream entry data, where entry data is an array of pairings with format [[field, entry], [field, entry], ...]. */ public T xrange(@NonNull String key, @NonNull StreamRange start, @NonNull StreamRange end) { ArgsArray commandArgs = buildArgs(ArrayUtils.addFirst(StreamRange.toArgs(start, end), key)); @@ -2772,8 +2772,7 @@ public T xrange(@NonNull String key, @NonNull StreamRange start, @NonNull Stream * * * @param count Maximum count of stream entries to return. - * @return Command Response - A Map of key to stream entry data, where entry data is - * an array of item pairings. + * @return Command Response - A Map of key to stream entry data, where entry data is an array of pairings with format [[field, entry], [field, entry], ...]. */ public T xrange( @NonNull String key, @NonNull StreamRange start, @NonNull StreamRange end, long count) { @@ -2783,6 +2782,69 @@ public T xrange( return getThis(); } + /** + * Returns stream entries matching a given range of IDs in reverse order.
    + * Equivalent to {@link #xrange(String, StreamRange, StreamRange)} but returns the entries in + * reverse order. + * + * @param key The key of the stream. + * @param end Ending stream ID bound for range. + *
      + *
    • Use {@link StreamRange.IdBound#of} to specify a stream ID. + *
    • Use {@link StreamRange.IdBound#ofExclusive} to specify an exclusive bounded stream + * ID. + *
    • Use {@link StreamRange.InfRangeBound#MAX} to end with the maximum available ID. + *
    + * + * @param start Starting stream ID bound for range. + *
      + *
    • Use {@link StreamRange.IdBound#of} to specify a stream ID. + *
    • Use {@link StreamRange.IdBound#ofExclusive} to specify an exclusive bounded stream + * ID. + *
    • Use {@link StreamRange.InfRangeBound#MIN} to start with the minimum available ID. + *
    + * + * @return Command Response - A Map of key to stream entry data, where entry data is an array of pairings with format [[field, entry], [field, entry], ...]. + */ + public T xrevrange(@NonNull String key, @NonNull StreamRange end, @NonNull StreamRange start) { + ArgsArray commandArgs = buildArgs(ArrayUtils.addFirst(StreamRange.toArgs(end, start), key)); + protobufTransaction.addCommands(buildCommand(XRevRange, commandArgs)); + return getThis(); + } + + /** + * Returns stream entries matching a given range of IDs in reverse order.
    + * Equivalent to {@link #xrange(String, StreamRange, StreamRange, long)} but returns the entries + * in reverse order. + * + * @param key The key of the stream. + * @param start Starting stream ID bound for range. + *
      + *
    • Use {@link StreamRange.IdBound#of} to specify a stream ID. + *
    • Use {@link StreamRange.IdBound#ofExclusive} to specify an exclusive bounded stream + * ID. + *
    • Use {@link StreamRange.InfRangeBound#MIN} to start with the minimum available ID. + *
    + * + * @param end Ending stream ID bound for range. + *
      + *
    • Use {@link StreamRange.IdBound#of} to specify a stream ID. + *
    • Use {@link StreamRange.IdBound#ofExclusive} to specify an exclusive bounded stream + * ID. + *
    • Use {@link StreamRange.InfRangeBound#MAX} to end with the maximum available ID. + *
    + * + * @param count Maximum count of stream entries to return. + * @return Command Response - A Map of key to stream entry data, where entry data is an array of pairings with format [[field, entry], [field, entry], ...]. + */ + public T xrevrange( + @NonNull String key, @NonNull StreamRange end, @NonNull StreamRange start, long count) { + ArgsArray commandArgs = + buildArgs(ArrayUtils.addFirst(StreamRange.toArgs(end, start, count), key)); + protobufTransaction.addCommands(buildCommand(XRevRange, commandArgs)); + return getThis(); + } + /** * Returns the remaining time to live of key that has a timeout, in milliseconds. * diff --git a/java/client/src/test/java/glide/api/RedisClientTest.java b/java/client/src/test/java/glide/api/RedisClientTest.java index 195940e30d..825290341f 100644 --- a/java/client/src/test/java/glide/api/RedisClientTest.java +++ b/java/client/src/test/java/glide/api/RedisClientTest.java @@ -176,6 +176,7 @@ import static redis_request.RedisRequestOuterClass.RequestType.XLen; import static redis_request.RedisRequestOuterClass.RequestType.XRange; import static redis_request.RedisRequestOuterClass.RequestType.XRead; +import static redis_request.RedisRequestOuterClass.RequestType.XRevRange; import static redis_request.RedisRequestOuterClass.RequestType.XTrim; import static redis_request.RedisRequestOuterClass.RequestType.ZAdd; import static redis_request.RedisRequestOuterClass.RequestType.ZCard; @@ -4120,20 +4121,20 @@ public void xrange_returns_success() { String key = "testKey"; StreamRange start = IdBound.of(9999L); StreamRange end = IdBound.ofExclusive("696969-10"); - Map completedResult = - Map.of(key, new String[] {"duration", "12345", "event-id", "2", "user-id", "42"}); + String[][] fieldValuesResult = {{"duration", "12345"}, {"event-id", "2"}, {"user-id", "42"}}; + Map completedResult = Map.of(key, fieldValuesResult); - CompletableFuture> testResponse = new CompletableFuture<>(); + CompletableFuture> testResponse = new CompletableFuture<>(); testResponse.complete(completedResult); // match on protobuf request - when(commandManager.>submitNewCommand( + when(commandManager.>submitNewCommand( eq(XRange), eq(new String[] {key, "9999", "(696969-10"}), any())) .thenReturn(testResponse); // exercise - CompletableFuture> response = service.xrange(key, start, end); - Map payload = response.get(); + CompletableFuture> response = service.xrange(key, start, end); + Map payload = response.get(); // verify assertEquals(testResponse, response); @@ -4148,14 +4149,14 @@ public void xrange_withcount_returns_success() { StreamRange start = InfRangeBound.MIN; StreamRange end = InfRangeBound.MAX; long count = 99L; - Map completedResult = - Map.of(key, new String[] {"duration", "12345", "event-id", "2", "user-id", "42"}); + String[][] fieldValuesResult = {{"duration", "12345"}, {"event-id", "2"}, {"user-id", "42"}}; + Map completedResult = Map.of(key, fieldValuesResult); - CompletableFuture> testResponse = new CompletableFuture<>(); + CompletableFuture> testResponse = new CompletableFuture<>(); testResponse.complete(completedResult); // match on protobuf request - when(commandManager.>submitNewCommand( + when(commandManager.>submitNewCommand( eq(XRange), eq( new String[] { @@ -4169,8 +4170,72 @@ public void xrange_withcount_returns_success() { .thenReturn(testResponse); // exercise - CompletableFuture> response = service.xrange(key, start, end, count); - Map payload = response.get(); + CompletableFuture> response = service.xrange(key, start, end, count); + Map payload = response.get(); + + // verify + assertEquals(testResponse, response); + assertEquals(completedResult, payload); + } + + @Test + @SneakyThrows + public void xrevrange_returns_success() { + // setup + String key = "testKey"; + StreamRange end = IdBound.of(9999L); + StreamRange start = IdBound.ofExclusive("696969-10"); + String[][] fieldValuesResult = {{"duration", "12345"}, {"event-id", "2"}, {"user-id", "42"}}; + Map completedResult = Map.of(key, fieldValuesResult); + + CompletableFuture> testResponse = new CompletableFuture<>(); + testResponse.complete(completedResult); + + // match on protobuf request + when(commandManager.>submitNewCommand( + eq(XRevRange), eq(new String[] {key, "9999", "(696969-10"}), any())) + .thenReturn(testResponse); + + // exercise + CompletableFuture> response = service.xrevrange(key, end, start); + Map payload = response.get(); + + // verify + assertEquals(testResponse, response); + assertEquals(completedResult, payload); + } + + @Test + @SneakyThrows + public void xrevrange_withcount_returns_success() { + // setup + String key = "testKey"; + StreamRange end = InfRangeBound.MAX; + StreamRange start = InfRangeBound.MIN; + long count = 99L; + String[][] fieldValuesResult = {{"duration", "12345"}, {"event-id", "2"}, {"user-id", "42"}}; + Map completedResult = Map.of(key, fieldValuesResult); + + CompletableFuture> testResponse = new CompletableFuture<>(); + testResponse.complete(completedResult); + + // match on protobuf request + when(commandManager.>submitNewCommand( + eq(XRevRange), + eq( + new String[] { + key, + MAXIMUM_RANGE_REDIS_API, + MINIMUM_RANGE_REDIS_API, + RANGE_COUNT_REDIS_API, + Long.toString(count) + }), + any())) + .thenReturn(testResponse); + + // exercise + CompletableFuture> response = service.xrevrange(key, end, start, count); + Map payload = response.get(); // verify assertEquals(testResponse, response); diff --git a/java/client/src/test/java/glide/api/models/TransactionTests.java b/java/client/src/test/java/glide/api/models/TransactionTests.java index 2454e210d7..04d4b8f702 100644 --- a/java/client/src/test/java/glide/api/models/TransactionTests.java +++ b/java/client/src/test/java/glide/api/models/TransactionTests.java @@ -156,6 +156,7 @@ import static redis_request.RedisRequestOuterClass.RequestType.XLen; import static redis_request.RedisRequestOuterClass.RequestType.XRange; import static redis_request.RedisRequestOuterClass.RequestType.XRead; +import static redis_request.RedisRequestOuterClass.RequestType.XRevRange; import static redis_request.RedisRequestOuterClass.RequestType.XTrim; import static redis_request.RedisRequestOuterClass.RequestType.ZAdd; import static redis_request.RedisRequestOuterClass.RequestType.ZCard; @@ -715,6 +716,21 @@ InfScoreBound.NEGATIVE_INFINITY, new ScoreBoundary(3, false), new Limit(1, 2)), RANGE_COUNT_REDIS_API, "99"))); + transaction.xrevrange("key", InfRangeBound.MAX, InfRangeBound.MIN); + results.add( + Pair.of(XRevRange, buildArgs("key", MAXIMUM_RANGE_REDIS_API, MINIMUM_RANGE_REDIS_API))); + + transaction.xrevrange("key", InfRangeBound.MAX, InfRangeBound.MIN, 99L); + results.add( + Pair.of( + XRevRange, + buildArgs( + "key", + MAXIMUM_RANGE_REDIS_API, + MINIMUM_RANGE_REDIS_API, + RANGE_COUNT_REDIS_API, + "99"))); + transaction.time(); results.add(Pair.of(Time, buildArgs())); diff --git a/java/integTest/src/test/java/glide/SharedCommandTests.java b/java/integTest/src/test/java/glide/SharedCommandTests.java index 49a35ed4c2..90c874351a 100644 --- a/java/integTest/src/test/java/glide/SharedCommandTests.java +++ b/java/integTest/src/test/java/glide/SharedCommandTests.java @@ -3183,7 +3183,7 @@ public void xdel(BaseClient client) { @SneakyThrows @ParameterizedTest(autoCloseArguments = false) @MethodSource("getClients") - public void xrange(BaseClient client) { + public void xrange_and_xrevrange(BaseClient client) { String key = UUID.randomUUID().toString(); String key2 = UUID.randomUUID().toString(); @@ -3210,16 +3210,28 @@ public void xrange(BaseClient client) { assertEquals(2L, client.xlen(key).get()); // get everything from the stream - Map result = client.xrange(key, InfRangeBound.MIN, InfRangeBound.MAX).get(); + Map result = client.xrange(key, InfRangeBound.MIN, InfRangeBound.MAX).get(); assertEquals(2, result.size()); assertNotNull(result.get(streamId1)); assertNotNull(result.get(streamId2)); + // get everything from the stream using a reverse range search + Map revResult = + client.xrevrange(key, InfRangeBound.MAX, InfRangeBound.MIN).get(); + assertEquals(2, revResult.size()); + assertNotNull(revResult.get(streamId1)); + assertNotNull(revResult.get(streamId2)); + // returns empty if + before - - Map emptyResult = + Map emptyResult = client.xrange(key, InfRangeBound.MAX, InfRangeBound.MIN).get(); assertEquals(0, emptyResult.size()); + // rev search returns empty if - before + + Map emptyRevResult = + client.xrevrange(key, InfRangeBound.MIN, InfRangeBound.MAX).get(); + assertEquals(0, emptyRevResult.size()); + assertEquals( streamId3, client @@ -3230,28 +3242,48 @@ public void xrange(BaseClient client) { .get()); // get the newest entry - Map newResult = + Map newResult = client.xrange(key, IdBound.ofExclusive(streamId2), IdBound.ofExclusive(5), 1L).get(); assertEquals(1, newResult.size()); assertNotNull(newResult.get(streamId3)); + // ...and from xrevrange + Map newRevResult = + client.xrevrange(key, IdBound.ofExclusive(5), IdBound.ofExclusive(streamId2), 1L).get(); + assertEquals(1, newRevResult.size()); + assertNotNull(newRevResult.get(streamId3)); // xrange against an emptied stream assertEquals(3, client.xdel(key, new String[] {streamId1, streamId2, streamId3}).get()); - Map emptiedResult = + Map emptiedResult = client.xrange(key, InfRangeBound.MIN, InfRangeBound.MAX, 10L).get(); assertEquals(0, emptiedResult.size()); + // ...and xrevrange + Map emptiedRevResult = + client.xrevrange(key, InfRangeBound.MAX, InfRangeBound.MIN, 10L).get(); + assertEquals(0, emptiedRevResult.size()); // xrange against a non-existent stream emptyResult = client.xrange(key2, InfRangeBound.MIN, InfRangeBound.MAX).get(); assertEquals(0, emptyResult.size()); + // ...and xrevrange + emptiedRevResult = client.xrevrange(key2, InfRangeBound.MAX, InfRangeBound.MIN).get(); + assertEquals(0, emptiedRevResult.size()); + // xrange against a non-stream value assertEquals(OK, client.set(key2, "not_a_stream").get()); ExecutionException executionException = assertThrows( ExecutionException.class, () -> client.xrange(key2, InfRangeBound.MIN, InfRangeBound.MAX).get()); assertInstanceOf(RequestException.class, executionException.getCause()); + // ...and xrevrange + executionException = + assertThrows( + ExecutionException.class, + () -> client.xrevrange(key2, InfRangeBound.MAX, InfRangeBound.MIN).get()); + assertInstanceOf(RequestException.class, executionException.getCause()); + // xrange when range bound is not valid ID executionException = assertThrows( ExecutionException.class, @@ -3269,6 +3301,26 @@ public void xrange(BaseClient client) { .xrange(key, InfRangeBound.MIN, IdBound.ofExclusive("not_a_stream_id")) .get()); assertInstanceOf(RequestException.class, executionException.getCause()); + + // ... and xrevrange + + executionException = + assertThrows( + ExecutionException.class, + () -> + client + .xrevrange(key, IdBound.ofExclusive("not_a_stream_id"), InfRangeBound.MIN) + .get()); + assertInstanceOf(RequestException.class, executionException.getCause()); + + executionException = + assertThrows( + ExecutionException.class, + () -> + client + .xrevrange(key, InfRangeBound.MAX, IdBound.ofExclusive("not_a_stream_id")) + .get()); + assertInstanceOf(RequestException.class, executionException.getCause()); } @SneakyThrows diff --git a/java/integTest/src/test/java/glide/TransactionTestUtilities.java b/java/integTest/src/test/java/glide/TransactionTestUtilities.java index a6abc8ce87..51d5037ae3 100644 --- a/java/integTest/src/test/java/glide/TransactionTestUtilities.java +++ b/java/integTest/src/test/java/glide/TransactionTestUtilities.java @@ -668,6 +668,8 @@ private static Object[] streamCommands(BaseTransaction transaction) { .xread(Map.of(streamKey1, "0-2")) .xrange(streamKey1, IdBound.of("0-1"), IdBound.of("0-1")) .xrange(streamKey1, IdBound.of("0-1"), IdBound.of("0-1"), 1L) + .xrevrange(streamKey1, IdBound.of("0-1"), IdBound.of("0-1")) + .xrevrange(streamKey1, IdBound.of("0-1"), IdBound.of("0-1"), 1L) .xtrim(streamKey1, new MinId(true, "0-2")) .xdel(streamKey1, new String[] {"0-3", "0-5"}); @@ -679,8 +681,11 @@ private static Object[] streamCommands(BaseTransaction transaction) { Map.of( streamKey1, Map.of("0-3", new String[][] {{"field3", "value3"}})), // xread(Map.of(key9, "0-2")); - Map.of("0-1", new String[] {"field1", "value1"}), // .xrange(streamKey1, "0-1", "0-1") - Map.of("0-1", new String[] {"field1", "value1"}), // .xrange(streamKey1, "0-1", "0-1", 1l) + Map.of("0-1", new String[][] {{"field1", "value1"}}), // .xrange(streamKey1, "0-1", "0-1") + Map.of("0-1", new String[][] {{"field1", "value1"}}), // .xrange(streamKey1, "0-1", "0-1", 1l) + Map.of("0-1", new String[][] {{"field1", "value1"}}), // .xrevrange(streamKey1, "0-1", "0-1") + Map.of( + "0-1", new String[][] {{"field1", "value1"}}), // .xrevrange(streamKey1, "0-1", "0-1", 1l) 1L, // xtrim(streamKey1, new MinId(true, "0-2")) 1L, // .xdel(streamKey1, new String[] {"0-1", "0-5"}); };