#!/usr/bin/env python3
"""Run EoH evolution for TSP GLS generator programs."""

from __future__ import annotations

import sys
import os
from typing import List, Dict, Any, Tuple
import numpy as np

# Add the project root to the path
_project_root = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..', '..', '..', '..'))
if _project_root not in sys.path:
    sys.path.insert(0, _project_root)
# Add eoh/src to path for eoh module
_eoh_src = os.path.join(_project_root, 'eoh', 'src')
if _eoh_src not in sys.path:
    sys.path.insert(0, _eoh_src)

from eoh import eoh
from eoh.utils.getParas import Paras
from .prob import TSPGLSGeneratorProblem

def run_generator_evolution_with_population(
    config, 
    heuristics: List[str],
    sigma_h: np.ndarray,
    n_cities: int,
    output_dir: str = None,
    continue_from_generation: int = 0,
    evolution_context: str | None = None,
    eoh_eval_n_instances: int = None, 
) -> Tuple[Dict[str, Any], float] | List[Tuple[Dict[str, Any], float]]:
    """Run EoH evolution to evolve a generator with initial population.
    
    Args:
        config: HeuPSROConfig
        heuristics: List of solver heuristic codes 
        sigma_h: Mixture weights over heuristics 
        n_cities: Number of cities
        output_dir: Output directory 
        continue_from_generation: Continue from this generation 
        evolution_context: Evolution context string 
        eoh_eval_n_instances: Number of instances 
        
    Returns:
        Tuple of (best_generator_config, best_score) or list of tuples
    """
    use_top_k = getattr(config, 'psro_use_top_k', False)
    top_k = getattr(config, 'psro_top_k', 1)
    return_top_k = top_k if use_top_k else 1
    
    eval_n_instances = eoh_eval_n_instances or config.eoh_eval_n_instances
    
    # Create the problem instance
    problem = TSPGLSGeneratorProblem(
        config=config,
        heuristics=heuristics,
        sigma_h=sigma_h,
        n_cities=n_cities,
        n_inst_eva=eval_n_instances,
    )
    
    # Set evolution context if provided
    if evolution_context is not None:
        problem.set_evolution_context(evolution_context, enabled=True)
    else:
        problem.set_evolution_context(None, enabled=False)
    
    # Set parameters
    paras = Paras()
    
    # Get generator_n_pop with backward compatibility
    generator_n_pop = getattr(config, 'generator_n_pop', getattr(config, 'n_pop', 2))
    
    paras.set_paras(
        method="eoh",
        problem=problem,
        llm_api_endpoint=config.llm_api_endpoint,
        llm_api_key=config.llm_api_key,
        llm_model=config.llm_model,
        llm_use_local=config.llm_use_local,  
        ec_operators=config.ec_operators,
        ec_pop_size=config.pop_size,
        ec_n_pop=generator_n_pop,
        ec_m=config.ec_m,
        exp_n_proc=config.exp_n_proc,
        exp_debug_mode=config.exp_debug_mode,  
        eva_numba_decorator=config.eva_numba_decorator,  
        eva_timeout=config.eoh_framework_timeout,  
        exp_output_path=os.path.abspath(output_dir) if output_dir else os.path.abspath("./results/generator_eoh"),
        management=config.eoh_management_strategy,
        llm_use_async=getattr(config, 'llm_use_async', True),
        llm_max_concurrent_requests=getattr(config, 'llm_max_concurrent_requests', 10),
        llm_rate_limit_per_minute=getattr(config, 'llm_rate_limit_per_minute', 60),
        llm_temperature=getattr(config, 'llm_temperature', 0.7),
        llm_top_p=getattr(config, 'llm_top_p', 0.95),
        diversity_threshold=getattr(config, 'eoh_diversity_threshold', 0.8),
        objective_precision=getattr(config, 'eoh_objective_precision', 1),
        max_per_objective=getattr(config, 'eoh_max_per_objective', 1)
    )
    
    # Set up continue from previous generation
    if continue_from_generation > 0:
        # Continue from previous generation
        continue_path = os.path.join(output_dir, "results", "pops", f"population_generation_{continue_from_generation}.json")
        if os.path.exists(continue_path):
            paras.exp_use_continue = True
            paras.exp_continue_path = continue_path
            paras.exp_continue_id = continue_from_generation
            # Update n_pop to continue for n_pop more generations
            paras.ec_n_pop = continue_from_generation + generator_n_pop
        else:
            # If previous generation doesn't exist, this is an error
            raise FileNotFoundError(f"Previous generation {continue_from_generation} not found at {continue_path}")
    else:
        # First run: EOH will create initial population automatically
        paras.exp_use_continue = False
        paras.ec_n_pop = generator_n_pop
    
    # Initialize and run evolution
    evolution = eoh.EVOL(paras)
    evolution.run()
    
    # save evolution object to module variable, for bridge to access time information (minimal intrusion)
    import sys
    sys.modules[__name__].__dict__['_last_evolution'] = evolution
    
    # Get the best generator config from the evolution results
    import json
    
    # Find the last generation file
    output_dir = paras.exp_output_path
    pops_dir = os.path.join(output_dir, "results", "pops")
    
    if os.path.exists(pops_dir):
        # Get all population files and find the last one
        pop_files = [f for f in os.listdir(pops_dir) if f.startswith("population_generation_") and f.endswith(".json")]
        if pop_files:
            # Sort by generation number
            pop_files.sort(key=lambda x: int(x.split("_")[2].split(".")[0]))
            last_pop_file = pop_files[-1]
            
            # Load the final population
            with open(os.path.join(pops_dir, last_pop_file), 'r') as f:
                final_population = json.load(f)
            
            # Get the top-k individuals (sort by objective to find the best)
            if final_population:
                # Sort by objective (lower is better for minimization)
                # For generator: we return negative values, so smallest negative = largest tour length
                final_population.sort(key=lambda x: x.get('objective', float('inf')))
                
                # Return top-k individuals
                candidates = []
                for i in range(min(return_top_k, len(final_population))):
                    ind = final_population[i]
                    code = ind.get('code', '')
                    score = ind.get('objective', float('inf'))
                    algorithm = ind.get('algorithm', 'Unknown algorithm')
                    
                    if code:
                        # Return (code, algorithm, params, score) format
                        params = {
                            "n_cities": n_cities,
                            "seed": 42
                        }
                        candidates.append((code, algorithm, params, score))
                
                # Return single tuple for backward compatibility, or list if k>1
                if return_top_k == 1:
                    return candidates[0] if candidates else None
                else:
                    return candidates
    
    # Fallback: return a simple uniform generator if no results found
    print("    Warning: No EoH results found, using fallback generator")
    fallback_code = ""
    fallback_algorithm = "Fallback generator"
    fallback_params = {
        "n_cities": n_cities,
        "seed": 42
    }
    
    # for fallback, we need to compute score
    # problem.evaluate 期望 code_string: str
    try:
        fallback_score = problem.evaluate(fallback_code)
        print(f"    Fallback generator score: {fallback_score:.4f}")
    except Exception as e:
        print(f"    Warning: Could not evaluate fallback generator: {e}")
        fallback_score = 0.0  # if evaluation fails, use default value
    
    result = (fallback_code, fallback_algorithm, fallback_params, fallback_score)
    
    # Return based on return_top_k
    if return_top_k == 1:
        return result
    else:
        return [result]  # Return as list for consistency

