from typing import Dict, Any
import os
import time
import uuid

import numpy as np
from transformers import DistilBertTokenizer, DistilBertModel
import torch
from scipy.spatial.distance import cosine
from scipy.stats import entropy

from dotenv import load_dotenv
load_dotenv()
import traceback

from llm_proxy_client import LLMProxyClient
from code_behaviour import code_behavior_score
from utils import extract_function_name, extract_dict, extract_code, add_indented_strings, delete_self_reset, delete_variable, run_code

import ray
from configs import Configs
configs = Configs()
MODEL = configs.model
EXPERIMENT = configs.experiment

# Initialize LLM client
llm_client = LLMProxyClient()

unit_test_info_part_1 = '''
import gym
from gym import spaces
import numpy as np
import os
import json
import time
import imageio
from PIL import Image, ImageDraw, ImageFont
from rembg import remove
import random
import sys
root_dir = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..'))
sys.path.append(root_dir)
from mech_mcts import PlayMCTS

str_world = """AAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAA
AA&@OAAAAAA&AAOAAA
AAAAAAAAAAAAAAAAAA
AAA#AAAAAAAA#AAAAA
AAAAAAAAAAAAAAAAAA"""
str_map_wo_chars = """AAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAA
AAAAOAAAAAAAAAOAAA
AAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAA"""
'''

unit_test_info_part_2 = r'''
walkables = ['A', 'B']
interactive_object_tiles = ['O']
enemy_tiles = ["#"]
npc_tiles = ["&"]
env_image = dict()

folder_path = r"/gmd/world_tileset_data"


env_image["A"] = Image.open(r"/world_tileset_data/td_world_floor_grass_c.png").convert("RGBA")
env_image["B"] = Image.open(r"/world_tileset_data/td_world_wall_stone_h_a.png").convert("RGBA")
env_image["O"] = Image.open(r"/world_tileset_data/td_world_chest.png").convert("RGBA")
env_image["@"] = Image.open(r"/character_sprite_data/td_monsters_archer_d1.png").convert("RGBA")
env_image["#"] = Image.open(r"/character_sprite_data/td_monsters_witch_d1.png").convert("RGBA")
env_image["&"] = Image.open(r"/character_sprite_data/td_monsters_goblin_captain_d1.png").convert("RGBA")

env = GameMechEnv(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)'''


mcts_imports = r'''
import os
import sys
parent_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
sys.path.append(parent_dir)
from mech_mcts import MCTS
import numpy as np
import json
import time
import imageio
from PIL import Image, ImageDraw, ImageFont
from rembg import remove
import random

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
import ray
import gc'''

