"""Load the numeric payoff matrix A and the action-pair matrix M.

Both are built directly from `data/payoff_matrix.csv` + `config.json` via
`src.ingest.matrix_loader`, so the Nash stage is self-contained. The
action-pair matrix M is kept alongside A because A alone cannot recover
Pr[(C, C)] at an equilibrium (the inverse map from payoff value to action
pair is one-to-many).
"""

from __future__ import annotations

import json
from dataclasses import dataclass
from pathlib import Path
from typing import List

import numpy as np  # type: ignore

from src.ingest.matrix_loader import (
    build_payoff_matrix,
    load_action_pair_matrix,
    resolve_undefined,
)
from src.ingest.payoffs import ActionPair


@dataclass(frozen=True)
class LoadBundle:
    bot_names: List[str]
    A: np.ndarray                     # float64; shape (N, N)
    M: List[List[ActionPair]]         # action-pair matrix; M[i][j] = ('C'|'D','C'|'D')
    config: dict                      # parsed config.json
    raw_csv: Path
    config_path: Path


def load_bundle(raw_csv: Path, config_path: Path) -> LoadBundle:
    """Return everything the Nash stage needs in one call."""
    with open(config_path) as f:
        config = json.load(f)
    pd_params = config["prisoners_dilemma"]
    b, c = pd_params["b"], pd_params["c"]

    bot_names, pair_matrix_raw = load_action_pair_matrix(raw_csv)
    M = resolve_undefined(bot_names, pair_matrix_raw, config)
    A = build_payoff_matrix(M, b, c)

    return LoadBundle(
        bot_names=bot_names,
        A=A,
        M=M,
        config=config,
        raw_csv=Path(raw_csv),
        config_path=Path(config_path),
    )


def hash_input_csv(raw_csv: Path) -> str:
    """SHA-256 of the raw action-pair CSV file contents."""
    import hashlib

    h = hashlib.sha256()
    with open(raw_csv, "rb") as f:
        for chunk in iter(lambda: f.read(65536), b""):
            h.update(chunk)
    return h.hexdigest()
