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

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 EVOL
from eoh.utils.getParas import Paras
from .prob import CVRPGeneratorProblem

def run_generator_evolution_with_population(
    config,
    heuristics: List[str],
    sigma_h: np.ndarray,
    num_customers: int,
    output_dir: str = None,
    continue_from_generation: int = 0,
    evolution_context: str | None = None,
    eoh_eval_n_instances: int = None,  
) -> Tuple[str, str, Dict[str, Any], float] | List[Tuple[str, str, Dict[str, Any], float]]:
    """Run EoH evolution to evolve a generator with initial population.
    
    Args:
        config: HeuPSROConfig 
        heuristics: List of solver codes 
        sigma_h: Mixture weights over heuristics
        num_customers: Number of customers 
        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 (code, algorithm, params, score) or list of tuples
    """
    # calculate return_top_k from config
    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 = CVRPGeneratorProblem(
        config=config,
        heuristics=heuristics,
        sigma_h=sigma_h,
        num_customers=num_customers,
        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)
    
    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
        # Use paras.exp_output_path which already handles None case
        output_path = paras.exp_output_path if hasattr(paras, 'exp_output_path') else (output_dir or "./results/generator_eoh")
        continue_path = os.path.join(output_path, "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
    
    # Run evolution
    evolution = EVOL(paras)
    
    # Store evolution object for time tracking
    sys.modules[__name__].__dict__['_last_evolution'] = evolution
    
    evolution.run()
    
    # Get the best generator config from the evolution results
    import json
    
    # Find the last generation file
    output_dir = paras.exp_output_path if hasattr(paras, 'exp_output_path') else (output_dir or "./results/generator_eoh")
    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 gap
                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 format: (code, algorithm, params, score)
                        vehicle_capacity = getattr(config, 'vehicle_capacity', 100)
                        params = {
                            "num_customers": num_customers,
                            "vehicle_capacity": vehicle_capacity,
                            "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 = (
        "def generate_instances(seeds, num_customers, vehicle_capacity):\n"
        "    import numpy as np\n"
        "    instances = []\n"
        "    for seed in seeds:\n"
        "        rng = np.random.default_rng(seed)\n"
        "        depot = rng.uniform(0, 100, 2).tolist()\n"
        "        customers = []\n"
        "        for i in range(num_customers):\n"
        "            coords = rng.uniform(0, 100, 2).tolist()\n"
        "            demand = rng.integers(1, vehicle_capacity // 2 + 1)\n"
        "            customers.append({'coords': coords, 'demand': int(demand)})\n"
        "        instances.append({'depot': depot, 'customers': customers})\n"
        "    return instances\n"
    )
    vehicle_capacity = getattr(config, 'vehicle_capacity', 100)
    fallback_params = {
        "num_customers": num_customers,
        "vehicle_capacity": vehicle_capacity,
        "seed": 42
    }
    if return_top_k == 1:
        return (fallback_code, "Fallback uniform generator", fallback_params, float('inf'))
    else:
        return [(fallback_code, "Fallback uniform generator", fallback_params, float('inf'))]