mcts_body = r'''

def make(name):


    str_world = """BBBBBBBBBBB
BAAAAAAAAAB
BAAAOAAAAAB
BA#@OAAAAAB
BA#AAAAAAAB
BBBBBBBBBBB"""
    str_map_wo_chars = """BBBBBBBBBBB
BAAAAAAAAAB
BAAOOAAAAAB
BAAAOAAAAAB
BAAAAAAAAAB
BBBBBBBBBBB"""

    walkables = ['A', 'B']
    interactive_object_tiles = ['O']
    enemy_tiles = ["#"]
    npc_tiles = ["&"]
    env_image = dict()

    folder_path = r"/gmd/world_tileset_data"


    env_image["A"] = Image.open(r"/gmd/world_tileset_data/td_world_floor_grass_c.png").convert("RGBA")
    env_image["B"] = Image.open(r"/gmd/world_tileset_data/td_world_wall_stone_h_a.png").convert("RGBA")
    env_image["C"] = Image.open(r"/gmd/world_tileset_data/td_world_floor_grass_c.png").convert("RGBA")
    env_image["O"] = Image.open(r"/gmd/world_tileset_data/td_world_chest.png").convert("RGBA")
    env_image["@"] = Image.open(r"/gmd/character_sprite_data/td_monsters_archer_d1.png").convert("RGBA")
    env_image["#"] = Image.open(r"/gmd/character_sprite_data/td_monsters_witch_d1.png").convert("RGBA")
    env_image["&"] = Image.open(r"/gmd/character_sprite_data/td_monsters_goblin_captain_d1.png").convert("RGBA")
    
    
    env = GameMechEnv(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)
    env.reset = pufferlib.utils.silence_warnings(env.reset)
    env = GymCompatibilityWrapper(env)
    env = pufferlib.postprocess.EpisodeStats(env)
    return env
    
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


def evaluate_new_mechanic(env, focus_mechanic):
    mcts = MCTS(env.clone(), focus_mechanic, iterations=500, simulation_iterations=75, exploration_weight=1.81, max_mechanics=2)
    stats = mcts.run()

    #print("\nFinal MCTS Tree:")
    #mcts.pretty_print_tree()

    all_nodes = mcts.get_all_nodes()
    
    # Group nodes by the length of their mechanics
    nodes_by_length = {}
    for node in all_nodes:
        if node != mcts.root:
            length = len(node.mechanics)
            if length not in nodes_by_length:
                nodes_by_length[length] = []
            nodes_by_length[length].append(node)

    # Function to get top n nodes
    def get_top_n(nodes, key_func, n=1):
        return sorted(nodes, key=key_func, reverse=True)[:n]

    # Print top performers for each length
    #for length, nodes in sorted(nodes_by_length.items()):
    #    print(f"\nTop performers for {length} mechanics:")

    #    print("\nTop by visits:")
    #    for node in get_top_n(nodes, lambda x: x.visits):
    #        print(f"{node.mechanics}: visits={node.visits}, value={node.value:.2f}, reward={node.reward:.2f}")

    #    print("\nTop by value:")
    #    for node in get_top_n(nodes, lambda x: x.value):
    #        print(f"{node.mechanics}: visits={node.visits}, value={node.value:.2f}, reward={node.reward:.2f}")

    #    print("\nTop by reward:")
    #    for node in get_top_n(nodes, lambda x: x.reward):
    #        print(f"{node.mechanics}: visits={node.visits}, value={node.value:.2f}, reward={node.reward:.2f}")

    def get_depth(node):
        depth = 0
        while node.parent:
            depth += 1
            node = node.parent
        return depth

    total_depth = 0
    # Calculate mechanic combinations and pairing counts
    mechanic_combinations = {}
    for node in all_nodes:
        if node != mcts.root:
            depth = get_depth(node)
            depth_factor = 1 + (depth / 10)
            combination = tuple(node.mechanics)
            mechanic_combinations[combination] = mechanic_combinations.get(combination, 0) + (node.visits * depth_factor)
            total_depth += 1
    
    print(f"Total depth of MCTS: {total_depth}")


    total_value = sum(mechanic_combinations.values())
    mechanic_combinations = {k: v / total_value for k, v in mechanic_combinations.items()}
    
    pairing_counts = {f"{i+1}_mechanic": 0 for i in range(len(env.mechanic_to_action))}
    
    for combination, usage in mechanic_combinations.items():
        if usage > 0: 
            pairing_counts[f"{len(combination)}_mechanic"] += 1

    # Calculate main tree stats
    total_nodes = len(all_nodes)
    #avg_depth = sum(node.value for node in all_nodes) / total_nodes if total_nodes > 0 else 0
    #max_depth = max(node.value for node in all_nodes)
    #avg_visits = sum(node.visits for node in all_nodes) / total_nodes if total_nodes > 0 else 0
    #max_visits = max(node.visits for node in all_nodes)

    #print("\nMain Tree Stats:")
    #print(f"Total nodes of MCTS: {total_nodes}")
    #print(f"Average depth: {avg_depth:.2f}")
    #print(f"Max depth: {max_depth}")
    #print(f"Average visits: {avg_visits:.2f}")
    #print(f"Max visits: {max_visits}")

    #print(f"Best child: {mcts.best_child(mcts.root)}")

    return pairing_counts, mechanic_combinations, all_nodes


def fitness_function(mechanic_combination, usage_stats, pairing_counts):
    # Convert mechanic_combination to a tuple for dictionary lookup
    combo = tuple(sorted(mechanic_combination))

    usage = usage_stats.get(combo, 0)

    pairing_diversity = pairing_counts[f"{len(combo)}_mechanic"] / sum(pairing_counts.values())
    #print(f"pairing diversity: {pairing_diversity}")
    # 3. Combination size bonus
    size_bonus = len(combo)# / len(usage_stats)  # Normalize by total number of combinations
    #print(f"size bonus: {size_bonus}")
    # 4. Synergy bonus (assuming combinations with higher usage have better synergy)
    synergy_bonus = usage * len(combo)
    #print(f"synergy bonus: {synergy_bonus}")
    # 5. Novelty bonus (reward less common combinations)
    novelty_bonus = 1 - usage  # Invert usage to favor less common combinations
    #print(f"novelty bonus: {novelty_bonus}")
    # Weighted sum of factors
    if usage > 0:
        fitness = (
            synergy_bonus #+  # Primary factor
            #0.25 * pairing_diversity +
            #0.25 * size_bonus +
            #0.25 * synergy_bonus +
            #0.05 * novelty_bonus
        )
    else:
        fitness = 0
    return fitness

#for node in all_nodes:
#    print(f"fitness for {node.mechanics}: {fitness_function(node.mechanics, usage_stats, pairing_counts)}")

def calculate_interaction_score(fitnesses, focus_mechanic):
    # Extract all fitness values for combinations including the focus mechanic
    mechanic_fitnesses = [fitness for mechanics, fitness in fitnesses.items() if focus_mechanic in mechanics]
    
    if not mechanic_fitnesses:
        return 0  # Return 0 if no combinations include the focus mechanic
    
    # Calculate metrics
    avg_fitness = np.mean(mechanic_fitnesses)
    max_fitness = np.max(mechanic_fitnesses)
    std_fitness = np.std(mechanic_fitnesses)
    #print(f"avg_fitness: {avg_fitness}")
    #print(f"max_fitness: {max_fitness}")
    #print(f"std_fitness: {std_fitness}")
    # Count high-performing combinations (e.g., above 75th percentile)
    threshold = np.percentile(mechanic_fitnesses, 99)
    #print(f"threshold: {threshold}")
    #for fitness in mechanic_fitnesses:
        #if fitness >= threshold:
        #    print(f"fitness: {fitness} Above threshold")
        #else:
        #    print(f"fitness: {fitness} Below threshold")
    high_performing_count = sum(1 for fitness in mechanic_fitnesses if fitness >= threshold)
    #print(f"high_performing_count: {high_performing_count}")
    # Normalize the high-performing count
    normalized_high_performing = high_performing_count / len(mechanic_fitnesses)
    #print(f"normalized_high_performing: {normalized_high_performing}")
    # Combine metrics into a single score
    # You can adjust these weights based on what you consider most important
    score = (
        #0.3 * avg_fitness +
        max_fitness * high_performing_count
        #0.1 * (1 / (std_fitness + 1e-5)) +  # Lower standard deviation is better
        #0.1 * normalized_high_performing
    )
    
    return score'''

