Skip to content

Commit

Permalink
Support NOVALUES parameter for HSCAN
Browse files Browse the repository at this point in the history
Issue #3153

The NOVALUES parameter instructs HSCAN to only return the hash keys,
without values.
  • Loading branch information
gerzse committed Mar 18, 2024
1 parent 037d108 commit 7beaaea
Show file tree
Hide file tree
Showing 3 changed files with 52 additions and 5 deletions.
7 changes: 6 additions & 1 deletion redis/_parsers/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -354,7 +354,12 @@ def parse_scan(response, **options):

def parse_hscan(response, **options):
cursor, r = response
return int(cursor), r and pairs_to_dict(r) or {}
no_values = options.get("no_values", False)
if no_values:
payload = r or []
else:
payload = r and pairs_to_dict(r) or {}
return int(cursor), payload


def parse_zscan(response, **options):
Expand Down
25 changes: 21 additions & 4 deletions redis/commands/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -3102,6 +3102,7 @@ def hscan(
cursor: int = 0,
match: Union[PatternT, None] = None,
count: Union[int, None] = None,
no_values: Union[bool, None] = None,
) -> ResponseT:
"""
Incrementally return key/value slices in a hash. Also return a cursor
Expand All @@ -3111,20 +3112,26 @@ def hscan(
``count`` allows for hint the minimum number of returns
``no_values`` indicates to return only the keys, without values. The
return type in this case is a list of keys.
For more information see https://redis.io/commands/hscan
"""
pieces: list[EncodableT] = [name, cursor]
if match is not None:
pieces.extend([b"MATCH", match])
if count is not None:
pieces.extend([b"COUNT", count])
return self.execute_command("HSCAN", *pieces)
if no_values is not None:
pieces.extend([b"NOVALUES"])
return self.execute_command("HSCAN", *pieces, no_values=no_values)

def hscan_iter(
self,
name: str,
match: Union[PatternT, None] = None,
count: Union[int, None] = None,
no_values: Union[bool, None] = None,
) -> Iterator:
"""
Make an iterator using the HSCAN command so that the client doesn't
Expand All @@ -3133,11 +3140,18 @@ def hscan_iter(
``match`` allows for filtering the keys by pattern
``count`` allows for hint the minimum number of returns
``no_values`` indicates to return only the keys, without values
"""
cursor = "0"
while cursor != 0:
cursor, data = self.hscan(name, cursor=cursor, match=match, count=count)
yield from data.items()
cursor, data = self.hscan(
name, cursor=cursor, match=match, count=count, no_values=no_values
)
if no_values:
yield from data
else:
yield from data.items()

def zscan(
self,
Expand Down Expand Up @@ -3253,6 +3267,7 @@ async def hscan_iter(
name: str,
match: Union[PatternT, None] = None,
count: Union[int, None] = None,
no_values: Union[bool, None] = None,
) -> AsyncIterator:
"""
Make an iterator using the HSCAN command so that the client doesn't
Expand All @@ -3261,11 +3276,13 @@ async def hscan_iter(
``match`` allows for filtering the keys by pattern
``count`` allows for hint the minimum number of returns
``no_values`` indicates to return only the keys, without values
"""
cursor = "0"
while cursor != 0:
cursor, data = await self.hscan(
name, cursor=cursor, match=match, count=count
name, cursor=cursor, match=match, count=count, no_values=no_values
)
for it in data.items():
yield it
Expand Down
25 changes: 25 additions & 0 deletions tests/test_commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -2162,6 +2162,19 @@ def test_hscan(self, r):
assert dic == {b"a": b"1", b"b": b"2", b"c": b"3"}
_, dic = r.hscan("a", match="a")
assert dic == {b"a": b"1"}
_, dic = r.hscan("a_notset")
assert dic == {}

@skip_if_server_version_lt("7.4.0")
def test_hscan_novalues(self, r):
r.hset("a", mapping={"a": 1, "b": 2, "c": 3})
cursor, keys = r.hscan("a", no_values=True)
assert cursor == 0
assert keys == [b"a", b"b", b"c"]
_, keys = r.hscan("a", match="a", no_values=True)
assert keys == [b"a"]
_, keys = r.hscan("a_notset", no_values=True)
assert keys == []

@skip_if_server_version_lt("2.8.0")
def test_hscan_iter(self, r):
Expand All @@ -2170,6 +2183,18 @@ def test_hscan_iter(self, r):
assert dic == {b"a": b"1", b"b": b"2", b"c": b"3"}
dic = dict(r.hscan_iter("a", match="a"))
assert dic == {b"a": b"1"}
dic = dict(r.hscan_iter("a_notset"))
assert dic == {}

@skip_if_server_version_lt("7.4.0")
def test_hscan_iter_novalues(self, r):
r.hset("a", mapping={"a": 1, "b": 2, "c": 3})
keys = list(r.hscan_iter("a", no_values=True))
assert keys == [b"a", b"b", b"c"]
keys = list(r.hscan_iter("a", match="a", no_values=True))
assert keys == [b"a"]
keys = list(r.hscan_iter("a_notset", no_values=True))
assert keys == []

@skip_if_server_version_lt("2.8.0")
def test_zscan(self, r):
Expand Down

0 comments on commit 7beaaea

Please sign in to comment.