Added support for all Redis versions (>=1.0.0) #5

Merged
root merged 4 commits from support-earlier-redis-versions into main 2025-01-08 21:22:26 +00:00
4 changed files with 84 additions and 3 deletions
Showing only changes of commit 869bfc45ac - Show all commits

View file

@ -1,7 +1,10 @@
from functools import lru_cache
from uuid import uuid4 from uuid import uuid4
from cryptography.fernet import Fernet from cryptography.fernet import Fernet
from redis.asyncio import Redis from redis.asyncio import Redis
from redis.exceptions import ResponseError
from redis.typing import ResponseT
from pssecret_server.models import Secret from pssecret_server.models import Secret
@ -30,3 +33,26 @@ async def save_secret(data: Secret, redis: Redis) -> str:
await redis.setex(new_key, 60 * 60 * 24, data.data) await redis.setex(new_key, 60 * 60 * 24, data.data)
return new_key return new_key
@lru_cache
async def _is_getdel_available(redis: Redis) -> bool:
"""GETDEL is not available in Redis prior to version 6.2"""
try:
await redis.getdel("test:getdel:availability")
except ResponseError:
return False
return True
async def getdel(redis: Redis, key: str) -> ResponseT:
result: ResponseT
if await _is_getdel_available(redis):
result = await redis.getdel(key)
else:
result = await redis.getset(key, "")
await redis.delete(key)
return result

View file

@ -53,3 +53,6 @@ reportUnusedCallResult = "none"
[tool.pytest.ini_options] [tool.pytest.ini_options]
asyncio_mode = "auto" asyncio_mode = "auto"
[tool.isort]
profile = "black"

View file

@ -1,8 +1,8 @@
from unittest.mock import patch from unittest.mock import AsyncMock, Mock, patch
from redis.asyncio import Redis from redis.asyncio import Redis
from pssecret_server.utils import get_new_key, save_secret from pssecret_server.utils import get_new_key, getdel, save_secret
from ..factories import SecretFactory from ..factories import SecretFactory
@ -33,3 +33,35 @@ async def test_save_secret_data(redis_server: Redis) -> None:
assert redis_data is not None assert redis_data is not None
assert redis_data.decode() == secret.data assert redis_data.decode() == secret.data
@patch("pssecret_server.utils._is_getdel_available", side_effect=AsyncMock())
async def test_getdel_when_available(
is_getdel_available: Mock, redis_server: Redis
) -> None:
is_getdel_available.side_effect.return_value = True
test_value = "test_data"
test_key = "test_key"
await redis_server.set(test_key, test_value)
result = await getdel(redis_server, test_key)
assert result.decode() == test_value
assert not await redis_server.exists(test_key)
@patch("pssecret_server.utils._is_getdel_available", side_effect=AsyncMock())
async def test_getdel_when_not_available(
is_getdel_available: Mock, redis_server: Redis
) -> None:
is_getdel_available.side_effect.return_value = False
test_value = "test_data"
test_key = "test_key"
await redis_server.set(test_key, test_value)
result = await getdel(redis_server, test_key)
assert result.decode() == test_value
assert not await redis_server.exists(test_key)

View file

@ -1,7 +1,10 @@
from unittest.mock import AsyncMock
import pytest import pytest
from cryptography.fernet import Fernet, InvalidToken from cryptography.fernet import Fernet, InvalidToken
from redis.exceptions import ResponseError
from pssecret_server.utils import decrypt_secret, encrypt_secret from pssecret_server.utils import _is_getdel_available, decrypt_secret, encrypt_secret
from ..factories import SecretFactory from ..factories import SecretFactory
@ -29,3 +32,20 @@ def test_secret_is_not_decryptable_by_random_key(fernet: Fernet):
with pytest.raises(InvalidToken): with pytest.raises(InvalidToken):
decrypt_secret(encrypted_secret.data.encode(), random_fernet) decrypt_secret(encrypted_secret.data.encode(), random_fernet)
async def test_is_getdel_available_when_supported():
redis = AsyncMock()
result = await _is_getdel_available(redis)
assert result is True
async def test_is_getdel_available_when_not_supported():
redis = AsyncMock()
redis.getdel.side_effect = ResponseError
result = await _is_getdel_available(redis)
assert result is False