Skip to content

Commit

Permalink
further bitset fixes + native test
Browse files Browse the repository at this point in the history
  • Loading branch information
inponomarev committed Oct 27, 2024
1 parent 28008f9 commit a7daeee
Show file tree
Hide file tree
Showing 7 changed files with 113 additions and 85 deletions.
4 changes: 4 additions & 0 deletions .github/workflows/native-tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,10 @@ jobs:
- name: chmod
run: chmod +x native_tests/testnative

- name: string
run: ./testnative unit/type/string
working-directory: native_tests

- name: incr
run: ./testnative unit/type/incr
working-directory: native_tests
Expand Down
130 changes: 65 additions & 65 deletions native_tests/linux/tests/unit/type/string.tcl
Original file line number Diff line number Diff line change
Expand Up @@ -337,47 +337,47 @@ start_server {tags {"string"}} {
assert_error "*maximum allowed size*" {r setrange mykey [expr 512*1024*1024-4] world}
}

test "GETRANGE against non-existing key" {
r del mykey
assert_equal "" [r getrange mykey 0 -1]
}

test "GETRANGE against string value" {
r set mykey "Hello World"
assert_equal "Hell" [r getrange mykey 0 3]
assert_equal "Hello World" [r getrange mykey 0 -1]
assert_equal "orld" [r getrange mykey -4 -1]
assert_equal "" [r getrange mykey 5 3]
assert_equal " World" [r getrange mykey 5 5000]
assert_equal "Hello World" [r getrange mykey -5000 10000]
}

test "GETRANGE against integer-encoded value" {
r set mykey 1234
assert_equal "123" [r getrange mykey 0 2]
assert_equal "1234" [r getrange mykey 0 -1]
assert_equal "234" [r getrange mykey -3 -1]
assert_equal "" [r getrange mykey 5 3]
assert_equal "4" [r getrange mykey 3 5000]
assert_equal "1234" [r getrange mykey -5000 10000]
}

test "GETRANGE fuzzing" {
for {set i 0} {$i < 1000} {incr i} {
r set bin [set bin [randstring 0 1024 binary]]
set _start [set start [randomInt 1500]]
set _end [set end [randomInt 1500]]
if {$_start < 0} {set _start "end-[abs($_start)-1]"}
if {$_end < 0} {set _end "end-[abs($_end)-1]"}
assert_equal [string range $bin $_start $_end] [r getrange bin $start $end]
}
}

test {Extended SET can detect syntax errors} {
set e {}
catch {r set foo bar non-existing-option} e
set e
} {*syntax*}
#test "GETRANGE against non-existing key" {
# r del mykey
# assert_equal "" [r getrange mykey 0 -1]
#}

#test "GETRANGE against string value" {
# r set mykey "Hello World"
# assert_equal "Hell" [r getrange mykey 0 3]
# assert_equal "Hello World" [r getrange mykey 0 -1]
# assert_equal "orld" [r getrange mykey -4 -1]
# assert_equal "" [r getrange mykey 5 3]
# assert_equal " World" [r getrange mykey 5 5000]
# assert_equal "Hello World" [r getrange mykey -5000 10000]
#}

#test "GETRANGE against integer-encoded value" {
# r set mykey 1234
# assert_equal "123" [r getrange mykey 0 2]
# assert_equal "1234" [r getrange mykey 0 -1]
# assert_equal "234" [r getrange mykey -3 -1]
# assert_equal "" [r getrange mykey 5 3]
# assert_equal "4" [r getrange mykey 3 5000]
# assert_equal "1234" [r getrange mykey -5000 10000]
#}

#test "GETRANGE fuzzing" {
# for {set i 0} {$i < 1000} {incr i} {
# r set bin [set bin [randstring 0 1024 binary]]
# set _start [set start [randomInt 1500]]
# set _end [set end [randomInt 1500]]
# if {$_start < 0} {set _start "end-[abs($_start)-1]"}
# if {$_end < 0} {set _end "end-[abs($_end)-1]"}
# assert_equal [string range $bin $_start $_end] [r getrange bin $start $end]
# }
#}

#test {Extended SET can detect syntax errors} {
# set e {}
# catch {r set foo bar non-existing-option} e
# set e
#} {*syntax*}

