from src.nbos.player import Player
from typing import Dict, Literal, List, Callable
from pydantic import BaseModel, ValidationError
import asyncio
from uuid import uuid4
import csv
from pathlib import Path

class Game:
    def __init__(
        self,
        man_model: str,
        woman_model: str,
        game_id: str = None,
        is_man_initiator: bool = True,
        total_messages: int = 0,
        temperature: float = 0.7,
        total_rounds: int = 10,
        version: str = "classic"
    ):
        self.game_id = game_id if game_id is not None else str(uuid4())
        self.man_model = man_model
        self.woman_model = woman_model
        self.temperature = temperature
        self.debug = False
        self.total_messages = total_messages
        self.total_rounds = total_rounds
        self.is_man_initiator = is_man_initiator
        self.man = Player(
            model=man_model,
            negotiation_role="initiator" if is_man_initiator else "responder",
            game_role="m",
            version=version,
            temperature=temperature,
            total_messages=total_messages,
            total_rounds=total_rounds
        )
        self.woman = Player(
            model=woman_model,
            negotiation_role="responder" if is_man_initiator else "initiator",
            game_role="w",
            version=version,
            temperature=temperature,
            total_messages=total_messages,
            total_rounds=total_rounds
        )


    def sanitize_reasoning(self, reasoning: str) -> str:
        sanitized = reasoning.replace('"', '""').replace('\n', ' ').replace('\r', '')
        if sanitized and sanitized[0] in ('=', '+', '-', '@'):
            sanitized = "'" + sanitized
        return f'"{sanitized}"'

    async def play(self):
        # Créer le fichier CSV et écrire l'en-tête
        csv_file = Path(f"game_{self.game_id}.csv")
        with open(csv_file, 'w', newline='') as f:
            writer = csv.writer(f)
            writer.writerow([
                'idGame', 'idRound', 'man_model', 'woman_model', 'is_man_initiator',
                'total_messages', 'total_rounds', 'temperature', 'manMove', 'womanMove',
                'manPrediction', 'womanPrediction', 'manAccuracy', 'womanAccuracy',
                'manPayoff', 'womanPayoff', 'manReasoning', 'womanReasoning'
            ])

        # Negotiate
        remaining_negotiation_round = self.total_messages
        if self.total_messages > 0:
            for round_id in range(1, self.total_messages + 1):
                if round_id % 2 == 1:
                    sender, message = await self.man.negotiate() if self.is_man_initiator else await self.woman.negotiate()
                else:
                    sender, message = await self.woman.negotiate() if self.is_man_initiator else await self.man.negotiate()
                self.man.negotiation_history.append({
                    "Sender": sender,
                    "Message": message
                })
                self.woman.negotiation_history.append({
                    "Sender": sender,
                    "Message": message
                })
                if self.debug:
                    self.print_negotiation_round_summary(round_id, sender, message)

        # Dans la boucle de jeu
        for round_id in range(1, self.total_rounds + 1):
            (man_move, man_prediction, man_reasoning), \
            (woman_move, woman_prediction, woman_reasoning) = await asyncio.gather(
                self.man.play_round(),
                self.woman.play_round()
            )
            man_score = self.man.determine_outcome(man_move, woman_move)
            woman_score = self.woman.determine_outcome(woman_move, man_move)

            self.man.update_score(man_score)
            self.woman.update_score(woman_score)

            if man_prediction == woman_move:
                man_prediction_score_round = 1
                self.man.prediction_score += 1
            else:
                man_prediction_score_round = 0
            if woman_prediction == man_move:
                woman_prediction_score_round = 1
                self.woman.prediction_score += 1
            else:
                woman_prediction_score_round = 0
            self.man.game_history.append({
                "Round": round_id,
                "Agent Move": man_move,
                "Opponent Move": woman_move,
                "Outcome": man_score
            })
            self.woman.game_history.append({
                "Round": round_id,
                "Agent Move": woman_move,
                "Opponent Move": man_move,
                "Outcome": woman_score
            })

            # Écrire la ligne dans le CSV
            with open(csv_file, 'a', newline='') as f:
                writer = csv.writer(f)
                writer.writerow([
                    self.game_id,
                    round_id,
                    self.man_model,
                    self.woman_model,
                    self.is_man_initiator,
                    self.total_messages,
                    self.total_rounds,
                    self.temperature,
                    man_move,
                    woman_move,
                    man_prediction,
                    woman_prediction,
                    man_prediction_score_round,
                    woman_prediction_score_round,
                    man_score,
                    woman_score,
                    self.sanitize_reasoning(man_reasoning),
                    self.sanitize_reasoning(woman_reasoning)

                ])

            if self.debug:
                self.print_game_round_summary(round_id, man_move, man_prediction, man_reasoning, man_score,
                                              woman_move, woman_prediction, woman_reasoning, woman_score)
        if self.debug:
            self.print_final_results()


    def print_negotiation_round_summary(self, round_id, sender, message):
        print(f"\n🔄 Negotiation Round {round_id}:")
        if sender == "M":
            print(f"👨 Man utters: {message}")
        else:
            print(f"👩 Woman utters: {message}")


    def print_game_round_summary(
        self, round_id, man_move, man_prediction, man_reasoning, man_score,
        woman_move, woman_prediction, woman_reasoning, woman_score
    ):
        print(f"\n🔄 Game Round {round_id}:")
        print(f"👨 Man: Move={man_move}, Prediction={man_prediction}, Score={man_score}")
        print(f"   Reasoning: {man_reasoning}")
        print(f"👩 Woman: Move={woman_move}, Prediction={woman_prediction}, Score={woman_score}")
        print(f"   Reasoning: {woman_reasoning}")
        print(f"✅ Prediction Accuracy So Far:")
        print(f"   Man: {self.man.prediction_score}/{round_id} ({self.man.prediction_score / round_id * 100:.1f}%)")
        print(f"   Woman: {self.woman.prediction_score}/{round_id} ({self.woman.prediction_score / round_id * 100:.1f}%)")
        print(f"🎯 Round Scores: Man={man_score}, Woman={woman_score}")
        print(f"🎯 Total Scores: Man={self.man.total_payoff}, Woman={self.woman.total_payoff}")

    def print_final_results(self):
        print(f"\n🏁 Final Results")
        print(f"Total Score: Man = {self.man.total_payoff}, Woman = {self.woman.total_payoff}")
        print(f"Prediction Accuracy:")
        print(f"  Man = {self.man.prediction_score}/{self.total_rounds} ({self.man.prediction_score / self.total_rounds * 100:.1f}%)")
        print(f"  Woman = {self.woman.prediction_score}/{self.total_rounds} ({self.woman.prediction_score / self.total_rounds * 100:.1f}%)")


async def main():
    game = Game(
        man_model= "qwen3", #"gpt-5",#"gpt-4.5-preview-2025-02-27", #"qwen3",
        woman_model= "qwen3", #"gpt-5", #"gpt-4.5-preview-2025-02-27", #"qwen3"
        game_id=1,
        is_man_initiator= True,
        total_messages=3,
        temperature=0.7,
        total_rounds=10,
        version="classic"
    )

    await game.play()

if __name__ == "__main__":
    asyncio.run(main())