def mcts_calls(mechanic_name): 
    return rf'''env = make('MechEnv-v1')
env.reset()  

focus_mechanic = "{mechanic_name}" 
pairing_counts, usage_stats, all_nodes = evaluate_new_mechanic(env, focus_mechanic)


fitnesses = {{}}
for node in all_nodes:
    fitnesses[str(node.mechanics)] = fitness_function(node.mechanics, usage_stats, pairing_counts)'''


tokenizer = DistilBertTokenizer.from_pretrained('distilbert-base-uncased')
model = DistilBertModel.from_pretrained('distilbert-base-uncased')

def get_game_mechanic_types():
    return [
        "Movement",
        "Interaction",
        "Combat",
        "Progression",
        "Environment",
        "Puzzle",
        "Resource Management",
        #"Social",
        "Exploration",
        "Time Manipulation"
    ]

def get_embedding(text):
    inputs = tokenizer(text, return_tensors="pt", padding=True, truncation=True, max_length=512)
    with torch.no_grad():
        outputs = model(**inputs)
    return outputs.last_hidden_state.mean(dim=1).squeeze().numpy()

def analyze_mechanic(mechanic):
    mechanic_lower = mechanic.lower()
    mechanic_embedding = get_embedding(mechanic_lower)
    behavior_vector = np.zeros(len(get_game_mechanic_types()))

    mechanic_types = get_game_mechanic_types()
    word_lists = [
        ["move", "walk", "run", "jump", "fly", "teleport", "dash", "swim", "climb", "crouch", "sprint"],
        ["pick", "use", "interact", "open", "close", "talk", "trade", "craft", "activate", "push", "pull"],
        ["attack", "fight", "hit", "shoot", "defend", "block", "dodge", "cast", "spell", "heal", "damage"],
        ["level", "upgrade", "unlock", "improve", "evolve", "progress", "achieve", "complete", "quest", "mission"],
        ["weather", "day", "night", "season", "climate", "destroy", "build", "terraform", "grow", "plant"],
        ["solve", "puzzle", "riddle", "match", "connect", "arrange", "decode", "decipher", "logic", "pattern"],
        ["collect", "gather", "manage", "inventory", "store", "spend", "earn", "balance", "allocate", "distribute"],
        #["ally", "team", "cooperate", "compete", "influence", "negotiate", "persuade", "reputation", "relationship", "communicate"],
        ["explore", "discover", "map", "reveal", "uncover", "navigate", "search", "investigate", "scout", "survey"],
        ["time", "slow", "fast", "rewind", "forward", "pause", "resume", "loop", "cycle", "sequence"]
    ]

    for i, word_list in enumerate(word_lists):
        category_embedding = get_embedding(" ".join(word_list))
        similarity = 1 - cosine(mechanic_embedding, category_embedding)
        behavior_vector[i] = similarity

    #print(f"Behavior_vector: {behavior_vector}\n")
    # Normalize the behavior vector
    most_similar_index = np.argmax(behavior_vector)
    most_similar_category = mechanic_types[most_similar_index]

    
    behavior_vector = behavior_vector / np.sum(behavior_vector)
    #print(f"Normalized behavior_vector: {behavior_vector}\n")
    # Find the most similar category
    #print(f"Most similar category: {most_similar_index}\n")
    behaviour_score = behavior_vector[most_similar_index] * (most_similar_index + 1)
    #print(f"Behaviour_score {behaviour_score}\n")
    # Calculate a single score based on the entropy of the behavior vector
    # Lower entropy means the mechanic is more specialized (higher score)
    # Higher entropy means the mechanic is more generalized (lower score)
    #max_entropy = np.log(len(behavior_vector))  # Maximum possible entropy
    #vector_entropy = entropy(behavior_vector)
    #print(f"\nvector_entropy {vector_entropy}\n")
    #specialization_score = 1 - (vector_entropy / max_entropy)
    #print(f"\nspecialization_score {specialization_score}\n")
    # Scale the score to be between 0 and 100
    #final_score = specialization_score * 100
    #print(f"final_score {final_score}\n")
    return behavior_vector, most_similar_category, behaviour_score

