Files
office_translator/core/logging.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

108 lines
3.0 KiB
Python

"""
Centralised logging configuration using structlog.
Goals (Story 6.4: Structlog Integration):
- JSON logs in production with fields: timestamp, level, event, request_id, user_id
- Pretty console logs in development
- Single configuration entry-point called from main.py
- Context helpers to bind request_id / user_id per request
"""
from __future__ import annotations
import logging
import os
import sys
from typing import Optional
import structlog
from structlog.contextvars import bind_contextvars, clear_contextvars, merge_contextvars
def configure_logging(
*,
json_logs: bool = True,
log_level: str = "INFO",
) -> None:
"""
Configure structlog + stdlib logging.
- json_logs=True → JSONRenderer (production, Docker-friendly)
- json_logs=False → ConsoleRenderer (development)
- log_level driven by LOG_LEVEL env (INFO/DEBUG/WARNING/ERROR/CRITICAL)
"""
level = getattr(logging, log_level.upper(), logging.INFO)
timestamper = structlog.processors.TimeStamper(fmt="iso")
# Common processor chain (used both for structlog and stdlib records)
pre_chain = [
merge_contextvars, # adds request_id / user_id when bound
structlog.stdlib.add_log_level,
timestamper,
structlog.processors.format_exc_info,
]
if json_logs:
renderer = structlog.processors.JSONRenderer()
else:
# Developer-friendly console output
renderer = structlog.dev.ConsoleRenderer(colors=True)
processor_formatter = structlog.stdlib.ProcessorFormatter(
processor=renderer,
foreign_pre_chain=pre_chain,
)
handler = logging.StreamHandler(sys.stdout)
handler.setFormatter(processor_formatter)
# Reset root logger handlers and attach our handler
root_logger = logging.getLogger()
root_logger.handlers.clear()
root_logger.addHandler(handler)
root_logger.setLevel(level)
structlog.configure(
processors=pre_chain + [structlog.stdlib.ProcessorFormatter.wrap_for_formatter],
logger_factory=structlog.stdlib.LoggerFactory(),
wrapper_class=structlog.stdlib.BoundLogger,
cache_logger_on_first_use=True,
)
def get_logger(name: Optional[str] = None):
"""
Return a structlog logger.
Fallbacks to stdlib logger if structlog misconfigured for any reason.
"""
try:
return structlog.get_logger(name) if name is not None else structlog.get_logger()
except Exception: # pragma: no cover - defensive fallback
return logging.getLogger(name)
def bind_request_context(
*,
request_id: Optional[str] = None,
user_id: Optional[str] = None,
) -> None:
"""
Bind per-request context so every log line includes these fields.
Safe to call multiple times; only provided keys are updated.
"""
kv = {}
if request_id:
kv["request_id"] = request_id
if user_id:
kv["user_id"] = user_id
if kv:
bind_contextvars(**kv)
def clear_request_context() -> None:
"""Clear contextvars after a request has been processed."""
clear_contextvars()