Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Python: add LCS command #1716

Merged
merged 6 commits into from
Jun 29, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@
* Python: Added FUNCTION FLUSH command ([#1700](https://github.com/aws/glide-for-redis/pull/1700))
* Python: Added FUNCTION DELETE command ([#1714](https://github.com/aws/glide-for-redis/pull/1714))
* Python: Added SSCAN command ([#1709](https://github.com/aws/glide-for-redis/pull/1709))
* Python: Added LCS command ([#1716](https://github.com/aws/glide-for-redis/pull/1716))

### Breaking Changes
* Node: Update XREAD to return a Map of Map ([#1494](https://github.com/aws/glide-for-redis/pull/1494))
Expand Down
170 changes: 170 additions & 0 deletions python/python/glide/async_commands/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -5488,3 +5488,173 @@ def try_get_pubsub_message(self) -> Optional[PubSubMsg]:
>>> pubsub_msg = listening_client.try_get_pubsub_message()
"""
...

async def lcs(
self,
key1: str,
key2: str,
) -> str:
"""
Returns the longest common subsequence between strings stored at key1 and key2.

Note that this is different than the longest common string algorithm, since
matching characters in the two strings do not need to be contiguous.

For instance the LCS between "foo" and "fao" is "fo", since scanning the two strings
from left to right, the longest common set of characters is composed of the first "f" and then the "o".

See https://valkey.io/commands/lcs for more details.

Args:
key1 (str): The key that stores the first string.
key2 (str): The key that stores the second string.

Returns:
A String containing the longest common subsequence between the 2 strings.
An empty String is returned if the keys do not exist or have no common subsequences.

Examples:
>>> await client.mset({"testKey1" : "abcd", "testKey2": "axcd"})
'OK'
>>> await client.lcs("testKey1", "testKey2")
'acd'

Since: Redis version 7.0.0.
"""
args = [key1, key2]

return cast(
str,
await self._execute_command(RequestType.LCS, args),
)

async def lcs_len(
self,
key1: str,
key2: str,
) -> int:
"""
Returns the length of the longest common subsequence between strings stored at key1 and key2.

Note that this is different than the longest common string algorithm, since
matching characters in the two strings do not need to be contiguous.

For instance the LCS between "foo" and "fao" is "fo", since scanning the two strings
from left to right, the longest common set of characters is composed of the first "f" and then the "o".

See https://valkey.io/commands/lcs for more details.

Args:
key1 (str): The key that stores the first string.
key2 (str): The key that stores the second string.

Returns:
The length of the longest common subsequence between the 2 strings.

Examples:
>>> await client.mset({"testKey1" : "abcd", "testKey2": "axcd"})
'OK'
>>> await client.lcs_len("testKey1", "testKey2")
3 # the length of the longest common subsequence between these 2 strings ("acd") is 3.

Since: Redis version 7.0.0.
"""
args = [key1, key2, "LEN"]

return cast(
int,
await self._execute_command(RequestType.LCS, args),
)

async def lcs_idx(
self,
key1: str,
key2: str,
min_match_len: Optional[int] = None,
with_match_len: Optional[bool] = False,
) -> Mapping[str, Union[list[list[Union[list[int], int]]], int]]:
"""
Returns the indices and length of the longest common subsequence between strings stored at key1 and key2.

Note that this is different than the longest common string algorithm, since
matching characters in the two strings do not need to be contiguous.

For instance the LCS between "foo" and "fao" is "fo", since scanning the two strings
from left to right, the longest common set of characters is composed of the first "f" and then the "o".

See https://valkey.io/commands/lcs for more details.

Args:
key1 (str): The key that stores the first string.
key2 (str): The key that stores the second string.
min_match_len (Optional[int]): The minimum length of matches to include in the result.
with_match_len (Optional[bool]): If True, include the length of the substring matched for each substring.

Returns:
A Mapping containing the indices of the longest common subsequence between the
2 strings and the length of the longest common subsequence. The resulting map contains two
keys, "matches" and "len":
- "len" is mapped to the length of the longest common subsequence between the 2 strings.
- "matches" is mapped to a three dimensional int array that stores pairs of indices that
represent the location of the common subsequences in the strings held by key1 and key2,
with the length of the match after each matches, if with_match_len is enabled.

Examples:
>>> await client.mset({"testKey1" : "abcd1234", "testKey2": "bcdef1234"})
'OK'
>>> await client.lcs_idx("testKey1", "testKey2")
{
'matches': [
[
[4, 7], # starting and ending indices of the subsequence "1234" in "abcd1234" (testKey1)
[5, 8], # starting and ending indices of the subsequence "1234" in "bcdef1234" (testKey2)
],
[
[1, 3], # starting and ending indices of the subsequence "bcd" in "abcd1234" (testKey1)
[0, 2], # starting and ending indices of the subsequence "bcd" in "bcdef1234" (testKey2)
],
],
'len': 7 # length of the entire longest common subsequence
}
>>> await client.lcs_idx("testKey1", "testKey2", min_match_len=4)
{
'matches': [
[
[4, 7],
[5, 8],
],
# the other match with a length of 3 is excluded
],
'len': 7
}
>>> await client.lcs_idx("testKey1", "testKey2", with_match_len=True)
{
'matches': [
[
[4, 7],
[5, 8],
4, # length of this match ("1234")
],
[
[1, 3],
[0, 2],
3, # length of this match ("bcd")
],
],
'len': 7
}

Since: Redis version 7.0.0.
"""
args = [key1, key2, "IDX"]

if min_match_len is not None:
args.extend(["MINMATCHLEN", str(min_match_len)])

if with_match_len:
args.append("WITHMATCHLEN")

return cast(
Mapping[str, Union[list[list[Union[list[int], int]]], int]],
await self._execute_command(RequestType.LCS, args),
)
104 changes: 104 additions & 0 deletions python/python/glide/async_commands/transaction.py
Original file line number Diff line number Diff line change
Expand Up @@ -3949,6 +3949,110 @@ def sscan(

return self.append_command(RequestType.SScan, args)

def lcs(
self: TTransaction,
key1: str,
key2: str,
) -> TTransaction:
"""
Returns the longest common subsequence between strings stored at key1 and key2.

Note that this is different than the longest common string algorithm, since
matching characters in the two strings do not need to be contiguous.

For instance the LCS between "foo" and "fao" is "fo", since scanning the two strings
from left to right, the longest common set of characters is composed of the first "f" and then the "o".

See https://valkey.io/commands/lcs for more details.

Args:
key1 (str): The key that stores the first string.
key2 (str): The key that stores the second string.

Command Response:
A String containing the longest common subsequence between the 2 strings.
An empty String is returned if the keys do not exist or have no common subsequences.

Since: Redis version 7.0.0.
"""
args = [key1, key2]

return self.append_command(RequestType.LCS, args)

def lcs_len(
self: TTransaction,
key1: str,
key2: str,
) -> TTransaction:
"""
Returns the length of the longest common subsequence between strings stored at key1 and key2.

Note that this is different than the longest common string algorithm, since
matching characters in the two strings do not need to be contiguous.

For instance the LCS between "foo" and "fao" is "fo", since scanning the two strings
from left to right, the longest common set of characters is composed of the first "f" and then the "o".

See https://valkey.io/commands/lcs for more details.

Args:
key1 (str): The key that stores the first string.
key2 (str): The key that stores the second string.

Command Response:
The length of the longest common subsequence between the 2 strings.

Since: Redis version 7.0.0.
"""
args = [key1, key2, "LEN"]

return self.append_command(RequestType.LCS, args)

def lcs_idx(
self: TTransaction,
key1: str,
key2: str,
min_match_len: Optional[int] = None,
with_match_len: Optional[bool] = False,
) -> TTransaction:
"""
Returns the indices and length of the longest common subsequence between strings stored at key1 and key2.

Note that this is different than the longest common string algorithm, since
matching characters in the two strings do not need to be contiguous.

For instance the LCS between "foo" and "fao" is "fo", since scanning the two strings
from left to right, the longest common set of characters is composed of the first "f" and then the "o".

See https://valkey.io/commands/lcs for more details.

Args:
key1 (str): The key that stores the first string.
key2 (str): The key that stores the second string.
min_match_len (Optional[int]): The minimum length of matches to include in the result.
with_match_len (Optional[bool]): If True, include the length of the substring matched for each substring.

Command Response:
A Map containing the indices of the longest common subsequence between the
2 strings and the length of the longest common subsequence. The resulting map contains two
keys, "matches" and "len":
- "len" is mapped to the length of the longest common subsequence between the 2 strings.
- "matches" is mapped to a three dimensional int array that stores pairs of indices that
represent the location of the common subsequences in the strings held by key1 and key2,
with the length of the match after each matches, if with_match_len is enabled.

Since: Redis version 7.0.0.
"""
args = [key1, key2, "IDX"]

if min_match_len is not None:
args.extend(["MINMATCHLEN", str(min_match_len)])

if with_match_len:
args.append("WITHMATCHLEN")

return self.append_command(RequestType.LCS, args)


class Transaction(BaseTransaction):
"""
Expand Down
Loading
Loading