Structura repo conform plan.md sect. 4, booteaza cu /healthz verde:
- app/main.py: FastAPI (lifespan init_db), /healthz (worker viu + last login + queue), /metrics
- app/api/v1: POST /v1/prezentari (enqueue + dedup idempotency UNIQUE), GET prezentari/{id}, nomenclator, mapari
- app/rar_client.py: client RAR real (login/JWT, nomenclator, postPrezentare, getFinalizate) cu User-Agent obligatoriu (fix WAF 403)
- app/worker: proces separat, claim atomic BEGIN IMMEDIATE, heartbeat, login+send (send dezactivat by default)
- app/web: dashboard Jinja2+HTMX (coada, banner alerta blocate, worker viu/mort, stari empty)
- app/db.py + schema.sql: SQLite WAL, tabele accounts/api_keys/operations_mapping/nomenclator_rar/submissions/worker_heartbeat
- app/idempotency.py + payload.py: hash continut canonic + builder payload (status FINALIZATA, fara tipPrestatie)
- Dockerfile + docker-compose.yml (api+worker, volum SQLite persistent, restart:always)
- tools/import_dbf.py: stub T5
Verificat live: login prin rar_client OK (token 259), nomenclator 18 coduri, worker heartbeat -> /healthz worker_alive=True.
Ramas: T3 validare Pydantic, T4 snapshot payload, T2 reconciliere/retry worker, T5 import DBF, auth API-key, middleware redactare creds, criptare PII.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
32 lines
1.2 KiB
Python
32 lines
1.2 KiB
Python
"""Cheie de idempotenta = hash de continut canonic.
|
|
|
|
RAR nu are camp nr. comanda si accepta duplicate -> dedup-ul e in sarcina noastra
|
|
(plan.md sect. 14). Hash stabil peste o reprezentare canonica a prezentarii.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import hashlib
|
|
import json
|
|
from typing import Any
|
|
|
|
|
|
def idempotency_key(account_id: int | None, prezentare: dict[str, Any]) -> str:
|
|
"""SHA-256 peste (account_id + campurile semnificative ale prezentarii).
|
|
|
|
Exclude obs si b64Image (cosmetice, nu definesc unicitatea declaratiei).
|
|
"""
|
|
canonic = {
|
|
"account_id": account_id,
|
|
"vin": (prezentare.get("vin") or "").strip().upper(),
|
|
"nr_inmatriculare": (prezentare.get("nr_inmatriculare") or "").strip().upper(),
|
|
"data_prestatie": prezentare.get("data_prestatie"),
|
|
"odometru_final": str(prezentare.get("odometru_final") or "").strip(),
|
|
"prestatii": sorted(
|
|
str(p.get("cod_prestatie") if isinstance(p, dict) else getattr(p, "cod_prestatie", ""))
|
|
for p in (prezentare.get("prestatii") or [])
|
|
),
|
|
}
|
|
blob = json.dumps(canonic, sort_keys=True, ensure_ascii=False, separators=(",", ":"))
|
|
return hashlib.sha256(blob.encode("utf-8")).hexdigest()
|