#!/usr/bin/env python3
"""
EoH evolution for BP Online solver heuristics
"""

import os
import sys
from typing import Callable, List, Dict, Any, Tuple, Union, Optional
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 BPOnlineSolverProblem

def run_solver_evolution_with_population(
    config, 
    generator_codes: Optional[List[str]] = None,
    generator_ids: Optional[List[int]] = None,
    generator_weights: Optional[np.ndarray] = None,
    output_dir: str = None,
    continue_from_generation: int = 0,
    evolution_context: Optional[str] = None,
    eoh_eval_n_instances: int = None, 
) -> Union[Tuple[str, float, str], List[Tuple[str, float, str]]]:
    """Run EoH evolution to evolve a solver heuristic with initial population.
    
    Args:
        config: HeuPSROConfig 
        generator_codes: Generator code strings 
        generator_ids: Generator IDs
        generator_weights: Generator weights 
        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_heuristic_code, best_score, best_algorithm) 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
    if generator_codes is not None and generator_ids is not None and generator_weights is not None:
        problem = BPOnlineSolverProblem(
            config=config,
            generator_codes=generator_codes,
            generator_ids=generator_ids,
            generator_weights=generator_weights,
            n_inst_eva=eval_n_instances,
        )
    else:
        raise ValueError("generator_codes, generator_ids, and generator_weights must be provided")
    
    # 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 - 所有从 config 获取
    paras = Paras()
    
    # Get solver_n_pop with backward compatibility
    solver_n_pop = getattr(config, 'solver_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=solver_n_pop,
        ec_m=config.ec_m,
        exp_n_proc=config.exp_n_proc,
        exp_debug_mode=config.exp_debug_mode,
        debug_mode=getattr(config, 'debug_mode', False),
        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/solver_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_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
            paras.ec_n_pop = continue_from_generation + solver_n_pop
        else:
            raise FileNotFoundError(f"Previous generation {continue_from_generation} not found at {continue_path}")
    else:
        paras.exp_use_continue = False
        paras.ec_n_pop = solver_n_pop
    
    # Initialize and run evolution
    evolution = EVOL(paras)
    evolution.run()
    
    # Get the best heuristic code from the evolution results
    import json
    
    output_dir = paras.exp_output_path
    pops_dir = os.path.join(output_dir, "results", "pops")
    
    if os.path.exists(pops_dir):
        pop_files = [f for f in os.listdir(pops_dir) if f.startswith("population_generation_") and f.endswith(".json")]
        if pop_files:
            pop_files.sort(key=lambda x: int(x.split("_")[2].split(".")[0]))
            last_pop_file = pop_files[-1]
            
            with open(os.path.join(pops_dir, last_pop_file), 'r') as f:
                final_population = json.load(f)
            
            if final_population:
                final_population.sort(key=lambda x: x.get('objective', float('inf')))
                
                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:
                        candidates.append((code, algorithm, {}, score))
                
                if return_top_k == 1:
                    return candidates[0] if candidates else None
                else:
                    return candidates
    
    # Fallback: return a simple heuristic if no results found
    print("    Warning: No EoH results found, using fallback solver")
    best_code = (
        "import numpy as np\n"
        "def score(item, bins):\n"
        "    # Fallback heuristic: prefer bins with more remaining capacity\n"
        "    scores = bins.copy()\n"
        "    return scores\n"
    )
    
    fallback_score = problem.evaluate(best_code)
    print(f"    Fallback solver score: {fallback_score:.4f}")
    result = (best_code, "Fallback solver", {}, fallback_score)
    
    sys.modules[__name__].__dict__['_last_evolution'] = evolution
    
    if return_top_k == 1:
        return result
    else:
        return [result]

