"""
copied from rsprunner.py of https://web.archive.org/web/20200521083624/http://www.rpscontest.com/leaderboard
"""

from abc import ABC, abstractmethod
from pathlib import Path
from typing import Any, Literal

from pcot.tasks.rsp.greenberg import GreenBerg

from .iocaine import Iocaine

Move = Literal["", "R", "S", "P"]


class Bot(ABC):
    """Basic bot class to wrap bot functions"""

    @abstractmethod
    def reset(self) -> None:
        raise NotImplementedError()

    @property
    @abstractmethod
    def name(self) -> str:
        raise NotImplementedError()

    @abstractmethod
    def get_move(self, input: Move) -> str:
        raise NotImplementedError()


class GreenbergBot(Bot):
    """
    Iocaine Bot
    Id	Symbol
    0   Rock
    1	Paper
    2	Scissors
    """

    def __init__(self):
        """
        name should be a unique identifier and must be a readable
        filename if code is not specified
        """
        self._name = "greenberg"
        self.greenberg = GreenBerg()

    @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 = -1
        elif input == "R":
            i = 0
        elif input == "P":
            i = 1
        elif input == "S":
            i = 2
        else:
            raise RuntimeError(f"Invalid input {input}")

        o = self.greenberg.move(i)

        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.greenberg = GreenBerg()


class IocaineBot(Bot):
    """
    Iocaine Bot
    Id	Symbol
    0   Rock
    1	Paper
    2	Scissors
    """

    def __init__(self):
        """
        name should be a unique identifier and must be a readable
        filename if code is not specified
        """
        self._name = "iocaine"
        self.iocaine = Iocaine()

    @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 = -1
        elif input == "R":
            i = 0
        elif input == "P":
            i = 1
        elif input == "S":
            i = 2
        else:
            raise RuntimeError(f"Invalid input {input}")

        o = self.iocaine.move(i)

        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.iocaine = Iocaine()


class BotFromScript(Bot):
    """Basic bot class to wrap bot functions"""

    def __init__(self, name: str, code: str | None = None):
        """
        name should be a unique identifier and must be a readable
        filename if code is not specified
        """
        self._name = name
        if code is None:
            self.load_code()
        else:
            self.code = code

        self.scope: dict[str, Any] = dict()
        self._code = None

    @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 self._code is None:
            self.compile_code()

        self.scope["input"] = input
        exec(self._code, self.scope)
        self.output = self.scope["output"]
        return self.output

    def compile_code(self) -> None:
        self._code = compile(self.code, "<string>", "exec")

    def reset(self) -> None:
        """Resets bot for another round.  This must be called before trying
        to pass the bot between workers, or you may see obscure errors from failures
        to pickle the bots scope dictionary."""
        self.scope = dict()
        self._code = None

    def load_code(self) -> None:
        """Load bot code from the file specified by the name attribute"""
        with open(self.name, "r") as f:
            self.code = f.read()


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

    d = (Path(__file__) / ".." / "bot_functions").resolve()
    bot1 = BotFromScript(name=str(d / "1st_place.py"))
    bot2 = BotFromScript(name=str(d / "2nd_place.py"))

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