import os, time, uuid, orjson as json
from typing import Any, Dict
from config import CFG

os.makedirs(CFG.LEDGER_DIR, exist_ok=True)

class DPLedger:
    def __init__(self, mode:str, claim_type:str, graph_meta:dict, seed:int):
        self.data: Dict[str, Any] = {
            "request_id": str(uuid.uuid4()),
            "mode": mode,                         # Exact / Surrogate / Fallback
            "claim_type": claim_type,             # RunWiseExact / TruncationOnly / NoCert
            "privacy_scope": "post_processing_only",   # Rev-3 posture for router
            "graph_meta": graph_meta,
            "route": [],
            "per_edge": [],                       # [{edge, sigma, rdp, delta}]
            "router_rdp_eps": None,               # ε from router-internal RDP bookkeeping
            "alpha_selected": None,
            "eps_delta": {"eps": None, "delta": str(CFG.DELTA)},  # stays None unless LDP atoms present
            "U_records": [],                      # [(nid, q064)]
            "N_records": [],                      # [(nid, N, Nub)]
            "budgets": {"eps_used":"0.0","eps_max":f"{CFG.EPS_MAX}",
                        "price_spent":0,"price_max":CFG.PRICE_MAX_CENTS,
                        "lat_ms":0,"slo_ms":CFG.SLO_MS, "budget_event":"None"},
            "guards": [],
            "dkey_pred": "0.0", "dkey_real": "0.0",
            "model_id": "", "adapter_id": "", "dp_cert_id": "",
            "eps_train": "", "delta_train": "",
            "timestamp": time.strftime("%Y-%m-%dT%H:%M:%SZ", time.gmtime()),
            "seed": seed
        }
        self._rdp_sum = 0.0

    # ---- logging helpers ----
    def log_uniform(self, nid:str, q064:int): self.data["U_records"].append([nid, str(q064)])
    def log_counts(self, nid:str, N:int, Nub:int): self.data["N_records"].append([nid, int(N), int(Nub)])
    def log_route_pop(self, nid:str): self.data["route"].append(nid)

    def log_edge(self, edge:str, sigma:float, rdp_edge:float, delta:float):
        self.data["per_edge"].append({"edge":edge,"sigma":round(sigma,3),"rdp":round(rdp_edge,8),"delta":delta})
        self._rdp_sum += rdp_edge

    def set_router_eps(self, eps:float, alpha:int):
        self.data["router_rdp_eps"] = f"{eps:.6f}"
        self.data["alpha_selected"] = int(alpha)

    def set_budget(self, eps_used:float, price:int, lat_ms:int, event:str="None"):
        self.data["budgets"]["eps_used"] = f"{eps_used:.6f}"
        self.data["budgets"]["price_spent"] = int(price)
        self.data["budgets"]["lat_ms"] = int(lat_ms)
        self.data["budgets"]["budget_event"] = event

    def set_model_meta(self, model_id:str="", adapter_id:str="", dp_cert_id:str="", eps_train:str="", delta_train:str=""):
        self.data["model_id"]=model_id; self.data["adapter_id"]=adapter_id; self.data["dp_cert_id"]=dp_cert_id
        self.data["eps_train"]=eps_train; self.data["delta_train"]=delta_train

    def add_guard(self, flag:str):
        if flag not in self.data["guards"]: self.data["guards"].append(flag)

    def save(self) -> str:
        path = os.path.join(CFG.LEDGER_DIR, f"{self.data['request_id']}.json")
        with open(path, "wb") as f: f.write(json.dumps(self.data))
        return path