def calculate_state_behavior(behavior):
    return np.linalg.norm(behavior)

@ray.remote
def mechanics_test(mechanics, previous_mechanics = "", init_mechs = False, generation=None):
    global updated_game_mech_class
    objectives = []
    measures = []

    #print("\nMECHANICS in create_mechanics.py:\n", mechanics)

    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 + "\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
        )
        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:\n" + mechanics + "\n Edit step method, if required, for the given game mechanic 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 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
        )
        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 + "\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
        )
        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 + "\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
        )
        return response['choices'][0]['message']['content'], class_add_method_prompt
    
    def class_add_methods_gen(other_methods):
        #print("Adding methods...")
        class_add_method_prompt = "Given the following init method of the class:\n" + init_method + "\n And all functions already present in the class:\n"+ other_methods + "\n And the following game mechanic you just created:\n" + mechanics + "\n Add any new methods needed for the " + extract_function_name(mechanics) +" to work, if required. Do not create or update __init__, reset, step, get_state, render, get_action_space, or get_mechanics_to_action methods. Do not repeat the game mechanic as a method or the present methods. Do not add any new variables. Only output the additional new method or methods required 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
        )
        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 + "\n Add any new variable to the __init__ method, if required, for the given game mechanic to work. Add to the code provided so that other initializations are not disturbed. You are not allowed to add new actions to the action space. Remember, there is an existing update_player method, no need to add it again. You are also not allowed to change the input arguments of the init method. 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
        )
        return response['choices'][0]['message']['content'], class_add_method_prompt

    def class_add_var_gen():
        #print("Adding variables...")
        class_add_var_prompt = "Given the following class:\n" + game_mech_class + "\n And the following game mechanic:\n" + mechanics + "\n Add any new variables required in the GameMechEnv for the given game mechanic to work. Always add to the existing code. Only output the new variables if they are added in a Python dictionary format and if not added, just output 'False'. The Python dictionary can look like {'var_name_1': init_value, 'var_name_2': init_value}. The keys should be the string of the variable name and the values should be the initial values. The values can never be a new argument to the init method. Do not output anything else."
        response = llm_client.chat_completion(
            model=MODEL,
            messages=[{"role": "user", "content": class_add_var_prompt}],
            temperature=1
        )
        return response['choices'][0]['message']['content'], class_add_var_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 mechanic:\n" + mechanics + "\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
        )
        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 mechanic:\n" + mechanics + "\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 "+ extract_function_name(mechanics) +": 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
        )
        return response['choices'][0]['message']['content'], class_add_method_prompt

    
    game_mech_class = """class GameMechEnv(gym.Env):
        def __init__(self, walkable_tiles,tiles_without_char,  tiles, str_map_without_chars, str_map, interactive_object_tiles, enemy_tiles, render_mode="human"):
            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.tile_size = 16  # or whatever size you want
            self.char_tile_size = 16  # or whatever size you want for characters
            self.frames = []

        
            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.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.picked_objects = []
            self.npc_tiles = ["&"]
            self.enemy_tiles = ["#"]
            self.player_health = 100
            self.enemy_health = 100
            self.current_score = 0
            self.done = False
            self.map = [list(row) for row in self.map_str]
            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
            self.done = False
            if action < 4:  # Movement actions
                self.move_player(action)
            self.done = reward > 0
            info = {}
            return self.get_state()["map"], reward, self.done, 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)
            
            draw = ImageDraw.Draw(env_img)
            font = ImageFont.load_default()
            text = f"Objects Picked: {len(self.picked_objects)}"
            draw.text((10, env_img.size[1] - 20), text, (255, 255, 255), font=font)
            
            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):
    # 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 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 = GameMechEnv(
        walkable_tiles=self.walkable_tiles,
        tiles_without_char=self.tiles_without_char,
        tiles=self.tiles,
        str_map_without_chars='\n'.join(self.map_str_without_chars),
        str_map='\n'.join(self.map_str),
        interactive_object_tiles=self.interactive_object_tiles,
        enemy_tiles=self.enemy_tiles
    )
    new_env.map = [row[:] for row in self.map]
    new_env.map_without_chars = [row[:] for row in self.map_without_chars]
    new_env.player_position = self.player_position
    new_env.current_tile = self.current_tile
    new_env.char_to_int = self.char_to_int
    new_env.char_set = self.char_set
    return new_env

def is_terminal(self):
    return self.done"""

    get_state_method = """def get_state(self):
    return {"map": self.map}"""

    step_method = """def step(self, action):
    reward = 0
    self.done = False
    if action < 4:  # Movement actions
        self.move_player(action)
    self.done = reward > 0
    info = {}
    return self.get_state()["map"], reward, self.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 3"""

    get_mechanics_to_action_method = """def get_mechanics_to_action(self):
    return {
        "move_player": 0,  # 0-3 for movement
    }"""

    move_player = """def move_player(self, action):
    moves = {0: (-1, 0), 1: (1, 0), 2: (0, -1), 3: (0, 1)}  # Up, Down, Left, Right
    dx, dy = moves[action]
    new_row = self.player_position[0] + dx
    new_col = self.player_position[1] + dy
    reward = 0
    if 0 <= new_row < len(self.map) and 0 <= new_col < len(self.map[0]):
        new_tile = self.map[new_row][new_col]
        if new_tile in self.walkable_tiles:
            self.update_player_position(new_row, new_col, new_tile)
    return reward"""


    init_method = r"""def __init__(self, walkable_tiles,tiles_without_char,  tiles, str_map_without_chars, str_map, interactive_object_tiles, enemy_tiles, render_mode="human"):
    super(GameMechEnv, 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()
    self.done = False
    self.tile_size = 16
    self.char_tile_size = 16
    self.frames = []
    max_width = max(len(row) for row in self.map_str)
    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 = "rgb_array"
    self.walkable_tiles = walkable_tiles
    self.interactive_object_tiles = interactive_object_tiles
    self.enemy_tiles = enemy_tiles
    self.picked_objects = []
    self.npc_tiles = ["&"]
    self.enemy_tiles = ["#"]
    self.player_health = 100
    self.enemy_health = 100
    self.current_score = 0
    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.reset()"""
    
    #def simulation_test(name):
    #    return f"""def simulate_mechanic():
    #        env.reset()
    #        reward = env.{name}()
    #        return reward"""

    def simulation_test_mcts(output_path):
        return 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")