test {Extended SET NX option} {
r del foo
Expand Down Expand Up @@ -415,38 +415,38 @@ start_server {tags {"string"}} {
assert {$ttl <= 10 && $ttl > 5}
}

test {GETRANGE with huge ranges, Github issue #1844} {
r set foo bar
r getrange foo 0 4294967297
} {bar}
#test {GETRANGE with huge ranges, Github issue #1844} {
# r set foo bar
# r getrange foo 0 4294967297
#} {bar}

set rna1 {CACCTTCCCAGGTAACAAACCAACCAACTTTCGATCTCTTGTAGATCTGTTCTCTAAACGAACTTTAAAATCTGTGTGGCTGTCACTCGGCTGCATGCTTAGTGCACTCACGCAGTATAATTAATAACTAATTACTGTCGTTGACAGGACACGAGTAACTCGTCTATCTTCTGCAGGCTGCTTACGGTTTCGTCCGTGTTGCAGCCGATCATCAGCACATCTAGGTTTCGTCCGGGTGTG}
set rna2 {ATTAAAGGTTTATACCTTCCCAGGTAACAAACCAACCAACTTTCGATCTCTTGTAGATCTGTTCTCTAAACGAACTTTAAAATCTGTGTGGCTGTCACTCGGCTGCATGCTTAGTGCACTCACGCAGTATAATTAATAACTAATTACTGTCGTTGACAGGACACGAGTAACTCGTCTATCTTCTGCAGGCTGCTTACGGTTTCGTCCGTGTTGCAGCCGATCATCAGCACATCTAGGTTT}
set rnalcs {ACCTTCCCAGGTAACAAACCAACCAACTTTCGATCTCTTGTAGATCTGTTCTCTAAACGAACTTTAAAATCTGTGTGGCTGTCACTCGGCTGCATGCTTAGTGCACTCACGCAGTATAATTAATAACTAATTACTGTCGTTGACAGGACACGAGTAACTCGTCTATCTTCTGCAGGCTGCTTACGGTTTCGTCCGTGTTGCAGCCGATCATCAGCACATCTAGGTTT}

test {STRALGO LCS string output with STRINGS option} {
r STRALGO LCS STRINGS $rna1 $rna2
} $rnalcs
#test {STRALGO LCS string output with STRINGS option} {
# r STRALGO LCS STRINGS $rna1 $rna2
#} $rnalcs

test {STRALGO LCS len} {
r STRALGO LCS LEN STRINGS $rna1 $rna2
} [string length $rnalcs]
#test {STRALGO LCS len} {
# r STRALGO LCS LEN STRINGS $rna1 $rna2
#} [string length $rnalcs]

test {LCS with KEYS option} {
r set virus1 $rna1
r set virus2 $rna2
r STRALGO LCS KEYS virus1 virus2
} $rnalcs
#test {LCS with KEYS option} {
# r set virus1 $rna1
# r set virus2 $rna2
# r STRALGO LCS KEYS virus1 virus2
#} $rnalcs

test {LCS indexes} {
dict get [r STRALGO LCS IDX KEYS virus1 virus2] matches
} {{{238 238} {239 239}} {{236 236} {238 238}} {{229 230} {236 237}} {{224 224} {235 235}} {{1 222} {13 234}}}
#test {LCS indexes} {
# dict get [r STRALGO LCS IDX KEYS virus1 virus2] matches
#} {{{238 238} {239 239}} {{236 236} {238 238}} {{229 230} {236 237}} {{224 224} {235 235}} {{1 222} {13 234}}}

test {LCS indexes with match len} {
dict get [r STRALGO LCS IDX KEYS virus1 virus2 WITHMATCHLEN] matches
} {{{238 238} {239 239} 1} {{236 236} {238 238} 1} {{229 230} {236 237} 2} {{224 224} {235 235} 1} {{1 222} {13 234} 222}}
#test {LCS indexes with match len} {
# dict get [r STRALGO LCS IDX KEYS virus1 virus2 WITHMATCHLEN] matches
#} {{{238 238} {239 239} 1} {{236 236} {238 238} 1} {{229 230} {236 237} 2} {{224 224} {235 235} 1} {{1 222} {13 234} 222}}

test {LCS indexes with match len and minimum match len} {
dict get [r STRALGO LCS IDX KEYS virus1 virus2 WITHMATCHLEN MINMATCHLEN 5] matches
} {{{1 222} {13 234} 222}}
#test {LCS indexes with match len and minimum match len} {
# dict get [r STRALGO LCS IDX KEYS virus1 virus2 WITHMATCHLEN MINMATCHLEN 5] matches
#} {{{1 222} {13 234} 222}}
}
Original file line number Diff line number Diff line change
Expand Up @@ -22,22 +22,11 @@ public RMBitMap(byte[] data) {
this.bitSet = BitSet.valueOf(reversed);
}

public RMBitMap(int size, BitSet bitSet) {
this.size = size;
this.bitSet = bitSet;
}

public int getSize() {
return size;
}

public void setBit(byte bit, int pos) {
int newSize = (pos + 7) / 8;

if (size < newSize) {
size = newSize;
}

bitSet.set(pos, bit == 1);
}

Expand All @@ -56,15 +45,14 @@ public final Slice getAsSlice() {
}

