import types

import re
import ast
import json
import random
import os
import time

def add_method_to_class(cls, method_name, method_string):
    code = compile(method_string, "<string>", "exec")
    function = types.FunctionType(code.co_consts[0], globals())
    setattr(cls, method_name, function)

def is_method_added(cls, method_name):
    return hasattr(cls, method_name)

def parse_grid(input_str):
    grid = [list(line) for line in input_str.strip().split('\n')]
    return grid

def extract_function_name(code):
    match = re.search(r'def\s+(\w+)\s*\(', code)
    if match:
        return match.group(1)
    return None

def add_class_variables(cls, vars_dict):
    for var_name, value in vars_dict.items():
        #print(f"Added {var_name} = {value}")
        setattr(cls, var_name, value)

def is_added(cls, method_name):
    return hasattr(cls, method_name)

def extract_list(string_data):
    try:
        # Search for the list pattern in the string, including those within ```json and ```python blocks
        pattern = r'```(?:json|python)?\s*(\[.*?\])\s*```|(\[.*?\])'
        matches = re.findall(pattern, string_data, re.DOTALL)
        if matches:
            # Flatten the matches and filter out empty strings
            matches = [match[0] or match[1] for match in matches if match[0] or match[1]]
            # Iterate over all matches to find the first valid list
            for match in matches:
                try:
                    # Try to parse as Python list
                    extracted_list = ast.literal_eval(match)
                    if isinstance(extracted_list, list):
                        return extracted_list
                except (ValueError, SyntaxError):
                    try:
                        # Try to parse as JSON list
                        extracted_list = json.loads(match)
                        if isinstance(extracted_list, list):
                            return extracted_list
                    except json.JSONDecodeError:
                        continue
            #print("No valid list found in the matches.")
            return []
        else:
            #print("No list-like pattern found in the string.")
            return []
    except ValueError as e:
        #print(f"Error converting string to list: {e}")
        return []
    except SyntaxError as e:
        #print(f"Syntax error in the string: {e}")
        return []

def extract_dict(string_data):
    try:
        # Search for the dictionary pattern in the string, including those within ```json and ```python blocks
        pattern = r'```(?:json|python)?\s*(\{.*?\})\s*```|(\{.*?\})'
        matches = re.findall(pattern, string_data, re.DOTALL)
        if matches:
            # Flatten the matches and filter out empty strings
            matches = [match[0] or match[1] for match in matches if match[0] or match[1]]
            # Iterate over all matches to find the first valid dictionary
            for match in matches:
                try:
                    # Try to parse as Python dictionary
                    mission_dict = ast.literal_eval(match)
                    if isinstance(mission_dict, dict):
                        # Add single quotes to string values
                        for key, value in mission_dict.items():
                            if isinstance(value, str):
                                mission_dict[key] = f"{str(value)}"
                        return mission_dict
                except (ValueError, SyntaxError):
                    try:
                        # Try to parse as JSON dictionary
                        mission_dict = json.loads(match)
                        if isinstance(mission_dict, dict):
                            # Add single quotes to string values
                            for key, value in mission_dict.items():
                                if isinstance(value, str):
                                    mission_dict[key] = f"{str(value)}"
                            return mission_dict
                    except json.JSONDecodeError:
                        continue
            #print("No valid dictionary found in the matches.")
            return {}
        else:
            #print("No dictionary-like pattern found in the string.")
            return {}
    except ValueError as e:
        #print(f"Error converting string to dictionary: {e}")
        return {}
    except SyntaxError as e:
        #print(f"Syntax error in the string: {e}")
        return {}
    

def add_indented_strings(string1, string2, no_indent=False, double_indent=False):
    # Split the second string into lines
    lines = string2.split('\n')
    #print("lines: ", lines)
    if no_indent:
        # For no_indent, simply join the lines without adding spaces
        indented_string2 = '\n'.join(lines)
        # Combine strings without an extra newline
        result = string1 + '\n' + indented_string2
    else:
        # Apply indentation for other cases
        if double_indent:
            indented_lines = ['        ' + line for line in lines]
        else:
            indented_lines = ['    ' + line for line in lines]
        
        # Join the indented lines back into a single string
        indented_string2 = '\n'.join(indented_lines)
        
        # Combine the first string with the indented second string
        result = string1 + '\n' + indented_string2
    
    return result

import re

def extract_code(text):
    # Try to match code enclosed in triple backticks
    code_block = re.search(r'```(?:python)?\s*(.*?)\s*```', text, re.DOTALL)
    if code_block:
        return code_block.group(1).strip()
    
    # If no triple backticks, try to match a multi-line string
    code_block = re.search(r'"""(.*?)"""', text, re.DOTALL)
    if code_block:
        return code_block.group(1).strip()
    
    code_block = re.search(r'```(.*?)```', text, re.DOTALL)
    if code_block:
        return code_block.group(1).strip()
    
    
    return None

def remove_blank_lines(function_code: str) -> str:
    lines = [line for line in function_code.splitlines() if line.strip()]
    return '\n'.join(lines)

def filter_functions_by_name(functions_string, function_names):
    function_blocks = functions_string.split('\n\n')

    filtered_functions = []
    for block in function_blocks:
        if not block.strip():
            continue

        for func_name in function_names:
            if f"def {func_name}" in block:
                filtered_functions.append(block)
                break
    
    return '\n\n'.join(filtered_functions)

import sys
import subprocess
import ast
import ray