def run_mcts(env):
    env.reset()
    mcts = PlayMCTS(env, iterations=50000, early_stop=True)
    scores = mcts.run()
    return scores, mcts

def simulate_mechanic():
    try:
        # Run MCTS (single execution without Ray)
        scores, mcts = run_mcts(env)
        
        # Extract results directly since we have a single result
        best_action, max_depth, best_action_sequence, action_score, best_rewards, dones = scores.values()
        
        #print("------TEST WITH MCTS-------")
        #print("Best action: ", best_action)
        #print("Max depth reached: ", max_depth)
        #print("Best action sequence: ", best_action_sequence)
        #print("Action score: ", action_score)
        #print("Best rewards: ", best_rewards)
        #print("Dones: ", dones)
        #print()
        
        visualize_action_sequence(best_action_sequence, env, "{output_path}")
        return sum(best_rewards)
    finally:
        if 'mcts' in locals():
            mcts.cleanup()"""

    #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
        try:
            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 not hasattr(mechanics_test, 'latest_methods'):
            #    mechanics_test.latest_methods = {
            latest_methods = {
                    'init': init_method,
                    'reset': reset_method,
                    'get_state': get_state_method,
                    'step': step_method,
                    'render': render_method,
                    'mechanic': move_player,
                    'other': other_methods,
                    'get_action_space': get_action_space_method,
                    'get_mechanics_to_action': get_mechanics_to_action_method
                }
            if not init_mechs:
                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(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(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(latest_methods["get_mechanics_to_action"])
            
            
            #print("Mechanic:\n", mechanic)
            updated_game_mech_class = "class GameMechEnv(gym.Env):\n"
            
            # Update init method if new variables are added
            if class_add_var != 'False' and not init_mechs:
                vars_dict = extract_dict(class_add_var)
                updated_init_method = 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()")
                #mechanics_test.latest_methods['init'] = updated_init_method
                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, latest_methods['init'])
            updated_game_mech_class = add_indented_strings(updated_game_mech_class, latest_methods['reset'])
            # Update get_state method if needed
            if class_edit_get_state_method != 'False' and not init_mechs:
                #mechanics_test.latest_methods['get_state'] = extract_code(class_edit_get_state_method) if "```" in class_edit_get_state_method else class_edit_get_state_method
                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, latest_methods['get_state'])
            # Update step method if needed
            if class_edit_step_method != 'False' and not init_mechs:
                #mechanics_test.latest_methods['step'] = extract_code(class_edit_step_method) if "```" in class_edit_step_method else class_edit_step_method
                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, latest_methods['step'])
            # Update render method if needed
            if class_edit_render_method != 'False' and not init_mechs:
                #mechanics_test.latest_methods['render'] = extract_code(class_edit_render_method) if "```" in class_edit_render_method else class_edit_render_method
                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, latest_methods['render'])
            if class_edit_get_action_space_method != 'False' and not init_mechs:
                #mechanics_test.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
                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, latest_methods['get_action_space'])
            if class_edit_get_mechanics_to_action_method != 'False' and not init_mechs:
                #print(f"Previous get_mechanics_to_action:\ {mechanics_test.latest_methods['get_mechanics_to_action']}")
                #mechanics_test.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
                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, latest_methods['get_mechanics_to_action'])
            
            #print("mechanics_test.latest_methods['mechanic']:\n", mechanics_test.latest_methods['mechanic'])
            updated_mechanics = latest_methods['mechanic']
            #print("\nmechanics:\n", mechanics)
            #print("\nmechanics[0]:\n", mechanics[0])
            updated_mechanics = add_indented_strings(updated_mechanics, mechanics, no_indent=True)
            #mechanics_test.latest_methods['mechanic'] = updated_mechanics
            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, latest_methods['mechanic'])
            #print("Updated game mechanic class:\n", updated_game_mech_class)
            updated_other_methods = latest_methods['other']#class_add_method
            
            if class_add_method != 'False' and not init_mechs 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)
            #mechanics_test.latest_methods['other'] = updated_other_methods
            latest_methods['other'] = updated_other_methods
            
            updated_game_mech_class = add_indented_strings(updated_game_mech_class, latest_methods['other'])
            #print("Updated game mechanic class:\n", updated_game_mech_class)

            unit_test_info = unit_test_info_part_1 + "\n\n" + updated_game_mech_class + "\n\n" + unit_test_info_part_2
        except Exception as e:
            #print(f"Error in generating methods for mechanic: {e}")
            #traceback.print_exc()
            return None, None, None
        #error_in_test = True
        error_break = False
        need_test_fix = False
        mechanics_error = None
        while not error_break:
            if not init_mechs:
                """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"""
                #print("\n\nUnit test function:\n", unit_test_func)
                #print("\n\nUnit test info:\n", unit_test_info)
                # Check if experiments folder exists, create if not
                
                unique_id = uuid.uuid4().hex[:8] 
                total_unit_test_code = unit_test_info + "\n" + simulation_test_mcts(f"cache/{EXPERIMENT}/{generation}_{extract_function_name(mechanics)}_{unique_id}.gif") + "\n" + f"__return_value__ = simulate_mechanic()"
                #print("\n\nTotal unit test code:\n", total_unit_test_code)
                try:	
                    namespace = {}
                    start_time = time.time()
                    #exec(total_unit_test_code, namespace)
                    #_rcf = run_code.remote(total_unit_test_code, generation, f"cache/{EXPERIMENT}/{generation}_{extract_function_name(mechanics)}_{unique_id}", is_mechanic=True)
                    result, error_in_code = run_code(total_unit_test_code, generation, f"cache/{EXPERIMENT}/{generation}_{extract_function_name(mechanics)}_{unique_id}", is_mechanic=True)
                    #result, error_in_code = ray.get(_rcf)
                    #TODO: If test is forward passed without errors, then there should be some reward.
                    total_time = time.time() - start_time
                    print(f"Time taken to run run_code in create_mechanics.py for {extract_function_name(mechanics)}: {total_time:.4f} seconds")
                    #print("\n\nResult of test: ", result)
                    #TODO: Recursive mechanic testing
                    #result = namespace.get('result', None)
                    
                    mechanics_error_break = True
                    error_break = True
                    if isinstance(result, int) or isinstance(result, float):
                        objectives.append(1)# 1 fitness if it works
        
                    if isinstance(result, tuple) or isinstance(result, list):
                        if result[0] != None:
                            objectives.append(1)# 1 fitness if it works
                            mechanics_error_break = True
                            error_break = True
                        else:# THIS IS HOW WE KNOW THAT THE MECHANIC IS NOT WORKING PROPERLY
                            
                            objectives.append(0)
                            mechanics_error_break = True
                            error_break = True
                    #if result != None:
                    #    #TODO: fix the mechanic here! 
                    #    #objectives, measures, updated_game_mech_class = mechanics_test(mechanic)
                    #    need_test_fix = True
                    #    tests_error = f"The following code has a game logic class 'GameMechEnv':\n{total_unit_test_code}\nAnd the following test for the mechanic {extract_function_name(mechanic)}:\n{unit_test_func}\n The test was supposed to only return the reward but it returned {result}. Fix the test to work properly. Only return the test function and nothing else."
                    #    continue
                    #    objectives.append(0)
                    if result == None:
                        #mechanics_error = f"You created the following game mechanic:\n {mechanic}\n This game mechanics fit in the following game logic class:\n {updated_game_mech_class}\n The test you created for the game mechanic is:\n {simulation_test(extract_function_name(mechanic))}\n It gave the following error: {error_in_code}. Fix the mechanics and only return the mechanics function and nothing else."
                        error_break = True
                        #del mechanics_test.latest_methods
                        return None, None, None
                    mechanics_error_break = True
                    error_break = True
                except Exception as e:
                    #TODO: fix the mechanic here! 
                    #print(f"Error in unit test: {e}")
                    mechanics_error = None
                    error_break = True
                    #del mechanics_test.latest_methods
                    return None, None, None
                    error_break = True
                    objectives.append(0)
                    #objectives, measures, updated_game_mech_class = mechanics_test(mechanic)
                    
            else:
                objectives.append(0)
                mechanics_error_break = True
                error_break = True
    
    #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([code_behavior_score(extract_code(mechanics) if "```python" in mechanics else mechanics), analyze_mechanic(extract_function_name(mechanics).replace('_', ' '))[2]], dtype=np.float32)
    if len(objectives) == 0:
        objectives.append(0)
    
    #print("The mechanic to be measured for behaviours:\n", mechanic)
    measures.append(behaviour_descs)
    # Print stats for the current mechanic
    print(f"\n--- Stats for Mechanic: {extract_function_name(mechanics)} ---")
    print(f"Objective Score: {objectives[-1]:.2f}")
    print(f"Behavior Measures:")
    print(f"  - Code Behavior Score: {behaviour_descs[0]:.2f}")
    print(f"  - Type Behavior Score: {behaviour_descs[1]:.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")
    #del mechanics_test.latest_methods
    return objectives, measures, updated_game_mech_class
