from agents.base_agent import BaseAgent_openai
from prompts.checker_agent_prompts import *
from utils.extractors import extract_json
import csv
import os


def get_prompt(agent_name, **params):
    """
    Return prompt for a specific agent

    :param agent_name: name of the agent being checked
    :param params: additional parameters (e.g., the phase in which the agent under test is located)
    :return: prompt for checker
    """
    match agent_name:
        case "validator":
            return CHECK_VALIDATOR_PROMPT
        case "scorer":
            return CHECK_SCORER_PROMPT
        case "insighter":
            return CHECK_INSIGHT_PROMPT
        case "coder":
            match params["task_name"]:
                case "Exploratory data analysis":
                    return CHECK_CODER_PROMPT_EDA
                case "Data preparation and feature engineering":
                    return CHECK_CODER_PROMPT_FEATURE
                case "Model training":
                    return CHECK_CODER_PROMPT_MODELING
                case _:
                    raise KeyError(f"Task name {params['task_name']} didn't find")
        case _:
            raise KeyError(f"Phase {agent_name} was not found")


class CheckerAPI(BaseAgent_openai):
    def _save_to_csv(self, agent_name: str, thoughts: str | None, description: str, is_it_correct: bool):
        filepath = os.path.join(self.config['path_debug_log'], 'checker_metrics.csv')
        os.makedirs(os.path.dirname(filepath), exist_ok=True)
        file_exists = os.path.exists(filepath)
        with open(filepath, mode='a', newline='', encoding='utf-8') as csv_file:
            writer = csv.writer(csv_file, delimiter=',', quotechar='"', quoting=csv.QUOTE_MINIMAL)
            if not file_exists:
                writer.writerow(['agent_name', 'thoughts', 'description', 'is_it_correct'])
            writer.writerow([agent_name, thoughts, description, is_it_correct])

    def check_the_code_for_correctness(self, agent_name, agent_output, **params):
        """
        Check the result of another

        :param agent_name: name of the agent being checked
        :param agent_output: the result of the work of the agent being audited
        :param params: additional parameters
        :return: is_it_correct - Is the generated result correct? , description - description of defects, if any
        """
        prompt = get_prompt(agent_name, **params).format(agent_output=agent_output, **params)
        self.instructions = INSTRUCTIONS
        self.clear_context()
        self.sub_logger.info("🛃 Checker...")

        for attempt in range(self.config["number_of_attempts_checker"]):
            result = self.generate_response(prompt)
            if not result:
                self.add_message_to_context("user", KEYERROR_JSON)
                continue
            json_result = extract_json(result)
            if json_result is None:
                self.add_message_to_context("user", FAILED_JSON)
                continue
            if 'is_it_correct' not in json_result or "description" not in json_result or \
                    not isinstance(json_result['is_it_correct'], bool):
                self.add_message_to_context("user", KEYERROR_JSON)
                continue

            is_correct = json_result['is_it_correct']
            description = json_result['description']

            if "thoughts" in json_result:
                thoughts = json_result['thoughts']
            else:
                thoughts = None

            if not is_correct:
                self.sub_logger.info(f"Checker: {description}")

            self._save_to_csv(agent_name, thoughts, description, is_correct)
            return is_correct, description

        # If the checker failed to generate correct JSON, we consider the check to have been successful
        self._save_to_csv(agent_name, None, "Checker failed to produce valid JSON output.", True)
        return True, None
