import importlib.util
import inspect
import types
from dataclasses import dataclass
from pathlib import Path
from typing import Any

from pcot.tasks.rsp.bot import Bot, Move


@dataclass
class Observation:
    step: int  # step number of an episode, step >= 0
    # last action of opponent, 0 (Rock), 1 (Paper) or 2 (Scissors)
    lastOpponentAction: int | None

    def __getitem__(self, key: str):
        try:
            return getattr(self, key)
        except AttributeError:  # keep the mapping contract
            raise KeyError(key) from None


@dataclass
class Configuration:
    # Number of moves, it is 3 in all our experiments
    signs: int = 3

    def __getitem__(self, key: str):
        try:
            return getattr(self, key)
        except AttributeError:  # keep the mapping contract
            raise KeyError(key) from None


def load_kaggle_agent(path: str) -> types.FunctionType:
    """
    Dynamically import a Python file and return exactly one callable that
    matches the Kaggle RPS signature: (observation, configuration) -> int.

    Parameters
    ----------
    path : str
        Path to the .py file. Can be absolute or relative.

    Returns
    -------
    function
        The detected agent function.

    Raises
    ------
    FileNotFoundError
        If the file does not exist.
    LookupError
        If zero or multiple matching functions are found.
    TypeError
        If the selected object is not callable.
    """
    file_path = Path(path).expanduser().resolve()
    if not file_path.is_file():
        raise FileNotFoundError(file_path)

    # ── dynamic import ─────────────────────────────────────────────
    spec = importlib.util.spec_from_file_location(file_path.stem, file_path)
    module = importlib.util.module_from_spec(spec)  # type: ignore[arg-type]
    spec.loader.exec_module(module)  # type: ignore[union-attr]

    # ── find candidate functions ──────────────────────────────────
    candidates: list[types.FunctionType] = []
    for _, obj in inspect.getmembers(module, inspect.isfunction):
        params = list(inspect.signature(obj).parameters)
        if params[:2] == ["observation", "configuration"] or params[:2] == [
            "obs",
            "conf",
        ]:
            candidates.append(obj)

    if not candidates:
        raise LookupError(
            "No function with (observation, configuration) signature found."
        )
    if len(candidates) > 1:
        names = ", ".join(f.__name__ for f in candidates)
        raise LookupError(f"Multiple candidate functions found: {names}")

    agent_fn = candidates[0]
    if not callable(agent_fn):
        raise TypeError("The selected object is not callable.")

    return agent_fn


class DojoBot(Bot):
    """
    RSP Dojo Bot
    Id	Symbol
    0   Rock
    1	Paper
    2	Scissors
    https://www.kaggle.com/code/chankhavu/rps-dojo
    """

    def __init__(self, script_path: str):
        """
        name should be a unique identifier and must be a readable
        filename if code is not specified
        """
        self._name = Path(script_path).name
        self.script_path = script_path
        self.bot_fn = load_kaggle_agent(self.script_path)

        self.step = 0

    @property
    def name(self) -> str:
        return self._name

    def __eq__(self, other: Any):
        return self.name == other.name

    def get_move(self, input: Move) -> str:
        """Get the next move for the bot given input
        input must be "R", "P", "S" or ""
        """

        if input == "":
            i = None
        elif input == "R":
            i = 0
        elif input == "P":
            i = 1
        elif input == "S":
            i = 2
        else:
            raise RuntimeError(f"Invalid input {input}")

        o = self.bot_fn(
            Observation(step=self.step, lastOpponentAction=i), Configuration()
        )
        self.step += 1

        if o == 0:
            return "R"
        elif o == 1:
            return "P"
        elif o == 2:
            return "S"
        else:
            raise RuntimeError(f"Invalid output {o}")

    def reset(self) -> None:
        self.bot_fn = load_kaggle_agent(self.script_path)

        self.step = 0


if __name__ == "__main__":
    from pcot.tasks.rsp.contest import Contest

    d = (Path(__file__) / ".." / "rsp_dojo_agents").resolve()
    bot1 = DojoBot(str(d / "black_belt_centrifugal_bumblepuppy_v4.py"))
    bot2 = DojoBot(str(d / "black_belt_multi_armed_bandit_v15.py"))

    contest = Contest(bot1=bot1, bot2=bot2, rounds=50)
    result = contest.run()
    print(result)
