2025-01-08 20:44:41 +00:00
|
|
|
from functools import lru_cache
|
2022-12-17 12:44:03 +00:00
|
|
|
from uuid import uuid4
|
|
|
|
|
2025-01-03 15:06:08 +00:00
|
|
|
from cryptography.fernet import Fernet
|
2025-01-01 18:18:38 +00:00
|
|
|
from redis.asyncio import Redis
|
2025-01-08 20:44:41 +00:00
|
|
|
from redis.exceptions import ResponseError
|
|
|
|
from redis.typing import ResponseT
|
2022-12-17 12:44:03 +00:00
|
|
|
|
2025-01-01 19:01:10 +00:00
|
|
|
from pssecret_server.models import Secret
|
2022-12-17 12:44:03 +00:00
|
|
|
|
2025-01-01 18:18:38 +00:00
|
|
|
|
2025-01-03 15:06:08 +00:00
|
|
|
def encrypt_secret(data: Secret, fernet: Fernet) -> Secret:
|
|
|
|
encrypted = fernet.encrypt(data.data.encode()).decode()
|
|
|
|
return Secret(data=encrypted)
|
|
|
|
|
|
|
|
|
|
|
|
def decrypt_secret(secret: bytes, fernet: Fernet) -> bytes:
|
|
|
|
return fernet.decrypt(secret)
|
|
|
|
|
|
|
|
|
2025-01-01 18:18:38 +00:00
|
|
|
async def get_new_key(redis: Redis) -> str:
|
|
|
|
"""Returns free Redis key"""
|
2022-12-17 12:44:03 +00:00
|
|
|
while True:
|
|
|
|
new_key = str(uuid4())
|
|
|
|
|
|
|
|
if not await redis.exists(new_key):
|
|
|
|
return new_key
|
2025-01-01 18:18:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def save_secret(data: Secret, redis: Redis) -> str:
|
|
|
|
"""Save passed data, returns retrieval key"""
|
|
|
|
new_key = await get_new_key(redis)
|
|
|
|
await redis.setex(new_key, 60 * 60 * 24, data.data)
|
|
|
|
|
|
|
|
return new_key
|
2025-01-08 20:44:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
@lru_cache
|
|
|
|
async def _is_getdel_available(redis: Redis) -> bool:
|
2025-01-08 21:19:08 +00:00
|
|
|
"""Checks the availability of GETDEL command on the Redis server instance
|
|
|
|
|
|
|
|
GETDEL is not available in Redis prior to version 6.2
|
|
|
|
"""
|
2025-01-08 20:44:41 +00:00
|
|
|
try:
|
|
|
|
await redis.getdel("test:getdel:availability")
|
|
|
|
except ResponseError:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
async def getdel(redis: Redis, key: str) -> ResponseT:
|
2025-01-08 21:19:08 +00:00
|
|
|
"""Gets the value of key and deletes the key
|
|
|
|
|
|
|
|
Depending on the capabilities of Redis server this function
|
|
|
|
will either call GETDEL command, either first call GETSET with empty string
|
|
|
|
and DEL right after that.
|
|
|
|
"""
|
2025-01-08 20:44:41 +00:00
|
|
|
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
|