/**
* Produce little-endian bitewise representation of the bit set.
* Produce bitewise representation of the bit set.
*/
private byte[] toByteArray() {
long[] longs = bitSet.toLongArray();
int nBytes = (bitSet.length() + 7) / 8;
byte[] bytes = new byte[nBytes];
byte[] bytes = new byte[size];
int byteIndex = 0;
for (long value : longs) {
for (int i = 0; i < 8 && byteIndex < nBytes; i++) {
for (int i = 0; i < 8 && byteIndex < size; i++) {
bytes[byteIndex++] = reverseBits((byte) (value & 0xFF));
value >>>= 8;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,18 +22,23 @@ public SetRange(RedisBase base, List<Slice> params) {
protected Slice response() {
Slice key = params().get(0);
int offset = convertToInteger(params().get(1).toString());
Slice value = params().get(2);
if (offset < 0) {
return Response.error("ERR offset is out of range");
}
Slice value = params().get(2);
String oldValue = Optional.ofNullable(base().getRMString(key))
.map(RMString::getStoredDataAsString)
.orElse("");
String padding = "";
if (offset + value.length() > base().getProtoMaxBulkLen())
return Response.error("ERR string exceeds maximum allowed size (proto-max-bulk-len)");
if (offset > oldValue.length()) {
padding = new String(new byte[offset - oldValue.length()]);
}
String newValue =
oldValue.substring(0, Math.min(offset, oldValue.length()))
+ padding
+ value.toString();
+ value;
if (offset + value.length() < oldValue.length()) {
newValue += oldValue.substring(offset + value.length());
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,8 @@ public class RedisBase {
private final Map<String, String> cachedLuaScripts = new HashMap<>();
private final ExpiringKeyValueStorage keyValueStorage;

private final long protoMaxBulkLen = 512 * 1024 * 1024; //512 MB by default

public RedisBase(Supplier<Clock> clockSupplier) {
this.clockSupplier = Objects.requireNonNull(clockSupplier);
this.keyValueStorage = new ExpiringKeyValueStorage(clockSupplier, key -> watchedKeys
Expand Down Expand Up @@ -326,6 +328,10 @@ public void markKeyModified(Slice key) {
watchedKeys.getOrDefault(key, new HashSet<>()).forEach(OperationExecutorState::watchedKeyIsAffected);
}

public long getProtoMaxBulkLen() {
return protoMaxBulkLen;
}

public String getCachedLuaScript(String sha1) {
return cachedLuaScripts.get(sha1.toLowerCase());
}
Expand All @@ -338,7 +344,6 @@ public void flushCachedLuaScrips() {
cachedLuaScripts.clear();
}


public String addCachedLuaScript(String sha1, String script) {
return cachedLuaScripts.put(sha1, script);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@
import java.util.Collections;
import java.util.List;
import java.util.Random;
import java.util.concurrent.ThreadLocalRandom;

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

Expand Down Expand Up @@ -50,7 +49,6 @@ public void testGetOperationRepeatable(Jedis jedis) {
assertThat(buf2).containsExactlyInAnyOrder(buf);
}


@TestTemplate
void testValueAftersetbit(Jedis jedis) {
jedis.setbit("foo", 0L, true);
Expand Down Expand Up @@ -113,4 +111,18 @@ public void testLongSetBit(Jedis jedis) {
assertThat(redisBytes).isEqualTo(expectedTruncated);
}
}

@TestTemplate
public void testNoOp(Jedis jedis) {
jedis.setbit("noop", 100, false);
assertThat(jedis.get("noop")).hasSize((100 + 7) / 8);
}

@TestTemplate
public void loadZeroes(Jedis jedis) {
byte[] zeroes = new byte[17];
jedis.set("zeroes".getBytes(), zeroes);
jedis.setbit("zeroes", 0L, true);
assertThat(jedis.get("zeroes".getBytes())).hasSize(17);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
import redis.clients.jedis.Jedis;

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

@ExtendWith(ComparisonBase.class)
public class TestSetRange {
Expand Down Expand Up @@ -65,4 +66,17 @@ public void setRangeAgainstNonExistentPadding(Jedis jedis) {
assertThat(jedis.setrange("mykey", 1, "foo")).isEqualTo(4);
assertThat(jedis.get("mykey")).isEqualTo("\000foo");
}

@TestTemplate
public void setRangeWithOutOfRange(Jedis jedis) {
assertThatThrownBy(() -> jedis.setrange("mykey", 512 * 1024 * 1024 - 4, "world"))
.hasMessageStartingWith("ERR string exceeds maximum allowed size (proto-max-bulk-len)");
}

@TestTemplate
public void setRangeWithNegativeOffset(Jedis jedis) {
assertThatThrownBy(() -> jedis.setrange("mykey", -1, "world"))
.hasMessageStartingWith("ERR offset is out of range");
}

}

0 comments on commit a7daeee

Please sign in to comment.