Skip to content

Commit

Permalink
fix: Lcs response parse (redis#2970)
Browse files Browse the repository at this point in the history
* fix: Lcs response parse

* Handle parsing `len` and `matches` in any order;

* Increase test coverage

* Add more unit tests to handle the different cases;
* Add integration tests when the `LCS` command is available;
* Update the StringMatchResultOutput to avoid keeping track of withIdx
  parameter. This allows the use in the command factory.
  • Loading branch information
jabolina authored and thachlp committed Dec 31, 2024
1 parent 30e91f5 commit c1f0bcc
Show file tree
Hide file tree
Showing 5 changed files with 269 additions and 27 deletions.
2 changes: 1 addition & 1 deletion src/main/java/io/lettuce/core/RedisCommandBuilder.java
Original file line number Diff line number Diff line change
Expand Up @@ -2909,7 +2909,7 @@ Command<K, V, StringMatchResult> stralgoLcs(StrAlgoArgs strAlgoArgs) {

CommandArgs<K, V> args = new CommandArgs<>(codec);
strAlgoArgs.build(args);
return createCommand(STRALGO, new StringMatchResultOutput<>(codec, strAlgoArgs.isWithIdx()), args);
return createCommand(STRALGO, new StringMatchResultOutput<>(codec), args);
}

Command<K, V, Set<V>> sunion(K... keys) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,8 @@ public class OutputRegistry {
register(registry, StringListOutput.class, StringListOutput::new);
register(registry, VoidOutput.class, VoidOutput::new);

register(registry, StringMatchResultOutput.class, StringMatchResultOutput::new);

BUILTIN.putAll(registry);
}

Expand Down
36 changes: 19 additions & 17 deletions src/main/java/io/lettuce/core/output/StringMatchResultOutput.java
Original file line number Diff line number Diff line change
Expand Up @@ -19,15 +19,16 @@
*/
package io.lettuce.core.output;

import static io.lettuce.core.StringMatchResult.MatchedPosition;
import static io.lettuce.core.StringMatchResult.Position;
import io.lettuce.core.StringMatchResult;
import io.lettuce.core.codec.RedisCodec;

import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.List;

import io.lettuce.core.StringMatchResult;
import io.lettuce.core.codec.RedisCodec;
import static io.lettuce.core.StringMatchResult.MatchedPosition;
import static io.lettuce.core.StringMatchResult.Position;

/**
* Command output for {@code STRALGO} returning {@link StringMatchResult}.
Expand All @@ -37,38 +38,39 @@
*/
public class StringMatchResultOutput<K, V> extends CommandOutput<K, V, StringMatchResult> {

private final boolean withIdx;
private static final ByteBuffer LEN = StandardCharsets.US_ASCII.encode("len");

private String matchString;

private int len;

private List<Long> positions;

private boolean readingLen = true;

private final List<MatchedPosition> matchedPositions = new ArrayList<>();

public StringMatchResultOutput(RedisCodec<K, V> codec, boolean withIdx) {
public StringMatchResultOutput(RedisCodec<K, V> codec) {
super(codec, null);
this.withIdx = withIdx;
}

@Override
public void set(ByteBuffer bytes) {

if (!withIdx && matchString == null) {
matchString = (String) codec.decodeKey(bytes);
}
matchString = (String) codec.decodeKey(bytes);
readingLen = LEN.equals(bytes);
}

@Override
public void set(long integer) {

this.len = (int) integer;

if (positions == null) {
positions = new ArrayList<>();
if (readingLen) {
this.len = (int) integer;
} else {
if (positions == null) {
positions = new ArrayList<>();
}
positions.add(integer);
}
positions.add(integer);
matchString = null;
}

@Override
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,9 +21,11 @@

import static io.lettuce.TestTags.INTEGRATION_TEST;
import static io.lettuce.core.SetArgs.Builder.*;
import static io.lettuce.core.StringMatchResult.*;
import static org.assertj.core.api.Assertions.*;
import static io.lettuce.core.StringMatchResult.Position;
import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.api.Assertions.assertThatThrownBy;

import java.lang.reflect.Proxy;
import java.time.Duration;
import java.time.Instant;
import java.util.LinkedHashMap;
Expand All @@ -32,20 +34,21 @@

import javax.inject.Inject;

import org.junit.jupiter.api.Assumptions;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Tag;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestInstance;
import org.junit.jupiter.api.extension.ExtendWith;

import io.lettuce.core.GetExArgs;
import io.lettuce.core.KeyValue;
import io.lettuce.core.RedisException;
import io.lettuce.core.SetArgs;
import io.lettuce.core.StrAlgoArgs;
import io.lettuce.core.StringMatchResult;
import io.lettuce.core.TestSupport;
import io.lettuce.core.*;
import io.lettuce.core.api.StatefulConnection;
import io.lettuce.core.api.StatefulRedisConnection;
import io.lettuce.core.api.sync.RedisCommands;
import io.lettuce.core.dynamic.Commands;
import io.lettuce.core.dynamic.RedisCommandFactory;
import io.lettuce.core.dynamic.annotation.Command;
import io.lettuce.core.dynamic.annotation.Param;
import io.lettuce.test.KeyValueStreamingAdapter;
import io.lettuce.test.LettuceExtension;
import io.lettuce.test.condition.EnabledOnCommand;
Expand Down Expand Up @@ -376,4 +379,117 @@ void strAlgoWithIdx() {
assertThat(matchResult.getLen()).isEqualTo(6);
}

@Test
@EnabledOnCommand("LCS")
void lcs() {
redis.set("key1", "ohmytext");
redis.set("key2", "mynewtext");

// LCS key1 key2
CustomStringCommands commands = CustomStringCommands.instance(getConnection());
StringMatchResult matchResult = commands.lcs("key1", "key2");
assertThat(matchResult.getMatchString()).isEqualTo("mytext");

// LCS a b IDX MINMATCHLEN 4 WITHMATCHLEN
// Keys don't exist.
matchResult = commands.lcsMinMatchLenWithMatchLen("a", "b", 4);
assertThat(matchResult.getMatchString()).isNullOrEmpty();
assertThat(matchResult.getLen()).isEqualTo(0);
}

@Test
@EnabledOnCommand("LCS")
void lcsUsingKeys() {

redis.set("key1{k}", "ohmytext");
redis.set("key2{k}", "mynewtext");

CustomStringCommands commands = CustomStringCommands.instance(getConnection());

StringMatchResult matchResult = commands.lcs("key1{k}", "key2{k}");
assertThat(matchResult.getMatchString()).isEqualTo("mytext");

// STRALGO LCS STRINGS a b
matchResult = commands.lcsMinMatchLenWithMatchLen("a", "b", 4);
assertThat(matchResult.getMatchString()).isNullOrEmpty();
assertThat(matchResult.getLen()).isEqualTo(0);
}

@Test
@EnabledOnCommand("LCS")
void lcsJustLen() {
redis.set("one", "ohmytext");
redis.set("two", "mynewtext");

CustomStringCommands commands = CustomStringCommands.instance(getConnection());

StringMatchResult matchResult = commands.lcsLen("one", "two");

assertThat(matchResult.getLen()).isEqualTo(6);
}

@Test
@EnabledOnCommand("LCS")
void lcsWithMinMatchLen() {
redis.set("key1", "ohmytext");
redis.set("key2", "mynewtext");

CustomStringCommands commands = CustomStringCommands.instance(getConnection());

StringMatchResult matchResult = commands.lcsMinMatchLen("key1", "key2", 4);

assertThat(matchResult.getMatchString()).isEqualTo("mytext");
}

@Test
@EnabledOnCommand("LCS")
void lcsMinMatchLenIdxMatchLen() {
redis.set("key1", "ohmytext");
redis.set("key2", "mynewtext");

CustomStringCommands commands = CustomStringCommands.instance(getConnection());

// LCS key1 key2 IDX MINMATCHLEN 4 WITHMATCHLEN
StringMatchResult matchResult = commands.lcsMinMatchLenWithMatchLen("key1", "key2", 4);

assertThat(matchResult.getMatches()).hasSize(1);
assertThat(matchResult.getMatches().get(0).getMatchLen()).isEqualTo(4);

Position a = matchResult.getMatches().get(0).getA();
Position b = matchResult.getMatches().get(0).getB();

assertThat(a.getStart()).isEqualTo(4);
assertThat(a.getEnd()).isEqualTo(7);
assertThat(b.getStart()).isEqualTo(5);
assertThat(b.getEnd()).isEqualTo(8);
assertThat(matchResult.getLen()).isEqualTo(6);
}

protected StatefulConnection<String, String> getConnection() {
StatefulRedisConnection<String, String> src = redis.getStatefulConnection();
Assumptions.assumeFalse(Proxy.isProxyClass(src.getClass()), "Redis connection is proxy, skipping.");
return src;
}

private interface CustomStringCommands extends Commands {

@Command("LCS :k1 :k2")
StringMatchResult lcs(@Param("k1") String k1, @Param("k2") String k2);

@Command("LCS :k1 :k2 LEN")
StringMatchResult lcsLen(@Param("k1") String k1, @Param("k2") String k2);

@Command("LCS :k1 :k2 MINMATCHLEN :mml")
StringMatchResult lcsMinMatchLen(@Param("k1") String k1, @Param("k2") String k2, @Param("mml") int mml);

@Command("LCS :k1 :k2 IDX MINMATCHLEN :mml WITHMATCHLEN")
StringMatchResult lcsMinMatchLenWithMatchLen(@Param("k1") String k1, @Param("k2") String k2, @Param("mml") int mml);

static CustomStringCommands instance(StatefulConnection<String, String> conn) {
RedisCommandFactory factory = new RedisCommandFactory(conn);
return factory.getCommands(CustomStringCommands.class);
}

}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,122 @@
package io.lettuce.core.output;

import static org.assertj.core.api.Assertions.assertThat;

import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import io.netty.buffer.Unpooled;
import org.junit.jupiter.api.Test;

import io.lettuce.core.StringMatchResult;
import io.lettuce.core.codec.StringCodec;
import io.lettuce.core.protocol.ProtocolVersion;
import io.lettuce.core.protocol.RedisStateMachine;

public class StringMatchResultOutputUnitTests {

@Test
void parseManually() {
byte[] rawOne = "%2\r\n$7\r\nmatches\r\n*1\r\n*3\r\n*2\r\n:4\r\n:7\r\n*2\r\n:5\r\n:8\r\n:4\r\n$3\r\nlen\r\n:6\r\n"
.getBytes(StandardCharsets.US_ASCII);
byte[] rawTwo = "%2\r\n$3\r\nlen\r\n:6\r\n$7\r\nmatches\r\n*1\r\n*3\r\n*2\r\n:4\r\n:7\r\n*2\r\n:5\r\n:8\r\n:4\r\n"
.getBytes(StandardCharsets.US_ASCII);
RedisStateMachine rsm = new RedisStateMachine();
rsm.setProtocolVersion(ProtocolVersion.RESP3);

StringMatchResultOutput<String, String> o1 = new StringMatchResultOutput<>(StringCodec.ASCII);
assertThat(rsm.decode(Unpooled.wrappedBuffer(rawOne), o1)).isTrue();

StringMatchResultOutput<String, String> o2 = new StringMatchResultOutput<>(StringCodec.ASCII);
assertThat(rsm.decode(Unpooled.wrappedBuffer(rawTwo), o2)).isTrue();

Map<String, Object> res1 = transform(o1.get());
Map<String, Object> res2 = transform(o2.get());

assertThat(res1).isEqualTo(res2);
}

private Map<String, Object> transform(StringMatchResult result) {
Map<String, Object> obj = new HashMap<>();
List<Object> matches = new ArrayList<>();
for (StringMatchResult.MatchedPosition match : result.getMatches()) {
Map<String, Object> intra = new HashMap<>();
Map<String, Object> a = new HashMap<>();
Map<String, Object> b = new HashMap<>();
a.put("start", match.getA().getStart());
a.put("end", match.getA().getEnd());

b.put("start", match.getB().getStart());
b.put("end", match.getB().getEnd());
intra.put("a", a);
intra.put("b", b);
intra.put("matchLen", match.getMatchLen());
matches.add(intra);
}
obj.put("matches", matches);
obj.put("len", result.getLen());
return obj;
}

@Test
void parseOnlyStringMatch() {
StringMatchResultOutput<String, String> output = new StringMatchResultOutput<>(StringCodec.ASCII);

String matchString = "some-string";
output.set(ByteBuffer.wrap(matchString.getBytes()));
output.complete(0);

StringMatchResult result = output.get();
assertThat(result.getMatchString()).isEqualTo(matchString);
assertThat(result.getMatches()).isEmpty();
assertThat(result.getLen()).isZero();
}

@Test
void parseOnlyLen() {
StringMatchResultOutput<String, String> output = new StringMatchResultOutput<>(StringCodec.ASCII);

output.set(42);
output.complete(0);

StringMatchResult result = output.get();
assertThat(result.getMatchString()).isNull();
assertThat(result.getMatches()).isEmpty();
assertThat(result.getLen()).isEqualTo(42);
}

@Test
void parseLenAndMatchesWithIdx() {
StringMatchResultOutput<String, String> output = new StringMatchResultOutput<>(StringCodec.ASCII);

output.set(ByteBuffer.wrap("len".getBytes()));
output.set(42);

output.set(ByteBuffer.wrap("matches".getBytes()));
output.set(0);
output.set(5);
output.set(10);
output.set(15);

output.complete(2);
output.complete(0);

StringMatchResult result = output.get();

assertThat(result.getMatchString()).isNull();
assertThat(result.getLen()).isEqualTo(42);
assertThat(result.getMatches()).hasSize(1).satisfies(m -> assertMatchedPositions(m.get(0), 0, 5, 10, 15));
}

private void assertMatchedPositions(StringMatchResult.MatchedPosition match, int... expected) {
assertThat(match.getA().getStart()).isEqualTo(expected[0]);
assertThat(match.getA().getEnd()).isEqualTo(expected[1]);
assertThat(match.getB().getStart()).isEqualTo(expected[2]);
assertThat(match.getB().getEnd()).isEqualTo(expected[3]);
}

}

0 comments on commit c1f0bcc

Please sign in to comment.