pssecret-server/pssecret_server/main.py

60 lines
1.7 KiB
Python
Raw Normal View History

from typing import Annotated
2025-01-03 11:04:51 +00:00
from cryptography.fernet import Fernet
from fastapi import Depends, FastAPI
from fastapi.exceptions import HTTPException
from redis.asyncio import Redis
2025-01-03 11:04:51 +00:00
from pssecret_server.fernet import get_fernet
2025-01-01 19:01:10 +00:00
from pssecret_server.models import Secret, SecretSaveResult
from pssecret_server.redis_db import get_redis
2025-01-03 11:04:51 +00:00
from pssecret_server.utils import decrypt_secret, encrypt_secret, save_secret
2022-06-12 10:47:07 +00:00
app = FastAPI()
RedisDep = Annotated[Redis, Depends(get_redis)]
2025-01-03 11:04:51 +00:00
FernetDep = Annotated[Fernet, Depends(get_fernet)]
2022-06-12 10:47:07 +00:00
2024-12-26 22:25:54 +00:00
@app.post(
"/secret",
summary="Store secret",
description=(
"Submit secret, it is saved on the server, get retrieval key in response. "
"Use that key to retrieve your data. Key could be used only once, "
"so use it wisely"
),
response_model=SecretSaveResult,
)
2025-01-03 14:32:01 +00:00
async def set_secret(
data: Secret, redis: RedisDep, fernet: FernetDep
) -> dict[str, str]:
2025-01-03 11:04:51 +00:00
data = encrypt_secret(data, fernet)
return {
"key": await save_secret(data, redis),
}
@app.get(
"/secret/{secret_key}",
2024-12-26 22:25:54 +00:00
summary="Retrieve secret",
description=(
"Returns previously saved data if it is still on the server. "
"Could be the other way around in two cases: "
"either it has already been retrieved, either storage timeout has expired"
),
response_model=Secret,
responses={404: {"description": "The item was not found"}},
)
2025-01-03 14:32:01 +00:00
async def get_secret(
secret_key: str, redis: RedisDep, fernet: FernetDep
) -> dict[str, bytes]:
data: bytes | None = await redis.getdel(secret_key)
if data is None:
raise HTTPException(404)
return {
2025-01-03 11:04:51 +00:00
"data": decrypt_secret(data, fernet),
}