from __future__ import annotations

from dataclasses import dataclass
from typing import Dict, List, Optional, Tuple


@dataclass
class SolverProgram:
    program_id: str
    code: str
    algorithm: str  # Algorithm name for EoH compatibility
    params: Dict  # Problem-specific parameters (can be empty for solver)
    metadata: Dict


@dataclass
class GeneratorProgram:
    program_id: str
    code: str  # Generator code (like SolverProgram)
    algorithm: str  # Algorithm name for EoH compatibility
    params: Dict  # Problem-specific parameters (e.g., n_cities, capacity, num_items)
    metadata: Dict


class StrategyPools:
    def __init__(self, meta=None) -> None:
        """StrategyPools: Solver and Generator pool
        """
        self.solver_pool: List[SolverProgram] = []
        self.generator_pool: List[GeneratorProgram] = []
        self.meta = meta

    def add_solver(self, program_id: str, code: str, algorithm: str, params: Optional[Dict] = None, metadata: Optional[Dict] = None) -> int:
        """
        
        Args:
            program_id: Solver identifier
            code: Solver code string
            algorithm: Algorithm name for EoH compatibility
            params: Problem-specific parameters (can be empty for solver)
            metadata: Optional metadata
        """
        self.solver_pool.append(SolverProgram(program_id, code, algorithm, params or {}, metadata or {}))
        idx = len(self.solver_pool) - 1
        
        if self.meta is not None:
            meta_source = metadata.get("source", "expanded") if metadata else "expanded"
            self.meta.add_solver(program_id, {"source": meta_source})
        
        return idx

    def add_generator(self, program_id: str, code: str, algorithm: str, params: Optional[Dict] = None, metadata: Optional[Dict] = None) -> int:
        """
        Args:
            program_id: Generator identifier
            code: Generator code string
            algorithm: Algorithm name for EoH compatibility
            params: Problem-specific parameters (e.g., n_cities, capacity, num_items)
            metadata: Optional metadata
        """
        self.generator_pool.append(GeneratorProgram(program_id, code, algorithm, params or {}, metadata or {}))
        idx = len(self.generator_pool) - 1
        
        if self.meta is not None:
            meta_source = metadata.get("source", "expanded") if metadata else "expanded"
            self.meta.add_generator(program_id, {"source": meta_source})
        
        return idx

    def remove_solver(self, idx: int) -> None:
        sp = self.solver_pool.pop(idx)
        if self.meta is not None:
            self.meta.remove_solver(idx)

    def remove_generator(self, idx: int) -> None:
        gp = self.generator_pool.pop(idx)
        if self.meta is not None:
            self.meta.remove_generator(idx)

    def get_solver(self, idx: int) -> SolverProgram:
        return self.solver_pool[idx]

    def get_generator(self, idx: int) -> GeneratorProgram:
        return self.generator_pool[idx]

    @property
    def n_solvers(self) -> int:
        return len(self.solver_pool)

    @property
    def n_generators(self) -> int:
        return len(self.generator_pool)



