"""BP Online-specific EoH bridge implementation."""

from __future__ import annotations

from typing import List, Dict, Any, Callable, Tuple, Union, Optional
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 BPOnlineEoHBridge:
    """BP Online-specific bridge to EoH for PSRO dual evolution.
    
    Responsibilities:
    - Run EoH evolution for solver heuristics 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,
        continue_from_generation: int = 0,
        generator_codes: Optional[List[str]] = None,
        generator_ids: Optional[List[int]] = None,
        generator_weights: Optional[np.ndarray] = None,
        evolution_context: Optional[str] = None,
    ) -> Union[Tuple[str, float, str], List[Tuple[str, float, str]]]:
        """Approximate solver BR under fixed generator mix σ_G with initial population."""
        
        results = run_solver_evolution_with_population(
            config=self.config,
            generator_codes=generator_codes,
            generator_ids=generator_ids,
            generator_weights=generator_weights,
            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,
        continue_from_generation: int = 0,
        evolution_context: Optional[str] = None,
    ) -> Union[Tuple[Dict[str, Any], float], List[Tuple[Dict[str, Any], float]]]:
        """Approximate generator BR against fixed solver mix σ_H with initial population."""
        
        results = run_generator_evolution_with_population(
            config=self.config,
            heuristics=heuristics,
            sigma_h=sigma_h,
            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

