"""Evolution context builder for PSRO.

This module builds context descriptions for EOH evolution,
connecting PSRO-level strategy distributions to EOH prompts.
"""

from __future__ import annotations

from typing import List
import numpy as np

from .pools import StrategyPools


def build_solver_context(
    pools: StrategyPools,
    sigma_g_full: np.ndarray,
    sigma_g: np.ndarray
) -> str:
    """
    Build evolution context for solver evolution.
    
    Creates a text description of the mixed generator distribution
    that will be used as context in EOH prompts.
    
    Args:
        pools: Strategy pools containing generators
        sigma_g_full: Full generator mixture (including simple/baseline generator at index 0)
        sigma_g: Evolved generator mixture (excluding simple/baseline)
        
    Returns:
        Context string describing the mixed generator distribution
    """
    context_lines = [
        "You are evolving a solver heuristic to handle instances from a mixed generator distribution:"
    ]
    
    # Filter strategies by weight threshold to avoid context noise
    weight_threshold = 0.02
    generator_items = []
    
    # Add simple/baseline generator (index 0)
    if sigma_g_full[0] >= weight_threshold:
        generator_items.append((
            "baseline",
            sigma_g_full[0],
            "Generates random instances"
        ))
    
    # Add evolved generators
    if pools.n_generators > 0 and sigma_g.size > 0:
        remaining_weight = 1.0 - sigma_g_full[0]
        for g_idx, weight in enumerate(sigma_g):
            if weight > 0:
                # Weight in full mixture
                weight_in_full = weight * remaining_weight if remaining_weight > 0 else 0
                if weight_in_full >= weight_threshold:
                    gen_program = pools.get_generator(g_idx)
                    gen_id = gen_program.program_id
                    algorithm = gen_program.algorithm or "Unknown generator algorithm"
                    generator_items.append((gen_id, weight_in_full, algorithm))
    
    # Format as structured list
    if generator_items:
        context_lines.append("\nCurrent generator mixture (weights sum to 1.0):")
        for gen_id, weight, algorithm in generator_items:
            context_lines.append(
                f'- {weight:.2f} :: id = {gen_id}, algorithm = "{algorithm}"'
            )
    
    context_lines.append("\nYour objective is to minimize the weighted average gap across this mixed generator distribution.")
    context_lines.append(
        "Before designing the algorithm, carefully read the algorithm descriptions of each generator above and "
        "reason about what types of instances they tend to produce. "
        "Then design a solver strategy that is robust against this mixture, rather than overfitting to a single generator."
    )
    return "\n".join(context_lines)


def build_generator_context(
    pools: StrategyPools,
    heuristics: List[str],
    sigma_h: np.ndarray
) -> str:
    """
    Build evolution context for generator evolution.
    
    Creates a text description of the mixed solver strategy
    that will be used as context in EOH prompts.
    
    Args:
        pools: Strategy pools containing solvers
        heuristics: List of solver heuristic codes
        sigma_h: Solver mixture weights
        
    Returns:
        Context string describing the mixed solver strategy
    """
    context_lines = [
        "You are evolving a generator to create hard instances for a mixed solver strategy:"
    ]
    
    # Filter strategies by weight threshold to avoid context noise
    weight_threshold = 0.02
    solver_items = []
    
    if pools.n_solvers > 0 and sigma_h.size > 0:
        for h_idx, weight in enumerate(sigma_h):
            if weight >= weight_threshold and h_idx < pools.n_solvers:
                solver_program = pools.get_solver(h_idx)
                solver_id = solver_program.program_id
                algorithm = solver_program.algorithm or "Unknown solver algorithm"
                solver_items.append((solver_id, weight, algorithm))
    
    # Format as structured list
    if solver_items:
        context_lines.append("\nCurrent solver mixture:")
        for solver_id, weight, algorithm in solver_items:
            context_lines.append(
                f'- {weight:.2f} :: id = {solver_id}, algorithm = "{algorithm}"'
            )
    
    context_lines.append("\nYour objective is to maximize the weighted average gap achieved by this solver distribution.")
    context_lines.append(
        "Before designing the generator, carefully read the algorithm descriptions of each solver above and "
        "reason about what kinds of instances are likely to be hard for this mixture as a whole. "
        "Then design a generator strategy that exploits weaknesses of the mixture, not just a single solver."
    )
    return "\n".join(context_lines)

