Skip to content

Commit

Permalink
Support TOPK.LIST with WITHCOUNT option (#3495)
Browse files Browse the repository at this point in the history
  • Loading branch information
sazzad16 committed Oct 18, 2023
1 parent 9713420 commit 4a84608
Show file tree
Hide file tree
Showing 10 changed files with 95 additions and 7 deletions.
31 changes: 31 additions & 0 deletions src/main/java/redis/clients/jedis/BuilderFactory.java
Original file line number Diff line number Diff line change
Expand Up @@ -378,6 +378,37 @@ public String toString() {
}
};

public static final Builder<Map<String, Long>> STRING_LONG_MAP = new Builder<Map<String, Long>>() {
@Override
@SuppressWarnings("unchecked")
public Map<String, Long> build(Object data) {
final List<Object> list = (List<Object>) data;
if (list.isEmpty()) return Collections.emptyMap();

if (list.get(0) instanceof KeyValue) {
final Map<String, Long> map = new LinkedHashMap<>(list.size(), 1f);
final Iterator iterator = list.iterator();
while (iterator.hasNext()) {
KeyValue kv = (KeyValue) iterator.next();
map.put(STRING.build(kv.getKey()), LONG.build(kv.getValue()));
}
return map;
} else {
final Map<String, Long> map = new LinkedHashMap<>(list.size() / 2, 1f);
final Iterator iterator = list.iterator();
while (iterator.hasNext()) {
map.put(STRING.build(iterator.next()), LONG.build(iterator.next()));
}
return map;
}
}

@Override
public String toString() {
return "Map<String, Long>";
}
};

