2023-05-09 18:17:28 +01:00
|
|
|
from typing import Optional, Tuple, Union
|
2020-08-26 18:04:57 +08:00
|
|
|
|
2023-04-27 16:33:43 +01:00
|
|
|
from redis.asyncio.client import Redis
|
|
|
|
|
from redis.asyncio.cluster import RedisCluster
|
2020-08-26 18:04:57 +08:00
|
|
|
|
2023-05-12 17:20:22 +01:00
|
|
|
from fastapi_cache.types import Backend
|
2020-08-26 18:04:57 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class RedisBackend(Backend):
|
2023-05-12 17:20:22 +01:00
|
|
|
def __init__(self, redis: Union["Redis[bytes]", "RedisCluster[bytes]"]):
|
2020-08-26 18:04:57 +08:00
|
|
|
self.redis = redis
|
2023-04-27 16:33:43 +01:00
|
|
|
self.is_cluster: bool = isinstance(redis, RedisCluster)
|
2020-08-26 18:04:57 +08:00
|
|
|
|
2023-05-09 18:17:28 +01:00
|
|
|
async def get_with_ttl(self, key: str) -> Tuple[int, Optional[bytes]]:
|
2023-01-15 21:54:16 +00:00
|
|
|
async with self.redis.pipeline(transaction=not self.is_cluster) as pipe:
|
2023-05-09 17:08:32 +01:00
|
|
|
return await pipe.ttl(key).get(key).execute() # type: ignore[union-attr,no-any-return]
|
2020-08-26 18:04:57 +08:00
|
|
|
|
2023-05-09 18:17:28 +01:00
|
|
|
async def get(self, key: str) -> Optional[bytes]:
|
2023-05-09 17:08:32 +01:00
|
|
|
return await self.redis.get(key) # type: ignore[union-attr]
|
2020-08-26 18:04:57 +08:00
|
|
|
|
2023-05-09 18:17:28 +01:00
|
|
|
async def set(self, key: str, value: bytes, expire: Optional[int] = None) -> None:
|
2023-05-09 17:08:32 +01:00
|
|
|
await self.redis.set(key, value, ex=expire) # type: ignore[union-attr]
|
2020-11-03 18:08:06 +08:00
|
|
|
|
2022-10-22 20:59:37 +04:00
|
|
|
async def clear(self, namespace: Optional[str] = None, key: Optional[str] = None) -> int:
|
2020-11-03 18:08:06 +08:00
|
|
|
if namespace:
|
|
|
|
|
lua = f"for i, name in ipairs(redis.call('KEYS', '{namespace}:*')) do redis.call('DEL', name); end"
|
2023-05-09 17:08:32 +01:00
|
|
|
return await self.redis.eval(lua, numkeys=0) # type: ignore[union-attr,no-any-return]
|
2020-11-03 18:08:06 +08:00
|
|
|
elif key:
|
2023-05-09 17:08:32 +01:00
|
|
|
return await self.redis.delete(key) # type: ignore[union-attr]
|
2023-02-15 10:45:19 +08:00
|
|
|
return 0
|