import os
import csv
import asyncio
from bos import BoS
from typing import Callable

CSV_FILE_PATH = "../../data/bos/bos.csv"

class BoSExperiment:
    def __init__(self):
        self.debug = False
        self.strategy = False
        self.models = [] # "gpt-4.5-preview-2025-02-27", "mixtral:8x7b", "llama3.3:latest", "deepseek-r1:7b", "mistral-small", "qwen3", "llama3", "deepseek-r1"
        self.roles = ["man", "woman"]
        self.opponent_strategies = {
           "football_opera": self.loop_football_opera,
            "opera_football": self.loop_opera_football
        }
        self.temperature = 0.7
        self.rounds = 10
        self.num_games_per_config = 30
        self.initialize_csv()

    def initialize_csv(self):
        if not os.path.exists(CSV_FILE_PATH):
            os.makedirs(os.path.dirname(CSV_FILE_PATH), exist_ok=True)
            with open(CSV_FILE_PATH, mode="w", newline="") as file:
                writer = csv.writer(file)
                writer.writerow([
                    "idGame", "model", "role", "opponentStrategy", "idRound",
                    "playerMove", "prediction", "opponentMove", "outcomeRound",
                    "currentPlayerScoreGame", "predictionRound", "currentPlayerPredictionScoreGame", "reasoning"
                ])

    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}"'

    def log_to_csv(self, game_id, model, role, opponent_strategy, round_id,
                   agent_move, prediction, opponent_move, outcome,
                   player_score_game, prediction_round_score, prediction_total_score, reasoning):
        sanitized_reasoning = self.sanitize_reasoning(reasoning)
        model_type = model + " strategy" if self.strategy else model
        with open(CSV_FILE_PATH, mode="a", newline="") as file:
            writer = csv.writer(file)
            writer.writerow([
                game_id, model_type, role, opponent_strategy, round_id,
                agent_move, prediction, opponent_move, outcome,
                player_score_game, prediction_round_score, prediction_total_score, sanitized_reasoning
            ])

    async def run_experiment(self):
        game_id = 1
        for model in self.models:
            for role in self.roles:
                for strategy_name, strategy_fn in self.opponent_strategies.items():
                    for _ in range(self.num_games_per_config):
                        await self.run_game(model, role, strategy_name, strategy_fn, game_id)
                        game_id += 1

    async def run_game(self, model, role, opponent_strategy_name, opponent_strategy_fn, game_id):
        game = BoS(
            model=model,
            role=role,
            prediction=True,
            version="classic",
            temperature=self.temperature,
            game_id=game_id,
            opponent_strategy_fn=opponent_strategy_fn,
            strategy=self.strategy,
            total_rounds=self.rounds
        )
        for i in range(1, self.rounds + 1):
            round_data = await game.play_round(i)
            prediction_round_score = 1.0 if round_data.get("Prediction") == round_data.get("Opponent Move") else 0.0
            prediction_total_score = game.prediction_score

            self.log_to_csv(
                game_id, model, role, opponent_strategy_name, i,
                round_data["Agent Move"], round_data["Prediction"],
                round_data["Opponent Move"], round_data["Outcome"],
                game.player_score_game, prediction_round_score, prediction_total_score,
                round_data["Reasoning"]
            )


    def loop_football_opera(self, history):
        return "Football" if len(history) % 2 == 0 else "Opera"


    def loop_opera_football(self, history):
        return "Opera" if len(history) % 2 == 0 else "Football"


if __name__ == "__main__":
    experiment = BoSExperiment()
    asyncio.run(experiment.run_experiment())
    print("BoS experiment completed. Results saved in", CSV_FILE_PATH)