import os
import time
import uuid
import traceback

import numpy as np

from dotenv import load_dotenv
load_dotenv()

from code_behaviour import code_behavior_score
from utils import extract_list, extract_dict, extract_code, add_indented_strings, delete_self_reset, delete_variable, randomly_delete_function, run_code, count_functions
from llm_proxy_client import LLMProxyClient

import ray

from configs import Configs
configs = Configs()
MODEL = configs.model
EXPERIMENT = configs.experiment

# Initialize LLM client
llm_client = LLMProxyClient()

from init_mechanics import mech_1

#@ray.remote
def get_games_scores(mechanics, game=None, displacement=False, generation=None):
    global updated_game_mech_class
    objectives = []
    measures = []

    def class_edit_get_state_method_gen():
        #print("Editing get_state method...")
        class_add_method_prompt = "Given the following get_state method of the class:\n" + get_state_method + "\n And the following game mechanic:\n" + mechanics[0] + "\n Edit get_state method, if required, for the given game mechanic to work. Do not repeat the game mechanic as a method. Only output whole of the edited get_state method and if not edited, just output 'False'. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_method_prompt}],
            temperature=1
        )
        #print("\n")
        #print(response['choices'][0]['message']['content'])
        #print("\n")
        return response['choices'][0]['message']['content'], class_add_method_prompt

    def class_edit_step_method_gen():
        #print("Editing step method...")
        class_add_method_prompt = "Given the following step method of the class:\n" + step_method + "\n And the following game mechanic(s):\n" + mechanics[0] + "\n Edit step method, if required, for the given game mechanic(s) to work. Always add to the existing code. Remember, there is an existing update_player method, no need to add it again. Do not repeat the game mechanic(s) as a method. Only output whole of the edited step method and if not edited, just output 'False'. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_method_prompt}],
            temperature=1
        )
        #print("\n")
        #print(response['choices'][0]['message']['content'])
        #print("\n")
        return response['choices'][0]['message']['content'], class_add_method_prompt

    def class_edit_reset_method_gen():
        #print("Editing reset method...")
        class_add_method_prompt = "Given the following reset method of the class:\n" + reset_method + "\n And the following game mechanic:\n" + mechanics[0] + "\n Edit reset method, if required, for the given game mechanic to work. Always add to the existing code. Do not repeat the game mechanic as a method. Only output the additional new method or a method you updates and if not added, just output 'False'. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_method_prompt}],
            temperature=1
        )
        #print("\n")
        #print(response['choices'][0]['message']['content'])
        #print("\n") 
        return response['choices'][0]['message']['content'], class_add_method_prompt

    def class_edit_render_method_gen():
        #print("Editing render method...")
        class_add_method_prompt = "Given the following render method of the class:\n" + render_method + "\n And the following game mechanic:\n" + mechanics[0] + "\n Edit render method, if required, for the given game mechanic to work. Always add to the existing code. Do not repeat the game mechanic as a method. Only output whole of the edited render method and if not edited, just output 'False'. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_method_prompt}],
            temperature=1
        )
        #print("\n")
        #print(response['choices'][0]['message']['content'])
        #print("\n")
        return response['choices'][0]['message']['content'], class_add_method_prompt
    
    def class_add_methods_gen(other_methods):
        #print("Adding other methods...")
        class_add_method_prompt = "Given the following game mechanic:\n" + mechanics[0] + "\n And the following other methods of the class:\n" + other_methods + "\n Add any additional methods, if required, for the given game mechanic to work. Do not repeat the game mechanic as a method. Only output the additional new method and if not added, just output 'False'. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_method_prompt}],
            temperature=1
        )
        #print("\n")
        #print(response['choices'][0]['message']['content'])
        #print("\n")
        return response['choices'][0]['message']['content'], class_add_method_prompt

    def class_edit_init_method_gen():
        #print("Editing init method...")
        class_add_method_prompt = "Given the following init method of the class:\n" + init_method + "\n And the following game mechanic:\n" + mechanics[0] + "\n Edit init method, if required, for the given game mechanic to work. Always add to the existing code. Do not repeat the game mechanic as a method. Only output whole of the edited init method and if not edited, just output 'False'. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_method_prompt}],
            temperature=1
        )
        #print("\n")
        #print(response['choices'][0]['message']['content'])
        #print("\n")
        return response['choices'][0]['message']['content'], class_add_method_prompt

    def class_add_var_gen():
        #print("Adding variables...")
        class_add_method_prompt = "Given the following game mechanic:\n" + mechanics[0] + "\n Add any new variables, if required, for the given game mechanic to work. Only output the additional new variables and if not added, just output 'False'. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_method_prompt}],
            temperature=1
        )
        #print("\n")
        #print(response['choices'][0]['message']['content'])
        #print("\n")
        return response['choices'][0]['message']['content'], class_add_method_prompt
    
    def class_add_get_action_space_method_gen(action_space):
        #print("Adding get_action_space method...")
        class_add_method_prompt = "Given the following class:\n" + game_mech_class + "\n Also, the following game mechanics:\n" + get_games_scores.latest_methods['mechanic'] + "\n" + mechanics[0] + "\n And the following get_action_space method:\n" + action_space + "\n . Edit the get_action_space method to add more actions, if required, for the given game mechanic to work. Do not repeat the game mechanic as a method. Only output the edited get_action_space method and if not edited, just output 'False'. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_method_prompt}],
            temperature=1
        )
        #print("\n")
        #print(response['choices'][0]['message']['content'])
        #print("\n")
        return response['choices'][0]['message']['content'], class_add_method_prompt
    
    def class_add_get_mechanics_to_action_method_gen(mech_to_action):
        #print("Adding get_mechanics_to_action method...")
        class_add_method_prompt = "Given the following class:\n" + game_mech_class + "\n Also, the following game mechanics:\n" + get_games_scores.latest_methods['mechanic'] + "\n" + mechanics[0] + "\n And the following get_mechanics_to_action method:\n" + mech_to_action + "\n . Edit the get_mechanics_to_action method to add more actions, if required, for the given game mechanic to work. The edition should be the 'mechanic_function_name': action_number. Do not repeat the game mechanic as a method. Only output the edited get_mechanics_to_action method and if not edited, just output 'False'. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_method_prompt}],
            temperature=1
        )
        #print("\n")
        #print(response['choices'][0]['message']['content'])
        #print("\n")
        return response['choices'][0]['message']['content'], class_add_method_prompt
    
    def class_add_is_terminal_method_gen():
        #print("Adding is_terminal method...")
        line_prompt = "Given the game mechanics:\n" + get_games_scores.latest_methods['mechanic'] + "\n" + mechanics[0] + "\n write one line that describes the win condition for the game that will use these mechanics. "
        line_response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": line_prompt}],
            temperature=1
        )
        #print("\n")
        #print(line_response['choices'][0]['message']['content'])
        #print("\n")
        example_is_terminal = r'''def is_terminal(self):
    if condition_1: # condition_1 triggering the action for the one of the mechanic
        if condition_2: # condition_2 triggering the action for the one of the mechanic
            if condition_3: # condition_3 triggering the action for the one of the mechanic
                return True
    return False'''
        is_terminal_prompt = "Given the game mechanics:\n" + get_games_scores.latest_methods['mechanic'] + "\n" + mechanics[0] + "\n and the init function:\n" + get_games_scores.latest_methods['init'] + "\n We want to train an RL agent to play a game that uses these mechanics. The layout of the game is the str_world in the following function:\n"+ get_games_scores.latest_methods['str_world'] +"\n and the following function describes what the tiles mean:\n"+get_games_scores.latest_methods['tiles']+"\nThe following line describes the situation of the win condition of the game:\n"+ "'"+line_response['choices'][0]['message']['content']+"'" +"\nWrite one method of the class which which wraps win condition in it and tells the agent when the game will end. It must focus on the mechanics in the game provided to you. All the mechanics should be used to fulfill the win condition. The name of the method should be is_terminal. Method should only return one boolean variable. Only return the method and nothing else."
        is_terminal_response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": is_terminal_prompt}],
            temperature=1
        )
        #print("\n")
        #print(is_terminal_response['choices'][0]['message']['content'])
        #print("\n")

        name_prompt = "Given the game mechanics:\n" + get_games_scores.latest_methods['mechanic'] + "\n" + mechanics[0] + "\nThe win condition for the game in is_terminal method:\n " + is_terminal_response['choices'][0]['message']['content'] + "\n And the line that explains the win condition:\n " + line_response['choices'][0]['message']['content'] + "\n Give the game a short and cool name that describes the game well. Only strictly output the name and nothing else. Should not have any special characters in the name. Do not highlight the name."
        name_response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": name_prompt}],
            temperature=1
        )
        print("NAME OF THE GAME: ", name_response['choices'][0]['message']['content'])


        return is_terminal_response['choices'][0]['message']['content'], is_terminal_prompt, name_response['choices'][0]['message']['content'], line_response['choices'][0]['message']['content']
    

    def func_make_game_gen():
        #print("Generating make_game function...")
        paths_to_tiles = r'''world_tileset_data/td_items_amulet_gold.png,
        world_tileset_data/td_items_gem_ruby.png,
        world_tileset_data/td_world_crate.png,
        world_tileset_data/tg_world_barrel.png,
        world_tileset_data/tg_world_floor_carpet_d.png,
        world_tileset_data/tg_world_floor_moss_e.png,
        world_tileset_data/tg_world_floor_sand_f.png,
        world_tileset_data/tg_world_floor_panel_steel_c.png,
        character_sprite_data/td_monsters_angel_d2.png,
        character_sprite_data/td_monsters_archer_u2.png,
        character_sprite_data/td_monsters_berserker_d1.png,
        character_sprite_data/td_monsters_demon_l1.png'''


        env_dict_func_prompt = "Given the game mechanics:\n" + mechanics[0] + "\nChange the following env_dict function to cater for the mechanics, if required.:\n" + env_dict_func + " Use the same paths for the same type of tiles that are being added in env_image, or use the following paths:\n"+ paths_to_tiles + "\n You don't have to use the paths provided if not needed. Strictly use the same paths. Do not use any other paths. Only add in env_image if needed. Return the full env_dict function."
        env_dict_func_changed = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": env_dict_func_prompt}],
            temperature=1
        )
        #print("\n")
        #print(env_dict_func_changed['choices'][0]['message']['content'])
        #print("\n")

        str_map_func_prompt = "Given the game mechanics:\n" + mechanics[0] + "\n And the env_dict function, which has the paths for the tiles:\n" + env_dict_func_changed['choices'][0]['message']['content'] + "\n Change the following str_world in the str_map function to cater for the mechanics, if required.:\n" + str_map_func + "\n 'str_world' is the string that represents the 2D game map. Change it if only needed. It must always have 1 and only 1 `@` character which represents the player. Return the full str_map function."
        str_map_func_changed = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": str_map_func_prompt}],
            temperature=1
        )
        #print("\n")
        #print(str_map_func_changed['choices'][0]['message']['content'])
        #print("\n")

        important_tiles_func_prompt = "Given the game mechanics:\n" + mechanics[0] + "\n And the env_dict function, which has the paths for the tiles:\n" + env_dict_func_changed['choices'][0]['message']['content'] + "\n And the str_map function, which has the string representation of the game map:\n" + str_map_func_changed['choices'][0]['message']['content'] + "\n Change the following important_tiles function to cater for the mechanics, if required.:\n" + important_tiles_func + " Return the full important_tiles function. Return all the tile types mentioned in the return statement of the function. Return empty list if the tile type is not needed."
        important_tiles_func_changed = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": important_tiles_func_prompt}],
            temperature=1
        )
        #print("\n")
        #print(important_tiles_func_changed['choices'][0]['message']['content'])
        #print("\n")


        #game_env_call_func_prompt = "Given the init function:\n" + get_games_scores.latest_methods['init'] + "\n And important tiles function:\n" + extract_code(important_tiles_func_changed['choices'][0]['message']['content']) + "\n Change the following function to call the GameEnv class with the appropriate parameters, if needed:\n" + game_env_call_func + "\n Only return the full game_env_call function with only the call to the GameEnv class as the function is structured right now."
        #game_env_call_func_changed = openai.ChatCompletion.create(model=MODEL, messages=[{"role": "user", "content": game_env_call_func_prompt}], temperature=1)
        #print("\n")
        #print(game_env_call_func_changed['choices'][0]['message']['content'])
        #print("\n")

        make_game_func = f'''{extract_code(env_dict_func_changed['choices'][0]['message']['content'])}

{extract_code(str_map_func_changed['choices'][0]['message']['content'])}

{extract_code(important_tiles_func_changed['choices'][0]['message']['content'])}

{str_wo_chars_func}

{pad_rows_func}

{remove_spaces_func}

{remove_extra_player_func}

{ensure_player_exists_func}

def make_game():
    str_world = str_map()

    str_world = remove_spaces(str_world)
    str_world = ensure_player_exists(str_world)
    str_world = remove_extra_players(str_world)
    str_world = pad_rows_to_max_length(str_world)
    
    walkables, non_walkables, interactive_object_tiles, collectible_tiles, npc_tiles, player_tile, enemy_tiles, extra_tiles = important_tiles()
    
    tile_mapping  = {{
        "walkable_tiles": walkables,
        "non_walkable_tiles": non_walkables,
        "interactive_object_tiles": interactive_object_tiles,
        "collectible_tiles": collectible_tiles,
        "npc_tiles": npc_tiles,
        "player_tile": player_tile,
        "enemy_tiles": enemy_tiles,
        "extra_tiles": extra_tiles
    }}

    str_map_wo_chars = create_all_a_map(str_world, interactive_object_tiles)
    
    env_image, _ = env_dict()

    env = GameEnv(walkable_tiles=walkables, 
                  tiles_without_char=str_map_wo_chars, 
                  tiles=env_image, 
                  str_map_without_chars=str_map_wo_chars, 
                  str_map=str_world, 
                  interactive_object_tiles=interactive_object_tiles, 
                  enemy_tiles=enemy_tiles,
                  collectible_tiles=collectible_tiles,
                  extra_tiles=extra_tiles,
                  render_mode="rgb_array")

    mechanics_to_actions = env.get_mechanics_to_action()

    #env = GymCompatibilityWrapper(env)
    return env, str_world, tile_mapping, env_image, mechanics_to_actions'''

        compatibility_wrapper_class = r'''class GymCompatibilityWrapper(gym.Wrapper):
    def step(self, action):
        obs, reward, done, truncated, info = self.env.step(action)
        return obs, reward, done, truncated, info
    def reset(self, **kwargs):
        obs = self.env.reset(**kwargs)
        return obs, {}

    def clone(self):
        new_env = GymCompatibilityWrapper(self.env.clone())
        new_env.action_space = self.action_space
        new_env.observation_space = self.observation_space
        #new_env.reward_range = self.reward_range
        new_env.metadata = self.metadata
        return new_env'''
        
        return make_game_func + "\n" + compatibility_wrapper_class, extract_code(str_map_func_changed['choices'][0]['message']['content']), extract_code(important_tiles_func_changed['choices'][0]['message']['content'])
    
    str_wo_chars_func = r'''def create_all_a_map(str_world, objects):
        lines = str_world.split('\n')
        new_lines = []
        
        for line in lines:
            new_line = ''.join(char if char in objects else 'A' for char in line)
            new_lines.append(new_line)
        
        return '\n'.join(new_lines)''' 
    
    pad_rows_func = r'''def pad_rows_to_max_length(text):
        lines = text.strip().split("\n")
        max_length = max(len(line) for line in lines)
        padded_lines = [line + line[-1] * (max_length - len(line)) if line else "" for line in lines]
        return "\n".join(padded_lines)'''
    
    remove_spaces_func = r'''def remove_spaces(map_str):
        lines = map_str.strip().split('\n')
        cleaned_lines = [line.replace(' ', '') for line in lines]
        return '\n'.join(cleaned_lines)'''
    
    remove_extra_player_func = r'''def remove_extra_players(input_string):
        special_chars = '@'

        first_occurrences = {char: False for char in special_chars}

        new_string = []
        for char in input_string:
            if char in special_chars:
                if not first_occurrences[char]:
                    new_string.append(char)
                    first_occurrences[char] = True
            else:
                new_string.append(char)

        return ''.join(new_string)'''
    
    ensure_player_exists_func = r'''def ensure_player_exists(str_world):
        lines = str_world.strip().split('\n')
        has_player = any('@' in line for line in lines)
        if not has_player:
            grid = [list(line.strip()) for line in lines]
            a_positions = []
            for i, row in enumerate(grid):
                for j, char in enumerate(row):
                    if char == 'A':
                        a_positions.append((i, j))
            if a_positions:
                import random
                i, j = random.choice(a_positions)
                grid[i][j] = '@'
                return '\n'.join(''.join(row) for row in grid)
        return str_world'''

    env_dict_func = '''def env_dict():
    env_image = dict()
    image_paths = dict()  # New dictionary to store paths
    
    # Define a function to load image and store path
    def load_image(char, path):
        env_image[char] = Image.open(path).convert("RGBA")
        image_paths[char] = path  # Store the path
    
    # Load all images
    base_path = r"/gmd"
    load_image("A", f"{base_path}/world_tileset_data/td_world_floor_grass_c.png")
    load_image("B", f"{base_path}/world_tileset_data/td_world_wall_stone_h_a.png")
    load_image("X", f"{base_path}/world_tileset_data/td_world_floor_grass_c.png")
    load_image("O", f"{base_path}/world_tileset_data/td_world_chest.png")
    load_image("I", f"{base_path}/world_tileset_data/td_world_chest.png")
    load_image("C", f"{base_path}/world_tileset_data/td_world_chest.png")
    load_image("@", f"{base_path}/character_sprite_data/td_monsters_archer_d1.png")
    load_image("#", f"{base_path}/character_sprite_data/td_monsters_witch_d1.png")
    load_image("&", f"{base_path}/character_sprite_data/td_monsters_goblin_captain_d1.png")

    # Here you add any other tiles that are needed for the game in the same format
    
    return env_image, image_paths '''

    str_map_func = '''def str_map():

        str_world = """AAAAAAAAAAAAAAAAAA
    AAAAAAAAAAAAAAAAAA
    AAA@OAXAAAAAAAAAAA
    AAAAAAICAAAAAAAAAA
    AAA#AAAAAAAAAAAAAA
    AAAAAAAAAAAAAAAAAA"""
    
        return str_world'''

    important_tiles_func = '''def important_tiles():
        walkables = ['A']  # Walkable tiles
        non_walkables = ['B']  # Non-walkable tiles
        interactive_object_tiles = ['O', 'I', 'C']  # Interactive objects (e.g., chests)
        collectible_tiles = []  # Can add collectible tiles if needed
        npc_tiles = []  # Assume there are no NPCs represented in the current setup
        player_tile = ['@']  # Player tile
        enemy_tiles = ['#', '&']  # Enemy tiles
        extra_tiles = [] # any other type of tiles for the game goes here
        return walkables, non_walkables, interactive_object_tiles, collectible_tiles, npc_tiles, player_tile, enemy_tiles, extra_tiles'''

    mcts_imports = r'''
import os
import sys
root_dir = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..'))
sys.path.append(root_dir)
from mech_mcts import PlayMCTS, calculate_mechanic_fitness, RandomAgent
from utils import extract_list
import numpy as np
import json
import time
import imageio
from PIL import Image, ImageDraw, ImageFont
from rembg import remove
import math
import copy
import random
import openai
import traceback
from dotenv import load_dotenv
load_dotenv()
openai.api_key = os.getenv("OPENAI_API_KEY")
#import pufferlib
#import pufferlib.emulation
#import pufferlib.environments
#import pufferlib.environment
#import pufferlib.postprocess
#import pufferlib.utils
import gymnasium as gym
from gymnasium import spaces
from gymnasium.envs.registration import register
#from multiprocessing import Pool
import ray
import gc'''

    def agent_body(new_actions, output_path, mcts_player, llm_player, game_name, win_condition):
        a = rf'''def visualize_action_sequence(action_sequence, env, output_path):
    env.reset()
    frames = []
    
    frame = env.render()
    if frame is not None:
        frames.append(frame)
    
    for action in action_sequence:
        obs, reward, done, truncated, info = env.step(action)
        frame = env.render()
        if frame is not None:
            frames.append(frame)
        
        if done:
            break
    
    if frames:
    
        frames.extend([frames[-1]] * 10)
        imageio.mimsave(
            output_path,
            frames,
            fps=2
        )
        #print(f"GIF saved to {{output_path}}")
    else:
        print("No frames were captured")
        
try:
    game_env, str_world, tile_mapping, env_image, mechanics_to_actions = make_game()  # Initialize with appropriate parameters
    done = False
    total_reward = 0
    game_env.reset()
except Exception as e:
    print(f"An error occurred during make_game initialization: {{e}}")
    #traceback.print_exc() 

if {llm_player}:
    action_prompt = rf"""You are a game-playing agent. You are playing the game: {game_name}. The win condition of the game is: '{win_condition}'. Now, Given the following state of the game:
{{str_world}}
and tile mappings for the alphanumeric characters:
{{tile_mapping}}
The player you are controlling is represented as `@`. The following function describes the possible actions you can choose from:
{{mechanics_to_actions}}
Create a sequece of actions that will lead to the win condition being met. Return a list of integers that corresponds to the actions in the dictionary provided to you. Strictly only return the list and nothing else."""
    MODEL = "gpt-4-turbo-2024-04-09"
    actions = llm_client.chat_completion(
        model=MODEL,
        messages=[{{"role": "user", "content": action_prompt}}],
        temperature=1
    )
    print("\n")
    print(actions['choices'][0]['message']['content'])
    print("\n")
    best_rewards = []
    llm_actions = extract_list(actions['choices'][0]['message']['content'])
    for action in llm_actions:
        obs, reward, done, truncated, info = game_env.step(action)
        best_rewards.append(reward)
    best_action_sequence = llm_actions

if {mcts_player}:

    try:
        # Run 3 MCTSs with different iteration counts

        mcts_1 = PlayMCTS(game_env, iterations=100001)
        #mcts_2 = PlayMCTS(game_env, iterations=1000) 
        #mcts_3 = PlayMCTS(game_env, iterations=10000)
        #mcts_4 = PlayMCTS(game_env, iterations=100000)
        #mcts_5 = PlayMCTS(game_env, iterations=100000)
        
        random_agent = RandomAgent(game_env)
        
        start_time = time.time()

        mcts_results = mcts_1.run()
        
        
        # Run all MCTSs and collect their rewards
        best_action_1, max_depth_1, best_action_sequence_1, action_score_1, best_rewards_1, done_1 = mcts_results[1].values()
        best_action_2, max_depth_2, best_action_sequence_2, action_score_2, best_rewards_2, done_2 = mcts_results[1000].values()
        best_action_3, max_depth_3, best_action_sequence_3, action_score_3, best_rewards_3, done_3 = mcts_results[10000].values()
        best_action_4, max_depth_4, best_action_sequence_4, action_score_4, best_rewards_4, done_4 = mcts_results[100000].values()
        #best_action_5, max_depth_5, best_action_sequence_5, action_score_5, best_rewards_5, done_5 = mcts_5.run()
        _, _, random_action_sequence, random_score, random_rewards, done_random = random_agent.run()
        #print("Random agent run successfully")

        # Inside the agent_body function, replace the sequential runs with:
        #mcts_results = mcts_1.run()
        #random_rewards = random_agent.run()
        
        # Unpack results
        #mcts_results, \
        #(_, _, random_action_sequence, random_score, random_rewards, done_random) = results

        #best_action_1, max_depth_1, best_action_sequence_1, action_score_1, best_rewards_1, done_1 = mcts_results[1].values()
        #best_action_2, max_depth_2, best_action_sequence_2, action_score_2, best_rewards_2, done_2 = mcts_results[1000].values()
        #best_action_3, max_depth_3, best_action_sequence_3, action_score_3, best_rewards_3, done_3 = mcts_results[10000].values()
        #best_action_4, max_depth_4, best_action_sequence_4, action_score_4, best_rewards_4, done_4 = mcts_results[100000].values()

        #print("Random agent run successfully")
        #print("MCTSs run successfully")
        #for i in [1, 1000, 10000, 100000]:
        #    print(f"\n--- MCTS at iteration {{i}} ---")
        #    print(f"Best action: {{mcts_results[i]['action']}}")
        #    print(f"Max depth reached: {{mcts_results[i]['max_depth']}}")
        #    print(f"Best action sequence: {{mcts_results[i]['action_sequence']}}")
        #    print(f"Action score: {{mcts_results[i]['action_score']}}")
        #    print(f"Best rewards: {{mcts_results[i]['rewards']}}")
        #    print(f"Done: {{mcts_results[i]['done']}}")
        #    print("----------------------------\n")


        total_rewards = [
            (sum(best_rewards_1), 5),
            (sum(random_rewards), 4),
            (sum(best_rewards_2), 3),
            (sum(best_rewards_3), 2),
            (sum(best_rewards_4), 1),
            #(sum(best_rewards_5), 1),
        ]

        #print(f"best_rewards_1: {{best_rewards_1}}")
        #print(f"best_rewards_2: {{best_rewards_2}}")
        #print(f"best_rewards_3: {{best_rewards_3}}")
        #print(f"best_rewards_4: {{best_rewards_4}}")
        #print(f"random_rewards: {{random_rewards}}")
        
        #print("MCTS-1    (1 iter)     : {{:.2f}}  (expected rank: 5)".format(total_rewards[0][0]))
        #print("Random                 : {{:.2f}}  (expected rank: 4)".format(total_rewards[1][0]))
        #print("MCTS-2    (1000 iter) : {{:.2f}}   (expected rank: 3)".format(total_rewards[2][0]))
        #print("MCTS-3    (10000 iter) : {{:.2f}}  (expected rank: 2)".format(total_rewards[3][0]))
        #print("MCTS-4    (100000 iter): {{:.2f}}   (expected rank: 1)".format(total_rewards[4][0]))
        #print("MCTS-5    (100000 iter) : {{:.2f}} (expected rank: 1)".format(total_rewards[5][0]))
        
        # Sort by actual performance
        actual_ranks = sorted(range(len(total_rewards)), key=lambda k: total_rewards[k][0], reverse=True)
        expected_ranks = [item[1] for item in total_rewards]
        
        # Calculate Kendall's Tau
        from scipy.stats import kendalltau
        tau, p_value = kendalltau(actual_ranks, expected_ranks)
        #print(f"Kendall's Tau: {{tau:.4f}} (p-value: {{p_value:.4f}})")

        dones = [done_1, done_2, done_3, done_4, done_random]
        #print(f"done_1: {{done_1}}")
        #print(f"done_2: {{done_2}}")
        #print(f"done_3: {{done_3}}")
        #print(f"done_4: {{done_4}}")
        #print(f"done_5: {{done_5}}")
        #print(f"done_random: {{done_random}}")

        
        # Calculate entropy of rewards from all MCTSs
        all_rewards = best_rewards_1 + best_rewards_2 + best_rewards_3 + best_rewards_4# + best_rewards_5
        rewards_array = np.array(all_rewards)
        # Add small constant to avoid log(0)
        rewards_prob = rewards_array / (np.sum(rewards_array) + 1e-10) + 1e-10
        entropy = -np.sum(rewards_prob * np.log(rewards_prob))
        #print(f"Entropy of rewards: {{entropy}}")
        
        # Use results from largest MCTS for rest of code
        best_rewards = best_rewards_3
        
        end_time = time.time()
        execution_time = end_time - start_time
        print(f"MCTS execution_time: {{execution_time:.4f}} seconds")
    except Exception as e:
        # Handle exceptions here
        print(f"An error occurred during MCTS execution: {{e}}")
        # You could also log the error, set error flags, or take recovery actions
        #traceback.print_exc() 
    finally:
        # Cleanup all MCTS instances
        if 'mcts_1' in locals(): mcts_1.cleanup()
        #if 'mcts_2' in locals(): mcts_2.cleanup()
        #if 'mcts_3' in locals(): mcts_3.cleanup()
        #if 'mcts_4' in locals(): mcts_4.cleanup()
        
        # If RandomAgent has a cleanup method, add it here too
        #if hasattr(random_agent, 'cleanup') and 'random_agent' in locals():
        random_agent.cleanup()
        random_agent = None
    gc.collect()

try: 
    visualize_action_sequence(best_action_sequence_4, game_env, "{output_path}")
    total_mechanics = game_env.action_space.n - 4
    #mechanic_fitness_1 = calculate_mechanic_fitness({str(new_actions)}, best_action_sequence_1, best_rewards_1)
    #mechanic_fitness_2 = calculate_mechanic_fitness({str(new_actions)}, best_action_sequence_2, best_rewards_2)
    #mechanic_fitness_3 = calculate_mechanic_fitness({str(new_actions)}, best_action_sequence_3, best_rewards_3)
    #mechanic_fitness_4 = calculate_mechanic_fitness({str(new_actions)}, best_action_sequence_4, best_rewards_4)
    #print(f"mechanic_fitness_1: {{mechanic_fitness_1}}")
    #print(f"mechanic_fitness_2: {{mechanic_fitness_2}}")
    #print(f"mechanic_fitness_3: {{mechanic_fitness_3}}")
    #print(f"mechanic_fitness_4: {{mechanic_fitness_4}}")
    #mechanic_fitness = sum([mechanic_fitness_1, mechanic_fitness_2, mechanic_fitness_3, mechanic_fitness_4])/4
    #print(f"mechanic_fitness: {{mechanic_fitness}}")
    #print(f"total_mechanics: {{total_mechanics}}")
    #if {mcts_player}:
    #    print(f"max_depth: {{max_depth_3}}")
    #if {llm_player}:
    #    print(f"best_action_sequence: {{best_action_sequence}}")
    #    print(f"best_rewards_sequence: {{best_rewards}}")
    #    print(f"total_reward: {{sum(best_rewards)}}")
    #if {mcts_player}:
    #    print(f"action_score: {{action_score_3:.4f}}")
    #    print(f"execution_time: {{execution_time:.4f}} seconds")
    #print("total score: ", (sum(best_rewards) * max_depth) / (execution_time + 1e-8))
    #print("total score: ", sum(best_rewards))
    mechanics_to_action = game_env.get_mechanics_to_action()
    if {mcts_player}:
        __return_value__ =  (tau, action_score_4, total_mechanics, 0, best_action_sequence_4, str_world, tile_mapping, mechanics_to_action, dones)
    if {llm_player}:
        __return_value__ =  (sum(best_rewards), 0, total_mechanics, mechanic_fitness, best_action_sequence, str_world, tile_mapping, mechanics_to_action, dones)
except Exception as e:
    print(f"An error occurred post MCTS execution: {{e}}")
    # You could also log the error, set error flags, or take recovery actions
    traceback.print_exc() '''

        return a# + "\n" + b + "\n" + c

    game_mech_class = r"""class GameEnv(gym.Env):
        def __init__(self, walkable_tiles,tiles_without_char,  tiles, str_map_without_chars, str_map, interactive_object_tiles, enemy_tiles, collectible_tiles, extras, render_mode="human"):
            super(GameEnv, self).__init__()
            self.map_str_without_chars = str_map_without_chars.strip().split('\n')
            self.map_str = str_map.strip().split('\n')
            self.map = [list(row) for row in self.map_str]
            self.map_without_chars = [list(row) for row in self.map_str_without_chars]
            self.tiles = tiles
            self.tiles_without_char = tiles_without_char
            self.action_space = spaces.Discrete(self.get_action_space())
            self.char_set = {'A': 0, 'B': 1, 'C': 2, 'D': 3, 'O': 4, '@': 5, '#': 6, '&': 7}
            self.char_to_int = lambda c: self.char_set.get(c, 0)
            self.mechanic_to_action = self.get_mechanics_to_action()
            max_width = max(len(row) for row in self.map_str)
            self.tile_size = 16  # or whatever size you want
            self.char_tile_size = 16  # or whatever size you want for characters
            self.frames = []
            self.observation_space = spaces.Box(
                low=0, 
                high=1,
                shape=(len(self.char_set), len(self.map_str), max_width),  # Use len(self.char_set) for channels
                dtype=np.int32
            )
            
            self.default_walkable_tile = 'A'
            self.render_mode = render_mode
            self.walkable_tiles = walkable_tiles
            self.interactive_object_tiles = interactive_object_tiles
            self.enemy_tiles = enemy_tiles
            self.collectible_tiles = collectible_tiles
            self.npc_tiles = ["&"]
            self.enemy_tiles = ["#"]
            self.extras = extras
            self.current_score = 0
            self.objects_on_target = 0
            self.collected_items = 0
            self.player_health = 100
            self.enemy_health = 100
            self.map = [list(row) for row in self.map_str]
            self.grid_width = max(len(row) for row in self.map)
            self.grid_height = len(self.map)
            self.current_tile = self.default_walkable_tile
            self.render_mode = "rgb_array"
            for i, row in enumerate(self.map):
                for j, tile in enumerate(row):
                    if tile == '@':
                        self.player_position = (i, j)
            
            self.reset()

        def step(self, action):
            reward = 0
            if action < 4:  # Movement actions
                self.move_player(action)
            done = self.is_terminal()
            if done:
                reward += 10
            info = {}
            return self.get_state()["map"], reward, done, False, info
                
        def update_player_position(self, new_row, new_col, new_tile):
            # Validate both current and new positions are within bounds
            if not (0 <= new_row < self.grid_height and 0 <= new_col < self.grid_width):
                return
                
            if not (0 <= self.player_position[0] < self.grid_height and 0 <= self.player_position[1] < self.grid_width):
                # If current position is invalid, just set the new position
                self.player_position = (new_row, new_col)
                self.current_tile = new_tile
                self.map[new_row][new_col] = '@'
                return
                    
            if new_tile not in self.walkable_tiles:
                return
                    
            # Reset the player's previous position to the original tile
            self.map[self.player_position[0]][self.player_position[1]] = self.current_tile
            self.map_without_chars[self.player_position[0]][self.player_position[1]] = self.current_tile
            
            # Update the player's position
            self.player_position = (new_row, new_col)
            self.current_tile = new_tile
            self.map[new_row][new_col] = '@'

        def get_state(self):
            return {"map": self.map}

        def render(self, mode='human'):

            env_img = Image.new('RGBA', (len(self.map[0]) * self.tile_size, len(self.map) * self.tile_size))

            # 1st layer: Default walkable tile
            for i in range(len(self.map)):
                for j in range(len(self.map[0])):
                    tile_img = self.tiles[self.default_walkable_tile].resize((self.tile_size, self.tile_size))
                    env_img.paste(tile_img, (j * self.tile_size, i * self.tile_size), tile_img)
            
            # 2nd layer: Map without characters
            for i, row in enumerate(self.map_without_chars):
                for j, tile in enumerate(row):
                    if tile in self.tiles and tile != self.default_walkable_tile:
                        tile_img = self.tiles[tile].resize((self.tile_size, self.tile_size))
                        env_img.paste(tile_img, (j * self.tile_size, i * self.tile_size), tile_img)
            
            # 3rd layer: Characters and objects
            for i, row in enumerate(self.map):
                for j, tile in enumerate(row):
                    if tile in self.tiles and tile not in self.walkable_tiles:
                        if tile.isalpha():
                            tile_img = self.tiles[tile].resize((self.tile_size, self.tile_size))
                        else:
                            tile_img = self.tiles[tile].resize((self.char_tile_size, self.char_tile_size))
                            # Center the character in the tile
                            x_offset = (self.tile_size - self.char_tile_size) // 2
                            y_offset = (self.tile_size - self.char_tile_size) // 2
                            env_img.paste(tile_img, (j * self.tile_size + x_offset, i * self.tile_size + y_offset), tile_img)
            
            frame = np.array(env_img.convert('RGB'))
            self.frames.append(frame)
            return frame
    """

    other_methods = r"""def update_player_position(self, new_row, new_col, new_tile):
    if not (0 <= new_row < self.grid_height and 0 <= new_col < self.grid_width):
        return
        
    if not (0 <= self.player_position[0] < self.grid_height and 0 <= self.player_position[1] < self.grid_width):
        self.player_position = (new_row, new_col)
        self.current_tile = new_tile
        self.map[new_row][new_col] = '@'
        return
            
    if new_tile not in self.walkable_tiles:
        return
            
    self.map[self.player_position[0]][self.player_position[1]] = self.current_tile
    self.map_without_chars[self.player_position[0]][self.player_position[1]] = self.current_tile
    
    self.player_position = (new_row, new_col)
    self.current_tile = new_tile
    self.map[new_row][new_col] = '@'

def find_player_position(self):
    for i, row in enumerate(self.map):
        for j, tile in enumerate(row):
            if tile == '@':
                return (i, j)
    return None
    
def clone(self):
    new_env = object.__new__(GameEnv)
    
    for attr, value in self.__dict__.items():
        if attr not in ['action_space', 'observation_space']:
            setattr(new_env, attr, copy.deepcopy(value))
    
    # Recreate action_space and observation_space
    new_env.action_space = spaces.Discrete(new_env.get_action_space())
    new_env.observation_space = spaces.Box(
        low=0, 
        high=1,
        shape=(len(new_env.char_set), len(new_env.map_str), max(len(row) for row in new_env.map_str)),
        dtype=np.int32
    )
    
    return new_env"""

    get_state_method = r"""def get_state(self):
    return {"map": self.map}"""

    step_method = """def step(self, action):
    reward = 0
    if action < 4:  # Movement actions
        self.move_player(action)
    done = self.is_terminal()
    if done:
        reward += 10
    info = {}
    return self.get_state()["map"], reward, done, False, info
    """

    reset_method = """def reset(self, seed=None):
    self.map = [list(row) for row in self.map_str]
    self.map_without_chars = [list(row) for row in self.map_str_without_chars]
    self.grid_width = max(len(row) for row in self.map)
    self.grid_height = len(self.map)
    for i, row in enumerate(self.map):
        for j, tile in enumerate(row):
            if tile == '@':
                self.player_position = (i, j)
    self.current_tile = self.map_without_chars[self.player_position[0]][self.player_position[1]]  # Set current tile to the player's starting position
    return self.get_state()["map"]"""

    render_method = """def render(self, mode='human'):

    env_img = Image.new('RGBA', (len(self.map[0]) * self.tile_size, len(self.map) * self.tile_size))

    # 1st layer: Default walkable tile
    for i in range(len(self.map)):
        for j in range(len(self.map[0])):
            tile_img = self.tiles[self.default_walkable_tile].resize((self.tile_size, self.tile_size))
            env_img.paste(tile_img, (j * self.tile_size, i * self.tile_size), tile_img)
    
    # 2nd layer: Map without characters
    for i, row in enumerate(self.map_without_chars):
        for j, tile in enumerate(row):
            if tile in self.tiles and tile != self.default_walkable_tile:
                tile_img = self.tiles[tile].resize((self.tile_size, self.tile_size))
                env_img.paste(tile_img, (j * self.tile_size, i * self.tile_size), tile_img)
    
    # 3rd layer: Characters and objects
    for i, row in enumerate(self.map):
        for j, tile in enumerate(row):
            if tile in self.tiles and tile not in self.walkable_tiles:
                if tile.isalpha():
                    tile_img = self.tiles[tile].resize((self.tile_size, self.tile_size))
                else:
                    tile_img = self.tiles[tile].resize((self.char_tile_size, self.char_tile_size))
                    # Center the character in the tile
                    x_offset = (self.tile_size - self.char_tile_size) // 2
                    y_offset = (self.tile_size - self.char_tile_size) // 2
                    env_img.paste(tile_img, (j * self.tile_size + x_offset, i * self.tile_size + y_offset), tile_img)
    
    frame = np.array(env_img.convert('RGB'))
    self.frames.append(frame)
    return frame"""

    get_action_space_method = """def get_action_space(self):
    return 4"""

    get_mechanics_to_action_method = """def get_mechanics_to_action(self):
    return {
        "move_player": 0,  # 0-3 for movement, 0 for Up,
        "move_player": 1,  # 1 for Down,
        "move_player": 2,  # 2 for Left,
        "move_player": 3,  # 3 for Right
    }"""


    init_method = r"""def __init__(self, walkable_tiles,tiles_without_char,  tiles, str_map_without_chars, str_map, interactive_object_tiles, enemy_tiles, collectible_tiles, extra_tiles, render_mode="rgb_array"):
    super(GameEnv, self).__init__()
    self.map_str_without_chars = str_map_without_chars.strip().split('\n')
    self.map_str = str_map.strip().split('\n')
    self.map = [list(row) for row in self.map_str]
    self.map_without_chars = [list(row) for row in self.map_str_without_chars]
    self.tiles = tiles
    self.tiles_without_char = tiles_without_char
    self.action_space = spaces.Discrete(self.get_action_space())
    self.char_set = {'A': 0, 'B': 1, 'C': 2, 'D': 3, 'O': 4, '@': 5, '#': 6, '&': 7}
    self.char_to_int = lambda c: self.char_set.get(c, 0)
    self.mechanic_to_action = self.get_mechanics_to_action()
    max_width = max(len(row) for row in self.map_str)
    self.tile_size = 16  # or whatever size you want
    self.char_tile_size = 16  # or whatever size you want for characters
    self.frames = []
    self.observation_space = spaces.Box(
        low=0, 
        high=1,
        shape=(len(self.char_set), len(self.map_str), max_width),  # Use len(self.char_set) for channels
        dtype=np.int32
    )
    self.render_mode = "rgb_array"
    self.default_walkable_tile = 'A'
    self.walkable_tiles = walkable_tiles
    self.interactive_object_tiles = interactive_object_tiles
    self.enemy_tiles = enemy_tiles
    self.npc_tiles = ["&"]
    self.enemy_tiles = ["#"]
    self.extra_tiles = extra_tiles # any other type of tiles for the game goes here, could be danger, hazard, safe. Anyy kind of tiles
    self.collectible_tiles = collectible_tiles
    self.current_score = 0
    self.objects_on_target = 0
    self.collected_items = 0
    self.player_health = 100
    self.enemy_health = 100
    self.map = [list(row) for row in self.map_str]
    self.grid_width = max(len(row) for row in self.map)
    self.grid_height = len(self.map)
    for i, row in enumerate(self.map):
        for j, tile in enumerate(row):
            if tile == '@':
                self.player_position = (i, j)
    self.current_tile = self.default_walkable_tile
    self.reset()"""


    #str_map_func = '''def str_map():

    #str_world = """BBBBBBBBBBB
    #BAAAAAAAAAB
    #BAA@OAXAAAB
    #BAAAAAICAAB
    #BAA#AAAAAAB
    #BBBBBBBBBBB"""

    #return str_world'''

    important_tiles_func = '''def important_tiles():
    walkables = ['A']  # Walkable tiles
    non_walkables = ['B']  # Non-walkable tiles
    interactive_object_tiles = ['O', 'I', 'C']  # Interactive objects (e.g., chests)
    collectible_tiles = []  # Can add collectible tiles if needed
    npc_tiles = []  # Assume there are no NPCs represented in the current setup
    player_tile = ['@']  # Player tile
    enemy_tiles = ['#', '&']  # Enemy tiles
    extra_tiles = [] # any other type of tiles for the game goes here
    return walkables, non_walkables, interactive_object_tiles, collectible_tiles, npc_tiles, player_tile, enemy_tiles, extra_tiles'''
    
    #def simulation_test(name):
    #    return f"""def simulate_mechanic():
    #        env.reset()
    #        reward = env.{name}()
    #        return reward"""


    #TODO: Add store previous methods to open-endedly construct the mechanic class
    #for idx, mechanic in enumerate(mechanics):
    test_error = ""
    mechanics_error = None
    need_test_fix = False
    mechanics_error_break = False
    #while not mechanics_error_break:
    #TODO: Implement this correctly
    #if mechanics_error:
    #    updated_mechanics = delete_function(mechanics_test.latest_methods['mechanic'], mechanic)
    #    updated_game_mech_class = delete_function(updated_game_mech_class, mechanic)
    #    mechanic = fixing_mechanics(mechanics_error)
    #    if "```python" in mechanic or "```" in mechanic:
    #        mechanic = extract_code(mechanic)
    #    else:
    #        mechanic = mechanic
    #    idx += 0.1
    class_edit_get_state_method = ""
    class_edit_step_method = ""
    class_edit_render_method = ""
    class_add_var = ""
    class_add_method = ""
    class_edit_get_action_space_method = ""
    class_edit_get_mechanics_to_action_method = ""

    if game is None:
        if not hasattr(get_games_scores, 'latest_methods'):
            get_games_scores.latest_methods = {
                'init': init_method,
                'reset': reset_method,
                'get_state': get_state_method,
                'step': step_method,
                'render': render_method,
                'mechanic': "",
                'other': other_methods,
                'get_action_space': get_action_space_method,
                'get_mechanics_to_action': get_mechanics_to_action_method,
                'is_terminal': "",
                'str_world': str_map_func,
                'tiles':important_tiles_func,
                "mechanic_counter": 0
            }

    else:
        if displacement:
            _mechs = randomly_delete_function(game[0]['mechanic'])
        else:
            _mechs = game[0]['mechanic']
        if not hasattr(get_games_scores, 'latest_methods'):
            get_games_scores.latest_methods = {
                'init': game[0]['init'],
                'reset': game[0]['reset'],
                'get_state': game[0]['get_state'],
                'step': game[0]['step'],
                'render': game[0]['render'],
                'mechanic': _mechs,
                'other': game[0]['other'],
                'get_action_space': get_action_space_method,#game[0]['get_action_space'],
                'get_mechanics_to_action': get_mechanics_to_action_method,#game[0]['get_mechanics_to_action'],
                'is_terminal': game[0]['is_terminal'],
                'str_world': game[0]['str_world'],
                'tiles': game[0]['tiles'],
                'mechanic_counter': game[0]['mechanic_counter']
            }

        
    class_edit_get_state_method, class_edit_get_state_method_prompt = class_edit_get_state_method_gen()
    class_edit_step_method, class_edit_step_method_prompt = class_edit_step_method_gen()
    class_edit_render_method, class_edit_render_method_prompt = class_edit_render_method_gen()
    #class_edit_init_method, class_edit_init_method_prompt = class_edit_init_method_gen()
    #class_edit_reset_method, class_edit_reset_method_prompt = class_edit_reset_method_gen()
    class_add_method, class_add_methods_prompt = class_add_methods_gen(get_games_scores.latest_methods["other"])
    class_add_var, class_add_var_prompt = class_add_var_gen()
    class_edit_get_action_space_method, class_edit_get_action_space_method_prompt = class_add_get_action_space_method_gen(get_games_scores.latest_methods["get_action_space"])
    class_edit_get_mechanics_to_action_method, class_edit_get_mechanics_to_action_method_prompt = class_add_get_mechanics_to_action_method_gen(get_games_scores.latest_methods["get_mechanics_to_action"])
    
    
    #print("Mechanic:\n", mechanic)
    updated_game_mech_class = "class GameEnv(gym.Env):\n"
    try:
        # Update init method if new variables are added
        if class_add_var != 'False':
            vars_dict = extract_dict(class_add_var)
            updated_init_method = get_games_scores.latest_methods['init']
            updated_init_method = delete_self_reset(updated_init_method, "self.reset()")
            for var_name, value in vars_dict.items():
                if mechanics_error:
                    if var_name in updated_init_method:
                        updated_init_method = delete_variable(updated_init_method, f"self.{var_name}")
                if isinstance(value, str):
                    updated_init_method = add_indented_strings(updated_init_method, f"self.{var_name} = {value}")
                else:
                    updated_init_method = add_indented_strings(updated_init_method, f"self.{var_name} = {value}")
            updated_init_method = add_indented_strings(updated_init_method, "self.reset()")
            get_games_scores.latest_methods['init'] = updated_init_method
            #print("Variables added to init method:\n", vars_dict)
            #print("\nUpdated init method:\n", updated_init_method)
            
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['init'])
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['reset'])
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, mech_1)
        # Update get_state method if needed
        if class_edit_get_state_method != 'False':
            get_games_scores.latest_methods['get_state'] = extract_code(class_edit_get_state_method) if "```" in class_edit_get_state_method else class_edit_get_state_method
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['get_state'])
        # Update step method if needed
        if class_edit_step_method != 'False':
            get_games_scores.latest_methods['step'] = extract_code(class_edit_step_method) if "```" in class_edit_step_method else class_edit_step_method
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['step'])
        # Update render method if needed
        if class_edit_render_method != 'False':
            get_games_scores.latest_methods['render'] = extract_code(class_edit_render_method) if "```" in class_edit_render_method else class_edit_render_method
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['render'])
        if class_edit_get_action_space_method != 'False':
            get_games_scores.latest_methods['get_action_space'] = extract_code(class_edit_get_action_space_method) if "```" in class_edit_get_action_space_method else class_edit_get_action_space_method
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['get_action_space'])
        #if class_edit_get_mechanics_to_action_method != 'False':
        #    #print(f"Previous get_mechanics_to_action:\ {mechanics_test.latest_methods['get_mechanics_to_action']}")
        #    get_games_scores.latest_methods['get_mechanics_to_action'] = extract_code(class_edit_get_mechanics_to_action_method) if "```" in class_edit_get_mechanics_to_action_method else class_edit_get_mechanics_to_action_method
        #    #print(f"Updated get_mechanics_to_action:\ {mechanics_test.latest_methods['get_mechanics_to_action']}")
        #updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['get_mechanics_to_action'])
        if class_edit_get_mechanics_to_action_method != 'False':
            previous_mechanics_to_action = get_games_scores.latest_methods['get_mechanics_to_action']
            new_mechanics_to_action = extract_code(class_edit_get_mechanics_to_action_method) if "```" in class_edit_get_mechanics_to_action_method else class_edit_get_mechanics_to_action_method
            get_games_scores.latest_methods['get_mechanics_to_action'] = new_mechanics_to_action

            # Extract dictionaries from both methods
            import ast
            
            def extract_dict_from_method(method_str):
                tree = ast.parse(method_str)
                for node in ast.walk(tree):
                    if isinstance(node, ast.Return) and isinstance(node.value, ast.Dict):
                        return ast.literal_eval(ast.unparse(node.value))
                return {}

            previous_dict = extract_dict_from_method(previous_mechanics_to_action)
            new_dict = extract_dict_from_method(new_mechanics_to_action)

            # Find new actions
            new_actions = [action for name, action in new_dict.items() if name not in previous_dict]

            #print("New actions added:", new_actions)

        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['get_mechanics_to_action'])
        #print("mechanics_test.latest_methods['mechanic']:\n", mechanics_test.latest_methods['mechanic'])
        updated_mechanics = get_games_scores.latest_methods['mechanic']
        updated_mechanics = add_indented_strings(updated_mechanics, mechanics[0])
        get_games_scores.latest_methods['mechanic'] = updated_mechanics
        #print("Updated mechanics:\n", mechanics_test.latest_methods['mechanic'])
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['mechanic'], no_indent=True)
        #print("Updated game mechanic class:\n", updated_game_mech_class)
        updated_other_methods = get_games_scores.latest_methods['other']#class_add_method
        
        if class_add_method != 'False' and not "__init__" in class_add_method:
            class_add_method = extract_code(class_add_method) if "```" in class_add_method else class_add_method
            updated_other_methods = add_indented_strings(updated_other_methods, class_add_method, no_indent=True)
        get_games_scores.latest_methods['other'] = updated_other_methods
        
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['other'])
        #print("Updated game mechanic class:\n", updated_game_mech_class)
        func_make_game, str_map_func_changed, important_tiles_func_changed = func_make_game_gen()
        get_games_scores.latest_methods["str_world"]  = str_map_func_changed
        get_games_scores.latest_methods["tiles"]  = important_tiles_func_changed

        class_add_is_terminal_method, class_add_is_terminal_method_prompt, game_name, win_condition = class_add_is_terminal_method_gen()

        get_games_scores.latest_methods['is_terminal'] = extract_code(class_add_is_terminal_method) if "```" in class_add_is_terminal_method else class_add_is_terminal_method
        updated_game_mech_class = add_indented_strings(updated_game_mech_class, get_games_scores.latest_methods['is_terminal'])
        if not displacement:
            #print(f"Mechanics counter before: {get_games_scores.latest_methods['mechanic_counter']}")
            get_games_scores.latest_methods['mechanic_counter'] = count_functions(get_games_scores.latest_methods['mechanic'])
            #print(f"Mechanics counter after: {get_games_scores.latest_methods['mechanic_counter']}")
    
    except Exception as e:
        print(f"Error in updating games: {e}")
        return None, None, None, None, None, None, None, None, None, [False]

    error_break = False
    need_test_fix = False
    mechanics_error = None
    idx = 0
    while not error_break:
        
        """if not need_test_fix:
            # TODO: Add manuel simulation test
            unit_test, game_mech_unit_test_prompt = new_mech_unit_test(unit_test_info, mechanic) 
        else:
            unit_test = fixing_test(tests_error)
            idx += 0.11
        if "```python" in unit_test or "```" in unit_test:
            unit_test_func = extract_code(unit_test)
        else:
            unit_test_func = unit_test"""
        try:	
            namespace = {}
            start_time = time.time()
            #exec(total_unit_test_code, namespace)
            #result = namespace.get('result', None)
            mechanics_error_break = True
            error_break = True
            game_name = game_name.replace(" ", "_")
            unique_id = uuid.uuid4().hex[:8] 
            total_mcts_code = mcts_imports  + "\n\n" + updated_game_mech_class + "\n\n" + func_make_game + "\n\n" + agent_body(new_actions, f"cache/{EXPERIMENT}/{generation}_{game_name}_{unique_id}.gif", mcts_player=True, llm_player=False, game_name=game_name, win_condition=win_condition)
            #print("\n\nRunning MCTS...")
            start_time_code = time.time()
            #_rcf = run_code.remote(total_mcts_code, idx, f"cache/{EXPERIMENT}/{generation}_{game_name}_{unique_id}")
            scores_mcts, error_in_mcts = run_code(total_mcts_code, idx, f"cache/{EXPERIMENT}/{generation}_{game_name}_{unique_id}")
            #scores_mcts, error_in_mcts = ray.get(_rcf)
            time_taken_code = time.time() - start_time_code
            print(f"Time taken to run run_code in create_games.py for {game_name}: {time_taken_code:.4f} seconds")
            #print(f"Fitness, behaviour 1 and 2 for MCTS is: {scores_mcts}")
            fitness_mcts, behaviour_1, _, mechanic_fitness, best_action_sequence, str_world, tile_mapping, mechanics_to_action, dones = scores_mcts[0], scores_mcts[1], scores_mcts[2], scores_mcts[3], scores_mcts[4], scores_mcts[5], scores_mcts[6], scores_mcts[7], scores_mcts[8]
            #behaviour_2 = code_behavior_score(get_games_scores.latest_methods['is_terminal'])
            #game_data = {
            #    'str_world': str_world,  # Map layout
            #    'tile_mapping': tile_mapping,  # Tile definitions
            #    'mechanics_to_action': mechanics_to_action,  # Action mappings
            #    #'env_image': env_image,  # Image/tile paths
            #    'win_condition': win_condition,  # Game win condition
            #    'game_name': game_name,  # Name of the game
            #    'game_class': updated_game_mech_class,  # Full game environment class
            #    'make_game': func_make_game,  # Function to create the game environment
            #    'mechanics_to_action': mechanics_to_action  # Current game mechanics
            #}
            #print(f"Behaviour 2 is: {behaviour_2}")
            mechanics_error_break = True
            error_break = True
            if fitness_mcts == None:
                print(f"Error in MCTS: {error_in_mcts}")
                objectives.append(0)
                mechanics_error_break = True
                error_break = True
                continue
            else:
                objectives.append(fitness_mcts)
            mechanics_error_break = True
            error_break = True
        except Exception as e:
            #TODO: fix the mechanic here! 
            print(f"Error in unit test: {e}")
            #print("Detailed exception information:")
            #traceback.print_exc() 
            mechanics_error = f"You created the following game mechanic:\n {mechanics[0]}\n This game mechanics fit in the following game logic class:\n {updated_game_mech_class}\n It gave the following error: {e}. Fix the mechanics and only return the mechanics function and nothing else."
            error_break = True
            
            
            del get_games_scores.latest_methods
            return None, None, None, None, None, None, None, None, None, [False]
            #objectives.append(0)
            #objectives, measures, updated_game_mech_class = mechanics_test(mechanic)
    
    #behaviour_descs = np.array([total_time * 100, semantic_code_analysis(extract_code(mechanic) if "```python" in mechanic else mechanic)], dtype=np.float32)
    
    #behaviour_descs = np.array([code_behavior_score(extract_code(mechanic) if "```python" in mechanic else mechanic), semantic_code_analysis(extract_code(mechanic) if "```python" in mechanic else mechanic)], dtype=np.float32)
    #behaviour_descs = np.array([behaviour_1, behaviour_2], dtype=np.float32)
    if len(objectives) == 0 and objectives is not None:
        objectives.append(0)

    
    #print("The mechanic to be measured for behaviours:\n", mechanic)
    measures.append(0)
        # Print stats for the current mechanic
    print(f"\n--- Stats for game: {game_name} ---")
    print(f"Objective Score: {objectives[-1]:.2f}")
    #print(f"Behavior Measures:")
    #print(f"  - Action score: {behaviour_descs[0]:.2f}")
    #print(f"  - Code behaviour score of is_terminal: {behaviour_descs[1]:.2f}")
    print(f"  - Fitness of the latest mechanic: {mechanic_fitness:.2f}")
    print("--------------------------------------------")
    #print(f"Measures: {measures}")
    #print(f"Objectives: {objectives}")
    #print(f"\n\nMechanic: {extract_function_name(mechanic)}\n\nObjective: {objectives[-1]}, Measure: {measures[-1]}\n")

    game_individual = get_games_scores.latest_methods
    del get_games_scores.latest_methods
    return objectives, measures, game_individual, mechanic_fitness, win_condition, game_name, best_action_sequence, mechanics_to_action, " ", dones