Files
office_translator/core/redis.py
Sepehr Ramezani 26bd096a06 feat: production deployment - full update with providers, admin, glossaries, pricing, tests
Major changes across backend, frontend, infrastructure:
- Provider system with model selection (Google, DeepL, OpenAI, Ollama, Google Cloud)
- Admin panel: user management, pricing, settings
- Glossary system with CSV import/export
- Subscription and tier quota management
- Security hardening (rate limiting, API key auth, path traversal fixes)
- Docker compose for dev, prod, and IONOS deployment
- Alembic migrations for new tables
- Frontend: dashboard, pricing page, landing page, i18n (en/fr)
- Test suite and verification scripts

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-04-25 15:01:47 +02:00

135 lines
4.0 KiB
Python

"""
Centralized Redis client for the application.
Single shared async and sync clients; REDIS_URL from environment.
Used by: rate limiting, tier quota, storage tracker, auth blocklist, admin sessions, health check.
"""
from __future__ import annotations
import os
import logging
logger = logging.getLogger(__name__)
_async_client = None
_sync_client = None
# Sentinel: False = no REDIS_URL or connection failed; None = not yet tried
_async_attempted = None
_sync_attempted = None
def get_redis_url() -> str:
"""Return REDIS_URL from environment (empty if not set)."""
return (os.getenv("REDIS_URL") or "").strip()
def get_async_redis():
"""
Return shared async Redis client or None.
Uses REDIS_URL from env. Lazy init; on failure returns None (graceful degradation).
"""
global _async_client, _async_attempted
if _async_attempted is not None:
return _async_client if _async_client is not None else None
url = get_redis_url()
if not url:
_async_attempted = True
return None
try:
import redis.asyncio as redis
_async_client = redis.Redis.from_url(url, decode_responses=True)
logger.info("Redis async client connected (shared)")
_async_attempted = True
return _async_client
except Exception as e:
logger.warning("Redis async connection failed: %s", e)
_async_client = None
_async_attempted = True
return None
def get_sync_redis():
"""
Return shared sync Redis client or None.
Uses REDIS_URL from env. Lazy init; on failure returns None.
Used by health check, admin sessions, auth blocklist.
"""
global _sync_client, _sync_attempted
if _sync_attempted is not None:
return _sync_client if _sync_client is not None else None
url = get_redis_url()
if not url:
_sync_attempted = True
return None
try:
import redis
_sync_client = redis.Redis.from_url(
url, decode_responses=True, socket_connect_timeout=5
)
_sync_client.ping()
logger.info("Redis sync client connected (shared)")
_sync_attempted = True
return _sync_client
except Exception as e:
logger.warning("Redis sync connection failed: %s", e)
_sync_client = None
_sync_attempted = True
return None
def ping_sync() -> tuple[bool, str]:
"""
Ping Redis (sync). Returns (success, error_message).
Use for health/readiness checks.
"""
client = get_sync_redis()
if not client:
return False, "not_configured"
try:
client.ping()
return True, ""
except Exception as e:
return False, str(e)
# ---------------------------------------------------------------------------
# Translation job status cache (for GET /api/v1/translations/{id})
# Key: translation:job:{job_id}, value: JSON, TTL e.g. 2h
# ---------------------------------------------------------------------------
JOB_STATUS_KEY_PREFIX = "translation:job"
JOB_STATUS_TTL = 2 * 3600 # 2 hours
async def set_job_status_async(job_id: str, data: dict, ttl: int = JOB_STATUS_TTL) -> bool:
"""Store job status in Redis for multi-instance progress polling. Returns True if stored."""
client = get_async_redis()
if not client:
return False
try:
import json
key = f"{JOB_STATUS_KEY_PREFIX}:{job_id}"
await client.set(key, json.dumps(data), ex=ttl)
return True
except Exception as e:
logger.warning("Redis set_job_status_async failed for %s: %s", job_id, e)
return False
async def get_job_status_async(job_id: str) -> dict | None:
"""Retrieve job status from Redis. Returns None if not found or Redis unavailable."""
client = get_async_redis()
if not client:
return None
try:
import json
key = f"{JOB_STATUS_KEY_PREFIX}:{job_id}"
data = await client.get(key)
return json.loads(data) if data else None
except Exception as e:
logger.warning("Redis get_job_status_async failed for %s: %s", job_id, e)
return None