import random
import sys
import time
import traceback
from logging import getLogger

from pcot.tasks.rsp.bot import Bot

logger = getLogger(__name__)


class ContestResult:
    """Used to track and report on the status of a contest.  Shared values
    are reported from the perspective of bot1.  For example, score > 0 indicates
    that bot1 won by that many points.  score < 0 indicates bot 1 lost by that
    many points."""

    # TODO bot-specific data should be a seperate object.  ContestResult
    # should track two of these objects and move most of the bot-specific
    # data below into them.
    def __init__(self, bot1: Bot, bot2: Bot):
        self.bot1 = bot1
        self.bot2 = bot2
        self.bot1_disqualified = False
        self.bot2_disqualified = False
        self.finalized = False
        self.errors = False
        self.error_string = ""

        self.wins1 = 0
        self.wins2 = 0
        self.ties1 = 0
        self.ties2 = 0
        self.losses1 = 0
        self.losses2 = 0
        self.score = 0
        self.played = 0
        self.history1: list[str] | str = []
        self.history2: list[str] | str = []
        self.score_history: list[int] = []
        self.start_time = 0
        self.end_time = 0
        self.run_time = 0.0
        self.winner = None
        self.loser = None

        self.scoring = {
            "R": {"R": 0, "P": -1, "S": 1},
            "P": {"R": 1, "P": 0, "S": -1},
            "S": {"R": -1, "P": 1, "S": 0},
        }

    def start(self):
        self.start_time = time.time()

    def score_moves(self, move1: str, move2: str):
        """This function is called to score and track each pair of moves
        from a contest."""

        score = 0
        try:
            score = self.scoring[move1][move2]
        except KeyError:
            # TODO disqualify bot and exit contest
            if move1 not in "RPS":
                score = -1
            elif move2 not in "RPS":
                score = 1
            else:
                raise Exception("Can't score %s and %s?!" % (move1, move2))

        if score > 0:
            self.wins1 += 1
            self.losses2 += 1
        elif score < 0:
            self.losses1 += 1
            self.wins2 += 1
        else:
            self.ties1 += 1
            self.ties2 += 1

        self.score += score
        self.history1.append(move1)
        self.history2.append(move2)
        self.score_history.append(score)
        self.played += 1

        return score

    def finalize(self, errors: bool = False, error_string: str = ""):
        """Called once a contest is complete to do some final bookkeeping.
        This is REQUIRED if multiprocessing features are in use."""
        # the bots must be reset before being passed between workers
        # see comments under Bot.reset()
        self.bot1.reset()
        self.bot2.reset()

        self.errors = errors
        self.error_string = error_string
        self.history1 = "".join(self.history1)
        self.history2 = "".join(self.history2)
        self.end_time = time.time()
        self.run_time = self.end_time - self.start_time

        if self.wins1 > self.wins2:
            self.winner = self.bot1
            self.loser = self.bot2
        elif self.wins1 < self.wins2:
            self.winner = self.bot2
            self.loser = self.bot1

        self.finalized = True

    def __str__(self):
        game = "%s vs %s:" % (self.bot1.name, self.bot2.name)
        if self.bot1_disqualified:
            return "%s bot 1 disqualified" % game
        elif self.bot2_disqualified:
            return "%s bot 2 disqualified" % game
        elif self.finalized:
            return "%s score %d, took %.2f seconds; %s won" % (
                game,
                self.score,
                self.run_time,
                self.winner.name if self.winner else None,
            )
        else:
            return "%s score %d -- not final" % (game, self.score)

    def to_dict(self) -> dict[str, str | int | float | None]:
        if self.bot1_disqualified:
            return {"won_bot": self.bot2.name, "reason": "disqualify"}
        elif self.bot2_disqualified:
            return {"won_bot": self.bot1.name, "reason": "disqualify"}
        elif self.finalized:
            if self.winner is None:
                return {
                    "won_bot": None,
                    "reason": "draw",
                    "score": self.score,
                    "run_time": self.run_time,
                }
            else:
                return {
                    "won_bot": self.winner.name,
                    "reason": "beat",
                    "score": self.score,
                    "run_time": self.run_time,
                }
        else:
            return {"won_bot": None, "reason": "not_final"}


class Contest:
    """Contest object handles running a contest between two sets of bots."""

    def __init__(self, bot1: Bot, bot2: Bot, rounds: int = 1000):
        self.bot1 = bot1
        self.bot2 = bot2
        self.rounds = rounds
        self.result = ContestResult(bot1, bot2)

        # isolate random number generator
        r1 = random.random()
        r2 = random.random()
        base_rng = random.getstate()

        random.seed(r1)
        self.bot1_rng = random.getstate()

        random.seed(r2)
        self.bot2_rng = random.getstate()

        random.setstate(base_rng)

    def run(self) -> ContestResult:
        """Runs the configured contest and reports a ContestResult"""
        self.result.start()

        base_rng = random.getstate()
        input1 = input2 = output1 = output2 = ""
        errors = False
        error_string = ""
        for _ in range(self.rounds):
            random.setstate(self.bot1_rng)
            try:
                output1 = self.bot1.get_move(input1)
            except KeyboardInterrupt:
                raise
            except:
                exc_type, exc_value, exc_traceback = sys.exc_info()
                exc_string = "".join(
                    traceback.format_exception(exc_type, exc_value, exc_traceback)
                )
                error_string = "Error from %s\n%s" % (self.bot1.name, exc_string)
                logger.info(error_string)
                errors = True
                self.result.bot1_disqualified = True
            else:
                if output1 not in "RPS":
                    errors = True
                    self.result.bot1_disqualified = True
                    error_string = "bot1 did not make a valid move"
            self.bot1_rng = random.getstate()

            random.setstate(self.bot2_rng)
            try:
                output2 = self.bot2.get_move(input2)
            except KeyboardInterrupt:
                raise
            except:
                exc_type, exc_value, exc_traceback = sys.exc_info()
                exc_string = "".join(
                    traceback.format_exception(exc_type, exc_value, exc_traceback)
                )
                error_string = "Error from %s\n%s" % (self.bot1.name, exc_string)
                logger.info(error_string)
                errors = True
                self.result.bot2_disqualified = True
            else:
                if output2 not in "RPS":
                    errors = True
                    self.result.bot2_disqualified = True
                    error_string = "bot2 did not make a valid move"
            self.bot2_rng = random.getstate()

            if errors:
                break

            self.result.score_moves(output1, output2)
            input1 = output2
            input2 = output1

            # TODO add early bail out like official contest

        self.result.finalize(errors=errors, error_string=error_string)

        random.setstate(base_rng)
        return self.result
