Skip to content

Commit

Permalink
fix ZRange and add REV support
Browse files Browse the repository at this point in the history
  • Loading branch information
mdvorchik committed Mar 13, 2022
1 parent 0047030 commit 52192df
Show file tree
Hide file tree
Showing 3 changed files with 92 additions and 60 deletions.
Original file line number Diff line number Diff line change
@@ -1,14 +1,16 @@
package com.github.fppt.jedismock.operations.sortedsets;

import com.github.fppt.jedismock.datastructures.RMHMap;
import com.github.fppt.jedismock.datastructures.Slice;
import com.github.fppt.jedismock.operations.AbstractRedisOperation;
import com.github.fppt.jedismock.operations.RedisCommand;
import com.github.fppt.jedismock.server.Response;
import com.github.fppt.jedismock.datastructures.Slice;
import com.github.fppt.jedismock.storage.RedisBase;

import java.util.Comparator;
import java.util.List;
import java.util.Map;
import java.util.function.ToDoubleFunction;
import java.util.stream.Collectors;
import java.util.stream.Stream;

Expand All @@ -18,6 +20,12 @@
class ZRange extends AbstractRedisOperation {

private static final String WITH_SCORES = "WITHSCORES";
private static final String IS_REV = "REV";

private static final Comparator<Map.Entry<Slice, Double>> zRangeComparator = Comparator
.comparingDouble((ToDoubleFunction<Map.Entry<Slice, Double>>) Map.Entry::getValue)
.thenComparing(Map.Entry::getKey);
private static final Comparator<Map.Entry<Slice, Double>> zRevRangeComparator = (o1, o2) -> zRangeComparator.compare(o2, o1);

ZRange(RedisBase base, List<Slice> params) {
super(base, params);
Expand Down Expand Up @@ -50,15 +58,28 @@ protected Slice response() {
end = map.size() - 1;
}

final boolean withScores = params().size() == 4 && WITH_SCORES.equalsIgnoreCase(params().get(3).toString());
boolean withScores = false;
boolean isRev = false;

for (Slice param : params()) {
if (WITH_SCORES.equalsIgnoreCase(param.toString())){
withScores = true;
}
if (IS_REV.equalsIgnoreCase(param.toString())){
isRev = true;
}
}

boolean finalWithScores = withScores;
List<Slice> values = map.entrySet().stream()
.skip(start)
.limit(end - start + 1)
.flatMap(e -> withScores
? Stream.of(e.getKey(), Slice.create(e.getValue().toString()))
: Stream.of(e.getKey()))
.map(Response::bulkString)
.collect(Collectors.toList());
.sorted(isRev ? zRevRangeComparator : zRangeComparator)
.skip(start)
.limit(end - start + 1)
.flatMap(e -> finalWithScores
? Stream.of(e.getKey(), Slice.create(e.getValue().toString()))
: Stream.of(e.getKey()))
.map(Response::bulkString)
.collect(Collectors.toList());

return Response.array(values);
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,72 +1,29 @@
package com.github.fppt.jedismock.operations.sortedsets;

import com.github.fppt.jedismock.datastructures.RMHMap;
import com.github.fppt.jedismock.datastructures.Slice;
import com.github.fppt.jedismock.operations.AbstractRedisOperation;
import com.github.fppt.jedismock.operations.RedisCommand;
import com.github.fppt.jedismock.server.Response;
import com.github.fppt.jedismock.storage.RedisBase;

import java.util.Comparator;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import static com.github.fppt.jedismock.Utils.convertToInteger;

@RedisCommand("zrevrange")
class ZRevRange extends AbstractRedisOperation {

private static final String WITH_SCORES = "WITHSCORES";
private static final Comparator<Map.Entry<Slice, Double>> zRevRangeComparator = (o1, o2) -> {
int byScoreComparison = o2.getValue().compareTo(o1.getValue());
return byScoreComparison != 0 ? byScoreComparison : o2.getKey().compareTo(o1.getKey());
};
private static final String IS_REV = "REV";
private final ZRange zRange;


ZRevRange(RedisBase base, List<Slice> params) {
super(base, params);
List<Slice> updatedParams = new ArrayList<>(params);
updatedParams.add(Slice.create(IS_REV));
this.zRange = new ZRange(base, updatedParams);
}

@Override
protected Slice response() {
Slice key = params().get(0);
final RMHMap mapDBObj = getHMapFromBaseOrCreateEmpty(key);
final Map<Slice, Double> map = mapDBObj.getStoredData();

int start = convertToInteger(params().get(1).toString());
int end = convertToInteger(params().get(2).toString());

if (start < 0) {
start = map.size() + start;
if (start < 0) {
start = 0;
}
}

if (end < 0) {
end = map.size() + end;
if (end < 0) {
end = -1;
}
}

if (end >= map.size()) {
end = map.size() - 1;
}

final boolean withScores = params().size() == 4 && WITH_SCORES.equalsIgnoreCase(params().get(3).toString());

List<Slice> values = map.entrySet().stream()
.sorted(zRevRangeComparator)
.skip(start)
.limit(end - start + 1)
.flatMap(e -> withScores
? Stream.of(e.getKey(), Slice.create(e.getValue().toString()))
: Stream.of(e.getKey()))
.map(Response::bulkString)
.collect(Collectors.toList());

return Response.array(values);
return zRange.response();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
package com.github.fppt.jedismock.comparisontests.sortedsets;

import com.github.fppt.jedismock.comparisontests.ComparisonBase;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.TestTemplate;
import org.junit.jupiter.api.extension.ExtendWith;
import redis.clients.jedis.Jedis;

import java.util.ArrayList;
import java.util.Arrays;

import static org.junit.jupiter.api.Assertions.assertEquals;

@ExtendWith(ComparisonBase.class)
public class TestZRange {

private static final String ZSET_KEY = "myzset";

@BeforeEach
public void setUp(Jedis jedis) {
jedis.flushDB();
jedis.zadd(ZSET_KEY, 2, "aaaa");
jedis.zadd(ZSET_KEY, 3, "bbbb");
jedis.zadd(ZSET_KEY, 1, "cccc");
jedis.zadd(ZSET_KEY, 3, "bcbb");
jedis.zadd(ZSET_KEY, 3, "babb");
assertEquals(5L, jedis.zcount(ZSET_KEY, Integer.MIN_VALUE, Integer.MAX_VALUE));
}

@TestTemplate
public void whenUsingZrange_EnsureItReturnsEverythingInRightOrderWithPlusMinusMaxInteger(Jedis jedis) {
assertEquals(Arrays.asList("cccc", "aaaa", "babb", "bbbb", "bcbb"), new ArrayList<>(jedis.zrange(ZSET_KEY, Integer.MIN_VALUE, Integer.MAX_VALUE)));
}

@TestTemplate
public void whenUsingZrange_EnsureItReturnsListInRightOrderWithPositiveRange(Jedis jedis) {
assertEquals(Arrays.asList("aaaa", "babb", "bbbb"), new ArrayList<>(jedis.zrange(ZSET_KEY, 1, 3)));
}

@TestTemplate
public void whenUsingZrange_EnsureItReturnsListInRightOrderWithNegativeRange(Jedis jedis) {
assertEquals(Arrays.asList("babb", "bbbb", "bcbb"), new ArrayList<>(jedis.zrange(ZSET_KEY, -3, -1)));
}

@TestTemplate
public void whenUsingZrange_EnsureItReturnsListInRightOrderWithNegativeStartAndPositiveEndRange(Jedis jedis) {
assertEquals(Arrays.asList("cccc", "aaaa", "babb"), new ArrayList<>(jedis.zrange(ZSET_KEY, -5, 2)));
}

@TestTemplate
public void whenUsingZrange_EnsureItReturnsListInRightOrderWithPositiveStartAndNegativeEndRange(Jedis jedis) {
assertEquals(Arrays.asList("aaaa", "babb", "bbbb", "bcbb"), new ArrayList<>(jedis.zrange(ZSET_KEY, 1, -1)));
}
}

0 comments on commit 52192df

Please sign in to comment.