Skip to content

Commit

Permalink
Add Python cmd GetEX
Browse files Browse the repository at this point in the history
* Python: Add command GetEx
  • Loading branch information
tjzhang-BQ authored Jun 19, 2024
1 parent fe82f1b commit cfaa599
Show file tree
Hide file tree
Showing 8 changed files with 237 additions and 1 deletion.
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,8 @@
* Node: Added OBJECT REFCOUNT command ([#1568](https://github.com/aws/glide-for-redis/pull/1568))
* Python: Added SETBIT command ([#1571](https://github.com/aws/glide-for-redis/pull/1571))
* Python: Added GETBIT command ([#1575](https://github.com/aws/glide-for-redis/pull/1575))
* Python: Added GETEX command ([#0000](https://github.com/aws/glide-for-redis/pull/TODOLINK))
*
* Python: Added BITCOUNT command ([#1592](https://github.com/aws/glide-for-redis/pull/1592))

### Breaking Changes
Expand Down
1 change: 1 addition & 0 deletions glide-core/src/protobuf/redis_request.proto
Original file line number Diff line number Diff line change
Expand Up @@ -222,6 +222,7 @@ enum RequestType {
GeoSearch = 182;
Watch = 183;
UnWatch = 184;
GetEx = 190;
}

message Command {
Expand Down
3 changes: 3 additions & 0 deletions glide-core/src/request_type.rs
Original file line number Diff line number Diff line change
Expand Up @@ -192,6 +192,7 @@ pub enum RequestType {
GeoSearch = 182,
Watch = 183,
UnWatch = 184,
GetEx = 190,
}

fn get_two_word_command(first: &str, second: &str) -> Cmd {
Expand Down Expand Up @@ -387,6 +388,7 @@ impl From<::protobuf::EnumOrUnknown<ProtobufRequestType>> for RequestType {
ProtobufRequestType::GeoSearch => RequestType::GeoSearch,
ProtobufRequestType::Watch => RequestType::Watch,
ProtobufRequestType::UnWatch => RequestType::UnWatch,
ProtobufRequestType::GetEx => RequestType::GetEx,
}
}
}
Expand Down Expand Up @@ -578,6 +580,7 @@ impl RequestType {
RequestType::GeoSearch => Some(cmd("GEOSEARCH")),
RequestType::Watch => Some(cmd("WATCH")),
RequestType::UnWatch => Some(cmd("UNWATCH")),
RequestType::GetEx => Some(cmd("GETEX")),
}
}
}
2 changes: 2 additions & 0 deletions python/python/glide/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,10 @@
from glide.async_commands.core import (
ConditionalChange,
ExpireOptions,
ExpiryGetEx,
ExpirySet,
ExpiryType,
ExpiryTypeGetEx,
InfoSection,
InsertPosition,
StreamAddOptions,
Expand Down
112 changes: 112 additions & 0 deletions python/python/glide/async_commands/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,23 @@ class ExpiryType(Enum):
KEEP_TTL = 4, Type[None] # Equivalent to `KEEPTTL` in the Redis API


class ExpiryTypeGetEx(Enum):
"""GetEx option: The type of the expiry.
- EX - Set the specified expire time, in seconds. Equivalent to `EX` in the Redis API.
- PX - Set the specified expire time, in milliseconds. Equivalent to `PX` in the Redis API.
- UNIX_SEC - Set the specified Unix time at which the key will expire, in seconds. Equivalent to `EXAT` in the Redis API.
- UNIX_MILLSEC - Set the specified Unix time at which the key will expire, in milliseconds. Equivalent to `PXAT` in the
Redis API.
- PERSIST - Remove the time to live associated with the key. Equivalent to `PERSIST` in the Redis API.
"""

SEC = 0, Union[int, timedelta] # Equivalent to `EX` in the Redis API
MILLSEC = 1, Union[int, timedelta] # Equivalent to `PX` in the Redis API
UNIX_SEC = 2, Union[int, datetime] # Equivalent to `EXAT` in the Redis API
UNIX_MILLSEC = 3, Union[int, datetime] # Equivalent to `PXAT` in the Redis API
PERSIST = 4, Type[None] # Equivalent to `PERSIST` in the Redis API


class InfoSection(Enum):
"""
INFO option: a specific section of information:
Expand Down Expand Up @@ -324,6 +341,61 @@ def get_cmd_args(self) -> List[str]:
return [self.cmd_arg] if self.value is None else [self.cmd_arg, self.value]


class ExpiryGetEx:
"""GetEx option: Represents the expiry type and value to be executed with "GetEx" command."""

def __init__(
self,
expiry_type: ExpiryTypeGetEx,
value: Optional[Union[int, datetime, timedelta]],
) -> None:
"""
Args:
- expiry_type (ExpiryType): The expiry type.
- value (Optional[Union[int, datetime, timedelta]]): The value of the expiration type. The type of expiration
determines the type of expiration value:
- SEC: Union[int, timedelta]
- MILLSEC: Union[int, timedelta]
- UNIX_SEC: Union[int, datetime]
- UNIX_MILLSEC: Union[int, datetime]
- PERSIST: Type[None]
"""
self.set_expiry_type_and_value(expiry_type, value)

def set_expiry_type_and_value(
self,
expiry_type: ExpiryTypeGetEx,
value: Optional[Union[int, datetime, timedelta]],
):
if not isinstance(value, get_args(expiry_type.value[1])):
raise ValueError(
f"The value of {expiry_type} should be of type {expiry_type.value[1]}"
)
self.expiry_type = expiry_type
if self.expiry_type == ExpiryTypeGetEx.SEC:
self.cmd_arg = "EX"
if isinstance(value, timedelta):
value = int(value.total_seconds())
elif self.expiry_type == ExpiryTypeGetEx.MILLSEC:
self.cmd_arg = "PX"
if isinstance(value, timedelta):
value = int(value.total_seconds() * 1000)
elif self.expiry_type == ExpiryTypeGetEx.UNIX_SEC:
self.cmd_arg = "EXAT"
if isinstance(value, datetime):
value = int(value.timestamp())
elif self.expiry_type == ExpiryTypeGetEx.UNIX_MILLSEC:
self.cmd_arg = "PXAT"
if isinstance(value, datetime):
value = int(value.timestamp() * 1000)
elif self.expiry_type == ExpiryTypeGetEx.PERSIST:
self.cmd_arg = "PERSIST"
self.value = str(value) if value else None

def get_cmd_args(self) -> List[str]:
return [self.cmd_arg] if self.value is None else [self.cmd_arg, self.value]


class InsertPosition(Enum):
BEFORE = "BEFORE"
AFTER = "AFTER"
Expand Down Expand Up @@ -4288,3 +4360,43 @@ async def object_refcount(self, key: str) -> Optional[int]:
Optional[int],
await self._execute_command(RequestType.ObjectRefCount, [key]),
)

async def getex(
self,
key: str,
expiry: Optional[ExpiryGetEx] = None,
) -> Optional[str]:
"""
Get the value of `key` and optionally set its expiration. `GETEX` is similar to `GET`.
See https://valkey.io/commands/getex for more details.
Args:
key (str): The key to get.
expiry (Optional[ExpirySet], optional): set expiriation to the given key.
Equivalent to [`EX` | `PX` | `EXAT` | `PXAT` | `PERSIST`] in the Redis API.
Returns:
Optional[str]:
If `key` exists, return the value stored at `key`
If `key` does not exist, return `None`
Examples:
>>> await client.set("key", "value")
'OK'
>>> await client.getex("key")
'value'
>>> await client.getex("key", ExpiryGetEx(ExpiryTypeGetEx.SEC, 1))
'value'
>>> time.sleep(1)
>>> await client.getex("key")
None
Since: Redis version 6.2.0.
"""
args = [key]
if expiry is not None:
args.extend(expiry.get_cmd_args())
return cast(
Optional[str],
await self._execute_command(RequestType.GetEx, args),
)
25 changes: 25 additions & 0 deletions python/python/glide/async_commands/transaction.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
from glide.async_commands.core import (
ConditionalChange,
ExpireOptions,
ExpiryGetEx,
ExpirySet,
GeospatialData,
GeoUnit,
Expand Down Expand Up @@ -2974,6 +2975,30 @@ def object_refcount(self: TTransaction, key: str) -> TTransaction:
"""
return self.append_command(RequestType.ObjectRefCount, [key])

def getex(
self: TTransaction, key: str, expiry: Optional[ExpiryGetEx] = None
) -> TTransaction:
"""
Get the value of `key` and optionally set its expiration. GETEX is similar to GET.
See https://valkey.io/commands/getex for more details.
Args:
key (str): The key to get.
expiry (Optional[ExpirySet], optional): set expiriation to the given key.
Equivalent to [`EX` | `PX` | `EXAT` | `PXAT` | `PERSIST`] in the Redis API.
Command Response:
Optional[str]:
If `key` exists, return the value stored at `key`
If 'key` does not exist, return 'None'
Since: Redis version 6.2.0.
"""
args = [key]
if expiry is not None:
args.extend(expiry.get_cmd_args())
return self.append_command(RequestType.GetEx, args)


class Transaction(BaseTransaction):
"""
Expand Down
75 changes: 75 additions & 0 deletions python/python/tests/test_async_client.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,8 +16,10 @@
from glide.async_commands.core import (
ConditionalChange,
ExpireOptions,
ExpiryGetEx,
ExpirySet,
ExpiryType,
ExpiryTypeGetEx,
InfBound,
InfoSection,
InsertPosition,
Expand Down Expand Up @@ -4543,6 +4545,41 @@ async def test_object_refcount(self, redis_client: TRedisClient):
refcount = await redis_client.object_refcount(string_key)
assert refcount is not None and refcount >= 0

@pytest.mark.parametrize("cluster_mode", [True, False])
@pytest.mark.parametrize("protocol", [ProtocolVersion.RESP2, ProtocolVersion.RESP3])
async def test_getex(self, redis_client: TRedisClient):
min_version = "6.2.0"
if await check_if_server_version_lt(redis_client, min_version):
return pytest.mark.skip(reason=f"Redis version required >= {min_version}")

key1 = get_random_string(10)
non_existing_key = get_random_string(10)
value = get_random_string(10)

assert await redis_client.set(key1, value) == OK
assert await redis_client.getex(non_existing_key) is None
assert await redis_client.getex(key1) == value
assert await redis_client.ttl(key1) == -1

# setting expiration timer
assert (
await redis_client.getex(key1, ExpiryGetEx(ExpiryTypeGetEx.MILLSEC, 50))
== value
)
assert await redis_client.ttl(key1) != -1

# setting and clearing expiration timer
assert await redis_client.set(key1, value) == OK
assert (
await redis_client.getex(key1, ExpiryGetEx(ExpiryTypeGetEx.SEC, 10))
== value
)
assert (
await redis_client.getex(key1, ExpiryGetEx(ExpiryTypeGetEx.PERSIST, None))
== value
)
assert await redis_client.ttl(key1) == -1


class TestMultiKeyCommandCrossSlot:
@pytest.mark.parametrize("cluster_mode", [True])
Expand Down Expand Up @@ -4651,6 +4688,44 @@ def test_expiry_cmd_args(self):
)
assert exp_unix_millisec_datetime.get_cmd_args() == ["PXAT", "1682639759342"]

def test_get_expiry_cmd_args(self):
exp_sec = ExpiryGetEx(ExpiryTypeGetEx.SEC, 5)
assert exp_sec.get_cmd_args() == ["EX", "5"]

exp_sec_timedelta = ExpiryGetEx(ExpiryTypeGetEx.SEC, timedelta(seconds=5))
assert exp_sec_timedelta.get_cmd_args() == ["EX", "5"]

exp_millsec = ExpiryGetEx(ExpiryTypeGetEx.MILLSEC, 5)
assert exp_millsec.get_cmd_args() == ["PX", "5"]

exp_millsec_timedelta = ExpiryGetEx(
ExpiryTypeGetEx.MILLSEC, timedelta(seconds=5)
)
assert exp_millsec_timedelta.get_cmd_args() == ["PX", "5000"]

exp_millsec_timedelta = ExpiryGetEx(
ExpiryTypeGetEx.MILLSEC, timedelta(seconds=5)
)
assert exp_millsec_timedelta.get_cmd_args() == ["PX", "5000"]

exp_unix_sec = ExpiryGetEx(ExpiryTypeGetEx.UNIX_SEC, 1682575739)
assert exp_unix_sec.get_cmd_args() == ["EXAT", "1682575739"]

exp_unix_sec_datetime = ExpiryGetEx(
ExpiryTypeGetEx.UNIX_SEC,
datetime(2023, 4, 27, 23, 55, 59, 342380, timezone.utc),
)
assert exp_unix_sec_datetime.get_cmd_args() == ["EXAT", "1682639759"]

exp_unix_millisec = ExpiryGetEx(ExpiryTypeGetEx.UNIX_MILLSEC, 1682586559964)
assert exp_unix_millisec.get_cmd_args() == ["PXAT", "1682586559964"]

exp_unix_millisec_datetime = ExpiryGetEx(
ExpiryTypeGetEx.UNIX_MILLSEC,
datetime(2023, 4, 27, 23, 55, 59, 342380, timezone.utc),
)
assert exp_unix_millisec_datetime.get_cmd_args() == ["PXAT", "1682639759342"]

def test_expiry_raises_on_value_error(self):
with pytest.raises(ValueError):
ExpirySet(ExpiryType.SEC, 5.5)
Expand Down
18 changes: 17 additions & 1 deletion python/python/tests/test_transaction.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,13 @@
from glide import RequestError
from glide.async_commands.bitmap import BitmapIndexType, OffsetOptions
from glide.async_commands.command_args import Limit, ListDirection, OrderBy
from glide.async_commands.core import InsertPosition, StreamAddOptions, TrimByMinId
from glide.async_commands.core import (
ExpiryGetEx,
ExpiryTypeGetEx,
InsertPosition,
StreamAddOptions,
TrimByMinId,
)
from glide.async_commands.sorted_set import (
AggregationType,
GeoSearchByRadius,
Expand Down Expand Up @@ -58,6 +64,7 @@ async def transaction_test(
key18 = "{{{}}}:{}".format(keyslot, get_random_string(3)) # sort
key19 = "{{{}}}:{}".format(keyslot, get_random_string(3)) # bitmap
key20 = "{{{}}}:{}".format(keyslot, get_random_string(3)) # bitmap
key22 = "{{{}}}:{}".format(keyslot, get_random_string(3)) # getex

value = datetime.now(timezone.utc).strftime("%m/%d/%Y, %H:%M:%S")
value2 = get_random_string(5)
Expand Down Expand Up @@ -420,6 +427,15 @@ async def transaction_test(
)
args.append(4)

min_version = "6.2.0"
if not await check_if_server_version_lt(redis_client, min_version):
transaction.set(key22, "value")
args.append(OK)
transaction.getex(key22)
args.append("value")
transaction.getex(key22, ExpiryGetEx(ExpiryTypeGetEx.SEC, 1))
args.append("value")

min_version = "7.0.0"
if not await check_if_server_version_lt(redis_client, min_version):
transaction.zadd(key16, {"a": 1, "b": 2, "c": 3, "d": 4})
Expand Down

0 comments on commit cfaa599

Please sign in to comment.