"""CVRP-specific EoH bridge implementation."""

from __future__ import annotations

from typing import List, Dict, Any, Callable, Tuple
import numpy as np

from .evolution.solver.run_eoh import run_solver_evolution_with_population
from .evolution.generator.run_eoh import run_generator_evolution_with_population


class CVRPEoHBridge:
    """CVRP-specific bridge to EoH for PSRO dual evolution.
    
    Responsibilities:
    - Run EoH evolution for solver on mixed generator instances
    - Run EoH evolution for generator programs against mixed solver heuristics
    - Return evolved code/configs as approximate best responses
    """

    def __init__(self, config, solver_eoh_dir: str = None, generator_eoh_dir: str = None) -> None:
        self.config = config
        self.solver_eoh_dir = solver_eoh_dir
        self.generator_eoh_dir = generator_eoh_dir

    def evolve_solver_oracle_with_population(
        self,
        generator_samplers: List[Callable[[int], List[Dict]]] = None,
        sigma_g: np.ndarray = None,
        n_instances: int = None,
        num_customers: int = None,
        continue_from_generation: int = 0,
        generator_codes: List[str] | None = None,
        generator_ids: List[int] | None = None,
        generator_weights: np.ndarray | None = None,
        evolution_context: str | None = None,
    ) -> Tuple[str, str, Dict[str, Any], float] | List[Tuple[str, str, Dict[str, Any], float]]:
        """Approximate solver BR under fixed generator mix σ_G with initial population."""
        
        num_customers = num_customers or getattr(self.config, 'num_customers', None)
        if num_customers is None:
            raise ValueError("num_customers must be provided either as parameter or in config")
        
        results = run_solver_evolution_with_population(
            config=self.config,
            generator_codes=generator_codes,
            generator_ids=generator_ids,
            generator_weights=generator_weights,
            num_customers=num_customers,
            output_dir=self.solver_eoh_dir,
            continue_from_generation=continue_from_generation,
            evolution_context=evolution_context,
            eoh_eval_n_instances=n_instances,
        )
        
        try:
            from .evolution.solver import run_eoh
            evolution = getattr(run_eoh, '_last_evolution', None)
            if evolution and hasattr(evolution, 'method') and hasattr(evolution.method, '_code_generation_time'):
                self._last_solver_eoh_time_info = {
                    'code_generation_time': evolution.method._code_generation_time,
                    'evaluation_time': evolution.method._evaluation_time
                }
            else:
                self._last_solver_eoh_time_info = {}
        except Exception:
            self._last_solver_eoh_time_info = {}
        
        return results
    
    def evolve_generator_oracle_with_population(
        self,
        heuristics: List[str],
        sigma_h: np.ndarray,
        num_customers: int = None,
        continue_from_generation: int = 0,
        evolution_context: str | None = None,
    ) -> Tuple[str, str, Dict[str, Any], float] | List[Tuple[str, str, Dict[str, Any], float]]:
        """Approximate generator BR against fixed solver mix σ_H with initial population."""
        
        num_customers = num_customers or getattr(self.config, 'num_customers', None)
        if num_customers is None:
            raise ValueError("num_customers must be provided either as parameter or in config")
        
        results = run_generator_evolution_with_population(
            config=self.config,
            heuristics=heuristics,
            sigma_h=sigma_h,
            num_customers=num_customers,
            output_dir=self.generator_eoh_dir,
            continue_from_generation=continue_from_generation,
            evolution_context=evolution_context,
        )
        
        try:
            from .evolution.generator import run_eoh
            evolution = getattr(run_eoh, '_last_evolution', None)
            if evolution and hasattr(evolution, 'method') and hasattr(evolution.method, '_code_generation_time'):
                self._last_generator_eoh_time_info = {
                    'code_generation_time': evolution.method._code_generation_time,
                    'evaluation_time': evolution.method._evaluation_time
                }
            else:
                self._last_generator_eoh_time_info = {}
        except Exception:
            self._last_generator_eoh_time_info = {}
        
        return results