def fix_unquoted_strings(file_content):
    pattern = r'(=\s*)([A-Z])\b(?![\'"])'
    fixed_content = re.sub(pattern, r"\1'\2'", file_content)
    return fixed_content

#@ray.remote
def run_code(code, idx, mechanic, save_file=True, is_mechanic=False):
    import multiprocessing
    import os

    modified_code = code + "\n\nprint('RETURN_VALUE:', repr(__return_value__))"

    #unique_id = uuid.uuid4().hex[:8] 

    #if is_mechanic:
    #    temp_file = f'code_mechanic_{mechanic}_{idx}_{unique_id}.py'
    #else:
    #    temp_file = f'code_game_{mechanic}_{idx}_{unique_id}.py'

    temp_file = f"{mechanic}.py"

    with open(temp_file, 'w') as f:
        f.write(modified_code)

    with open(temp_file, 'r') as file:
        modified_code_content = file.read()

    # Record start time
    start_time = time.time()
    
    # Fix unquoted strings
    fixed_code_content = fix_unquoted_strings(modified_code_content)
    
    # Calculate time taken
    time_taken = time.time() - start_time
    #print(f"Time taken to fix unquoted strings: {time_taken:.4f} seconds")
    fixed_code_content = fix_unquoted_strings(modified_code_content)
    
    with open(temp_file, 'w') as f:
        f.write(fixed_code_content)
    
    # Get the number of available CPU cores
    num_cores = multiprocessing.cpu_count()
    
    # Set environment variables for parallel processing
    env = os.environ.copy()
    env['OMP_NUM_THREADS'] = str(num_cores)  # For OpenMP
    env['MKL_NUM_THREADS'] = str(num_cores)  # For MKL
    env['OPENBLAS_NUM_THREADS'] = str(num_cores)  # For OpenBLAS
    env['VECLIB_MAXIMUM_THREADS'] = str(num_cores)  # For VecLib
    env['NUMEXPR_NUM_THREADS'] = str(num_cores)  # For NumExpr
    env['BLIS_NUM_THREADS'] = str(num_cores)  # For BLIS
    
    # Run the code with parallel processing enabled
    result = subprocess.run([sys.executable, temp_file], 
                          capture_output=True, 
                          text=True,
                          env=env)
    
    # Uncomment to print the output/errors for debugging
    output_preview = "\n".join(result.stdout.split('\n')[:10])
    #print(f"Output preview from subprocess:\n{output_preview}\n...")
    
    # Always print any errors
    if result.stderr:
        #print(f"Errors from subprocess:\n{result.stderr}")
        pass

    return_value = None
    for line in result.stdout.split('\n'):
        if line.startswith('RETURN_VALUE:'):
            try:
                return_value = ast.literal_eval(line.split(':', 1)[1].strip())
                #print(f"Successfully extracted return value")
            except Exception as e:
                print(f"Error parsing return value: {e}")
                #import traceback
                #traceback.print_exc()   
                

    if not save_file:
        os.remove(temp_file)
    
    return return_value, result.stderr


def delete_function(class_string, function_name):
    # Pattern to match the function and its content
    pattern = rf'def\s+{re.escape(function_name)}\s*\([^)]*\):(?:\n(?:(?!def\s).)*)*'
    
    # Remove the function
    modified_string = re.sub(pattern, '', class_string, flags=re.DOTALL)
    
    # Remove any extra newlines that might have been left
    modified_string = re.sub(r'\n{3,}', '\n\n', modified_string)
    
    return modified_string.strip()

def delete_variable(code_string, variable_name):
    # Pattern to match the variable assignment
    pattern = rf'\b{re.escape(variable_name)}\s*=\s*[^;\n]+'
    
    # Remove the variable assignment
    modified_string = re.sub(pattern, '', code_string)
    
    # Remove any extra newlines that might have been left
    modified_string = re.sub(r'\n{3,}', '\n\n', modified_string)
    
    return modified_string.strip()

def delete_self_reset(code_string, variable_name):
    # Pattern to match variable assignments or method calls, including indented ones
    if variable_name == "self.reset()":
        pattern = rf'^\s*{re.escape(variable_name)}.*$'
    else:
        pattern = rf'^\s*self\.{re.escape(variable_name)}\s*=.*$'
    
    # Split the code into lines, remove matching lines, and rejoin
    lines = code_string.split('\n')
    modified_lines = [line for line in lines if not re.match(pattern, line)]
    modified_string = '\n'.join(modified_lines)
    
    # Remove excessive newlines
    modified_string = re.sub(r'\n{3,}', '\n\n', modified_string)
    
    return modified_string.strip()

def randomly_delete_function(code_string):
    pattern = r'(def\s+\w+\s*\([^)]*\):(?:\s+[^\n]+\n?(?:\s+[^\n]*\n?)*?)(?=\s*def|\s*$))'
    functions = re.findall(pattern, code_string, re.MULTILINE)
    
    if not functions:
        return code_string
    
    function_to_remove = random.choice(functions)
    #print(f"Mechanic to replace:\n{function_to_remove}")
    new_code = code_string.replace(function_to_remove, '')
    new_code = re.sub(r'\n{3,}', '\n\n', new_code)
    #print(f"New code:\n{new_code}")
    return new_code.strip()

def count_functions(code_string):

    pattern = r'def\s+[a-zA-Z_][a-zA-Z0-9_]*\s*\([^)]*\)\s*:'
    matches = re.findall(pattern, code_string)
    
    return len(matches)




