"""Provides the LLMEmitter."""
import numpy as np
import re

from ribs._utils import check_batch_shape, check_shape
from ribs.emitters._emitter_base import EmitterBase
from ribs.emitters.operators import _get_op

from code_behaviour import ast_difference_metric
from utils import remove_blank_lines, filter_functions_by_name
from llm_proxy_client import LLMProxyClient

def extract_function_name(code):
    match = re.search(r'def\s+(\w+)\s*\(', code)
    if match:
        return match.group(1)
    return None

class MechanicLLMEmitter(EmitterBase):
    """Emits solutions by using operator provided.

    If the archive is empty and ``self._initial_solutions`` is set, a call to
    :meth:`ask` will return ``self._initial_solutions``. If
    ``self._initial_solutions`` is not set, we operate on self.x0.

    Args:
        archive (ribs.archives.ArchiveBase): An archive to use when creating and
            inserting solutions. For instance, this can be
            :class:`ribs.archives.GridArchive`.
        x0 (numpy.ndarray): Initial solution.
        operator (str): OpenAI, Anthropic, etc.
        operator_kwargs (dict): Additional arguments to pass to the operator.
            See :mod:`ribs.emitters.operators` for the arguments allowed by each
            operator.
        initial_solutions (array-like): An (n, solution_dim) array of solutions
            to be used when the archive is empty. If this argument is None, then
            solutions will be sampled from a Gaussian distribution centered at
            ``x0`` with standard deviation ``sigma``.
        bounds (None or array-like): Bounds of the solution space. Solutions are
            clipped to these bounds. Pass None to indicate there are no bounds.
            Alternatively, pass an array-like to specify the bounds for each
            dim. Each element in this array-like can be None to indicate no
            bound, or a tuple of ``(lower_bound, upper_bound)``, where
            ``lower_bound`` or ``upper_bound`` may be None to indicate no bound.
        batch_size (int): Number of solutions to return in :meth:`ask`.
    Raises:
        ValueError: There is an error in x0 or initial_solutions.
        ValueError: There is an error in the bounds configuration.
    """

    def __init__(self,
                 archive,
                 *,
                 x0=None,
                 initial_solutions=None,
                 bounds=None,
                 mutation_individuals=1,
                 batch_size=64,
                 operator_kwargs=None,
                 operator=None,
                 model=None,
                 mutation_prompt=None):
        self._batch_size = batch_size
        self._x0 = x0
        self._initial_solutions = initial_solutions
        self._operator = operator
        self._mutation_individuals = mutation_individuals
        self._operator_kwargs = operator_kwargs
        self._model = model
        self._mutation_prompt = mutation_prompt
        self.llm_client = LLMProxyClient()
        EmitterBase.__init__(
            self,
            archive,
            solution_dim=archive.solution_dim,
            bounds=bounds,
        )

        if operator is None:
            raise ValueError("Operator must be provided.")

        if x0 is None and initial_solutions is None:
            raise ValueError("Either x0 or initial_solutions must be provided.")
        if x0 is not None and initial_solutions is not None:
            raise ValueError(
                "x0 and initial_solutions cannot both be provided.")

        if x0 is not None:
            #self._x0 = x0
            check_shape(self._x0, "x0", archive.solution_dim,
                        "archive.solution_dim")
        elif initial_solutions is not None:
            check_batch_shape(self._initial_solutions, "initial_solutions",
                              archive.solution_dim, "archive.solution_dim")

        if operator == "openai":
            pass  # We'll use LLMProxyClient instead
        elif operator == "anthropic":
            self._operator = operator
        else:
            raise ValueError("Operator must be 'openai' or 'anthropic'.")

    @property
    def x0(self):
        """numpy.ndarray: Initial Solution (if initial_solutions is not
        set)."""
        
        return self._x0

    @property
    def initial_solutions(self):
        """numpy.ndarray: The initial solutions which are returned when the
        archive is empty (if x0 is not set)."""
        return self._initial_solutions

    @property
    def batch_size(self):
        """int: Number of solutions to return in :meth:`ask`."""
        return self._batch_size
    
    @property
    def mutation_individuals(self):
        """int: Number of solutions to mutate in :meth:`ask`."""
        return self._mutation_individuals
    
    @staticmethod
    def extract_code(text):
        code_block = re.search(r'```(?:python)?(.*?)```', text, re.DOTALL)
        if code_block:
            return code_block.group(1).strip()
        return None
    
    @staticmethod
    def extract_function_name(code):
        match = re.search(r'def\s+(\w+)\s*\(', code)
        if match:
            return match.group(1)
        return None
    
    
    def ask_llm_solutions(self, focus_mechanic, batch_size, return_one=False):
        # Extract all mechanic names from the archive
        all_mechanics = []
        for solution in self.archive.data()["solution"]:
            mechanic_name = self.extract_function_name(solution[0])
            if mechanic_name and mechanic_name != focus_mechanic:
                all_mechanics.append(mechanic_name)
        
        # Remove duplicates and convert to a comma-separated string
        unique_mechanics = list(set(all_mechanics))
        mechanics_str = "\n ".join(unique_mechanics)
        
        # Ask LLM to select the best mechanics
        response = self.llm_client.chat_completion(
            model=self._model,
            messages=[
                {"role": "system", "content": "You are an expert game designer."},
                {"role": "user", "content": f"Given the following game mechanics: {mechanics_str}\n\nPlease select the best {batch_size} mechanics that will create a nice game when combined with the following code for the mechanics:\n{focus_mechanic} Do not include the above mentioned mechanics in your selection. Only return the names of the selected mechanics, separated by commas."}
            ],
            temperature=0.7
        )
        
        selected_mechanics = response['choices'][0]['message']['content'].split(", ")
        
        # Retrieve the selected mechanics from the archive
        selected_solutions = []
        solutions_str = ""
        for solution in self.archive.data()["solution"]:
            mechanic_name = self.extract_function_name(solution[0])
            if mechanic_name in selected_mechanics:
                solutions_str += solution[0] + "\n"
        
        if return_one:
            return solutions_str
        else:
            return solutions_str + "\n" + focus_mechanic
    
    def ask_random_solutions(self, batch_size, focus_mechanic=None):
        is_valid = False
        while not is_valid:
            solution = self.archive.sample_elites(1)["solution"]
            mechanic_name = self.extract_function_name(solution[0][0])
            print("Mechanic name:\n", mechanic_name, "focus_mechanic:\n", focus_mechanic)
            if isinstance(focus_mechanic, list):
                is_valid = mechanic_name not in focus_mechanic
            elif focus_mechanic is not None:
                is_valid = mechanic_name != focus_mechanic
            print("Is valid:\n", is_valid)
            if is_valid:
                return solution[0][0]

    def ask(self, evo_operator, mechanics_in_focus = None):
        """Creates solutions with operator provided.

        If the archive is empty and ``self._initial_solutions`` is set, we
        return ``self._initial_solutions``. If ``self._initial_solutions`` is
        not set and the archive is still empty, we operate on the initial
        solution (x0) provided. Otherwise, we sample parents from the archive
        to be used as input to the operator

        Returns:
            numpy.ndarray: If the archive is not empty, ``(batch_size,
            solution_dim)`` array -- contains ``batch_size`` new solutions to
            evaluate. If the archive is empty, we return
            ``self._initial_solutions``, which might not have ``batch_size``
            solutions.
        """

        if self.archive.empty and self._initial_solutions is not None:
            return self._initial_solutions
        
        if evo_operator == "diversity_mutation":
            selected_individuals = self.archive.sample_elites(self._mutation_individuals)["solution"] 
            solutions = []
            for i in range(self._batch_size):
                solutions_str = ""
                for individual in selected_individuals:
                    solutions_str += individual + "\n"   
                solutions.append(solutions_str)
        elif evo_operator == "compatibility_mutation":
            all_mechanics_list = self.archive.sample_elites(len(self.archive)-1)["solution"] 
            all_mechanics = ""
            for individual in all_mechanics_list:
                all_mechanics += individual[0] + "\n"   
            selected_individuals = filter_functions_by_name(all_mechanics, mechanics_in_focus)
            solutions = [selected_individuals]
        elif evo_operator == "mutation":
            solutions = self.archive.sample_elites(self.batch_size)["solution"] 
        elif evo_operator == "crossover":
            selected_individuals = self.archive.sample_elites(self.batch_size)["index"]
            archive_individuals = self.archive.sample_elites(len(self.archive)-1)["index"]
            behaviour_distance = {}
            solutions = []
            
            for selected_individual in selected_individuals:
                for archive_individual in archive_individuals:
                    if selected_individual != archive_individual:
                        code_similarity = ast_difference_metric(
                            self.archive._store.retrieve(selected_individual)[1]["solution"][0],
                            self.archive._store.retrieve(archive_individual)[1]["solution"][0]
                        )
                        behaviour_distance[code_similarity] = archive_individual
                
                solutions.append(
                    f"{self.archive._store.retrieve(selected_individual)[1]['solution'][0]} \n "
                    f"{self.archive._store.retrieve(list(behaviour_distance.values())[np.argmin(behaviour_distance)])[1]['solution'][0]}"
                )

        edited_solutions = []

        for solution in solutions:
            if self._operator == "openai":
                if evo_operator == "diversity_mutation":
                    response = self.llm_client.chat_completion(
                        model=self._model,
                        messages=[
                            {"role": "system", "content": "You are an amazing game designer, you can create new game mechanics. Remember that the game mechanic function should only take 'self' as parameter."},
                            {"role": "user", "content": f"Create a new game mechanic that is different, in terms of behavior of mechanics, from the ones provided:\n" + solution[0] + "\n Do not make any assumptions, if you want to add a new variable or a new funciton, you should do it within the game mechanic method. The mechanic must return a reward, which is an integer. If a tile is being assumed then it should be defined as a single capital alphabet character and not a word. If a player is being assumed then it should be '@' tile. Remember that the game mechanic function should only take 'self' as parameter. Only output the new game mechanic as Python function, nothing else."}
                        ],
                        temperature=1.0
                    )
                elif evo_operator == "compatibility_mutation":
                    response = self.llm_client.chat_completion(
                        model=self._model,
                        messages=[
                            {"role": "system", "content": "You are an amazing game designer, you can create new game mechanics. Remember that the game mechanic function should only take 'self' as parameter."},
                            {"role": "user", "content": f"Create a new game mechanic that will make the game better when combined with the following game mechanics:\n" + solution + "\n Do not make any assumptions, if you want to add a new variable or a new funciton, you should do it within the game mechanic method. The mechanic must return a reward, which is an integer. If a tile is being assumed then it should be defined as a single capital alphabet character and not a word. If a player is being assumed then it should be '@' tile. Remember that the game mechanic function should only take 'self' as parameter. The name of the mechanic should be coherent with the behaviour of it. Only output the new game mechanic as Python function, nothing else."}
                        ],
                        temperature=1.0
                    )
                elif evo_operator == "mutation":
                    response = self.llm_client.chat_completion(
                        model=self._model,
                        messages=[
                            {"role": "system", "content": "You are an amazing game designer, you can create new game mechanics. Remember that the game mechanic function should only take 'self' as parameter."},
                            {"role": "user", "content": f"Create a new game mechanic from the given mechanic that extends its features:\n" + solution[0] + "\n Do not make any assumptions, if you want to add a new variable or a new function, you should do it within the game mechanic method. The mechanic must return a reward, which is an integer. If a tile is being assumed then it should be defined as a single capital alphabet character and not a word. If a player is being assumed then it should be '@' tile. Remember that the game mechanic function should only take 'self' as parameter. Only output the new game mechanic as Python function, nothing else."}
                        ],
                        temperature=1.0
                    )
                elif evo_operator == "crossover":
                    response = self.llm_client.chat_completion(
                        model=self._model,
                        messages=[
                            {"role": "system", "content": "You are an amazing game designer, you can create new game mechanics. Remember that the game mechanic function should only take 'self' as parameter."},
                            {"role": "user", "content": f"Create a new game mechanic that combines the features of the given two mechanics to create a new game mechanic that combines the behavior of the both of them:\n" + solution + "\n Do not make any assumptions, if you want to add a new variable or a new method, you should do it within the function. The mechanic must return a reward, which is an integer. If a tile is being assumed then it should be defined as a single capital alphabet character and not a word. If a player is being assumed then it should be '@' tile. Remember that the game mechanic function should only take 'self' as parameter. The name of the mechanic should be coherent with the behaviour of it. Only output the new game mechanic as Python function, nothing else."}
                        ],
                        temperature=1.0
                    )

            #print("Response in llm_emitter.py:\n", response)
            
            edited_solution_str = response['choices'][0]['message']['content']
            edited_solution = self.extract_code(edited_solution_str)
            edited_solution = remove_blank_lines(edited_solution)
            edited_solutions.append(edited_solution)

        return edited_solutions, solutions[0]
    