public static final Builder<KeyedListElement> KEYED_LIST_ELEMENT = new Builder<KeyedListElement>() {
@Override
@SuppressWarnings("unchecked")
Expand Down
5 changes: 5 additions & 0 deletions src/main/java/redis/clients/jedis/CommandObjects.java
Original file line number Diff line number Diff line change
Expand Up @@ -4056,6 +4056,11 @@ public final CommandObject<List<String>> topkList(String key) {
return new CommandObject<>(commandArguments(TopKCommand.LIST).key(key), BuilderFactory.STRING_LIST);
}

public final CommandObject<Map<String, Long>> topkListWithCount(String key) {
return new CommandObject<>(commandArguments(TopKCommand.LIST).key(key)
.add(RedisBloomKeyword.WITHCOUNT), BuilderFactory.STRING_LONG_MAP);
}

public final CommandObject<Map<String, Object>> topkInfo(String key) {
return new CommandObject<>(commandArguments(TopKCommand.INFO).key(key), BuilderFactory.ENCODED_OBJECT_MAP);
}
Expand Down
5 changes: 5 additions & 0 deletions src/main/java/redis/clients/jedis/MultiNodePipelineBase.java
Original file line number Diff line number Diff line change
Expand Up @@ -4231,6 +4231,11 @@ public Response<List<String>> topkList(String key) {
return appendCommand(commandObjects.topkList(key));
}

@Override
public Response<Map<String, Long>> topkListWithCount(String key) {
return appendCommand(commandObjects.topkListWithCount(key));
}

@Override
public Response<Map<String, Object>> topkInfo(String key) {
return appendCommand(commandObjects.topkInfo(key));
Expand Down
5 changes: 5 additions & 0 deletions src/main/java/redis/clients/jedis/Pipeline.java
Original file line number Diff line number Diff line change
Expand Up @@ -4180,6 +4180,11 @@ public Response<List<String>> topkList(String key) {
return appendCommand(commandObjects.topkList(key));
}

@Override
public Response<Map<String, Long>> topkListWithCount(String key) {
return appendCommand(commandObjects.topkListWithCount(key));
}

@Override
public Response<Map<String, Object>> topkInfo(String key) {
return appendCommand(commandObjects.topkInfo(key));
Expand Down
5 changes: 5 additions & 0 deletions src/main/java/redis/clients/jedis/TransactionBase.java
Original file line number Diff line number Diff line change
Expand Up @@ -4280,6 +4280,11 @@ public Response<List<String>> topkList(String key) {
return appendCommand(commandObjects.topkList(key));
}

@Override
public Response<Map<String, Long>> topkListWithCount(String key) {
return appendCommand(commandObjects.topkListWithCount(key));
}

@Override
public Response<Map<String, Object>> topkInfo(String key) {
return appendCommand(commandObjects.topkInfo(key));
Expand Down
5 changes: 5 additions & 0 deletions src/main/java/redis/clients/jedis/UnifiedJedis.java
Original file line number Diff line number Diff line change
Expand Up @@ -4534,6 +4534,11 @@ public List<String> topkList(String key) {
return executeCommand(commandObjects.topkList(key));
}

@Override
public Map<String, Long> topkListWithCount(String key) {
return executeCommand(commandObjects.topkListWithCount(key));
}

@Override
public Map<String, Object> topkInfo(String key) {
return executeCommand(commandObjects.topkInfo(key));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -121,7 +121,7 @@ public byte[] getRaw() {
public enum RedisBloomKeyword implements Rawable {

CAPACITY, ERROR, NOCREATE, EXPANSION, NONSCALING, BUCKETSIZE, MAXITERATIONS, ITEMS, WEIGHTS,
COMPRESSION, OVERRIDE;
COMPRESSION, OVERRIDE, WITHCOUNT;

private final byte[] raw;

Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
package redis.clients.jedis.bloom.commands;

import java.util.Collections;
import java.util.List;
import java.util.Map;

Expand Down Expand Up @@ -35,6 +36,18 @@ public interface TopKFilterCommands {
*/
List<String> topkAdd(String key, String... items);

/**
* {@code TOPK.INCRBY {key} {item} {increment}}
*
* @param key
* @param item
* @param increment
* @return item dropped from list
*/
default String topkIncrBy(String key, String item, long increment) {
return topkIncrBy(key, Collections.singletonMap(item, increment)).get(0);
}

/**
* {@code TOPK.INCRBY {key} {item} {increment} [{item} {increment} ...]}
*
Expand Down Expand Up @@ -72,6 +85,14 @@ public interface TopKFilterCommands {
*/
List<String> topkList(String key);

/**
* {@code TOPK.LIST {key} WITHCOUNT}
*
* @param key
* @return k (or less) items in Top K list
*/
Map<String, Long> topkListWithCount(String key);

/**
* {@code TOPK.INFO {key}}
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,5 +24,7 @@ public interface TopKFilterPipelineCommands {

Response<List<String>> topkList(String key);

Response<Map<String, Long>> topkListWithCount(String key);

Response<Map<String, Object>> topkInfo(String key);
}
21 changes: 15 additions & 6 deletions src/test/java/redis/clients/jedis/modules/bloom/TopKTest.java
Original file line number Diff line number Diff line change
@@ -1,10 +1,11 @@
package redis.clients.jedis.modules.bloom;

import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue;

import java.util.Arrays;
import java.util.Collections;
import java.util.TreeSet;
import java.util.Map;
import org.junit.BeforeClass;
import org.junit.Test;

Expand Down Expand Up @@ -32,12 +33,20 @@ public void createTopKFilter() {

assertEquals(Arrays.asList(1L, 0L, 1L), client.topkCount("aaa", "bb", "gg", "cc"));

assertEquals(new TreeSet<>(Arrays.asList("bb", "cc")), new TreeSet<>(client.topkList("aaa")));
assertEquals(Arrays.asList("bb", "cc"), client.topkList("aaa"));

// assertEquals(null, client.topkIncrBy("aaa", "ff", 10));
assertEquals(Collections.<String>singletonList(null),
client.topkIncrBy("aaa", Collections.singletonMap("ff", 10L)));
Map<String, Long> listWithCount = client.topkListWithCount("aaa");
assertEquals(2, listWithCount.size());
listWithCount.forEach((item, count) -> {
assertTrue(Arrays.asList("bb", "cc").contains(item));
assertEquals(Long.valueOf(1), count);
});

assertEquals(null, client.topkIncrBy("aaa", "ff", 5));
assertEquals(Arrays.asList("ff", "bb", "cc"), client.topkList("aaa"));

assertEquals(new TreeSet<>(Arrays.asList("bb", "cc", "ff")), new TreeSet<>(client.topkList("aaa")));
assertEquals(Collections.<String>singletonList(null),
client.topkIncrBy("aaa", Collections.singletonMap("ff", 8L)));
assertEquals(Long.valueOf(13), client.topkListWithCount("aaa").get("ff"));
}
}

0 comments on commit 4a84608

Please sign in to comment.