class GameLLMEmitter(EmitterBase):
    """Emits solutions by mutating a 2D grid-based game environment to incorporate new mechanics.

    Args:
        archive (ribs.archives.ArchiveBase): An archive to use when creating and
            inserting solutions.
        x0 (numpy.ndarray): Initial solution.
        initial_solutions (array-like): An (n, solution_dim) array of initial solutions.
        bounds (None or array-like): Bounds of the solution space.
        batch_size (int): Number of solutions to return in :meth:`ask`.
        operator (str): OpenAI, Anthropic, etc.
        model (str): The specific model to use (e.g., "gpt-4").
        operator_kwargs (dict): Additional arguments to pass to the operator.
    """

    def __init__(self,
                 archive,
                 *,
                 x0=None,
                 initial_solutions=None,
                 bounds=None,
                 batch_size=1,
                 operator=None,
                 model=None,
                 operator_kwargs=None):
        self._batch_size = batch_size
        self._x0 = x0
        self._initial_solutions = initial_solutions
        self._operator = operator
        self._model = model
        self._operator_kwargs = operator_kwargs
        self.llm_client = LLMProxyClient()

        EmitterBase.__init__(
            self,
            archive,
            solution_dim=archive.solution_dim,
            bounds=bounds,
        )

        if operator is None:
            raise ValueError("Operator must be provided.")

        if x0 is None and initial_solutions is None:
            raise ValueError("Either x0 or initial_solutions must be provided.")
        if x0 is not None and initial_solutions is not None:
            raise ValueError("x0 and initial_solutions cannot both be provided.")

        if x0 is not None:
            check_shape(self._x0, "x0", archive.solution_dim, "archive.solution_dim")
        elif initial_solutions is not None:
            check_batch_shape(self._initial_solutions, "initial_solutions",
                              archive.solution_dim, "archive.solution_dim")

        if operator == "openai":
            pass  # We'll use LLMProxyClient instead
        elif operator == "anthropic":
            # Import Anthropic library if needed
            pass
        else:
            raise ValueError("Operator must be 'openai' or 'anthropic'.")

    @staticmethod
    def extract_code(text):
        code_block = re.search(r'```(?:python)?(.*?)```', text, re.DOTALL)
        if code_block:
            return code_block.group(1).strip()
        return None
    
    def ask_random_solutions(self, batch_size):
        solutions = self.archive.sample_elites(batch_size)["solution"]
        return solutions

    def ask(self, environment, new_mechanic, tileset):
        """Mutates the 2D grid-based game environment to incorporate the new mechanic.

        Args:
            environment_code (str): The code of the class representing the 2D grid-based environment.
            new_mechanic (str): The code of the new mechanic to be incorporated.

        Returns:
            list: A list of mutated game environment codes.
        """
        if self.archive.empty and self._initial_solutions is not None:
            return self._initial_solutions

        mutated_environments = []

        for _ in range(self._batch_size):
            if self._operator == "openai":
                response = self.llm_client.chat_completion(
                    model=self._model,
                    messages=[
                        {"role": "system", "content": "You are an expert game developer specializing in 2D grid-based games."},
                        {"role": "user", "content": f"Here's the code for a 2D grid-based game environment:\n\n{environment}\n\nAnd here's the code for the whole game:\n\n{game_class}\n\nAnd the name of the new mechanic to incorporate is: {new_mechanic}\n\nPlease modify the game environment to integrate this new mechanic. Here are the the tileset available to you:\n\n{tileset}\n\nEnsure the changes are coherent and maintain the game's integrity. Only output the modified environment, if needed, nothing else. Only output the modified environment, if needed, nothing else."}
                    ],
                    temperature=0.7
                )
                mutated_environment = response['choices'][0]['message']['content']
            elif self._operator == "anthropic":
                # Implement Anthropic API call here
                pass

            mutated_environments.append(mutated_environment)

        return mutated_environments

    @property
    def x0(self):
        """numpy.ndarray: Initial Solution (if initial_solutions is not set)."""
        return self._x0

    @property
    def initial_solutions(self):
        """numpy.ndarray: The initial solutions which are returned when the
        archive is empty (if x0 is not set)."""
        return self._initial_solutions

    @property
    def batch_size(self):
        """int: Number of solutions to return in :meth:`ask`."""
        return self._batch_size

    def _make_llm_request(self, messages, temperature=1.0):
        """Make an LLM request through the proxy."""
        return self.llm_client.chat_completion(
            model=self._model,
            messages=messages,
            temperature=temperature
        )