"""CVRP-specific instance sampling logic."""

from __future__ import annotations

from typing import List, Dict, Any
import numpy as np
import types

from ...core.pools import StrategyPools


def prepare_exec_namespace() -> Dict[str, Any]:
    """
    Prepare a namespace for exec() that includes numpy.
    
    Returns:
        Dictionary with numpy and other standard imports
    """
    namespace = {
        '__builtins__': __builtins__,
        'numpy': np,
        'np': np,
    }
    return namespace


class CVRPInstanceSampler:
    """CVRP-specific instance sampler."""
    
    def __init__(self, pools: StrategyPools, num_customers: int, vehicle_capacity: int, 
                 num_vehicles: int = None, debug_mode: bool = False):
        """
        Initialize CVRP instance sampler.
        
        Args:
            pools: StrategyPools instance
            num_customers: Number of customers (excluding depot)
            vehicle_capacity: Vehicle capacity
            num_vehicles: Number of vehicles (optional)
            debug_mode: Whether to print debug information
        """
        self.pools = pools
        self.num_customers = num_customers
        self.vehicle_capacity = vehicle_capacity
        self.num_vehicles = num_vehicles
        self.debug_mode = debug_mode
    
    def sample_from_single_generator(self, g_idx: int, n: int) -> List[Dict]:
        """
        Sample n instances from a single generator.
        
        Args:
            g_idx: Generator index
            n: Number of instances to sample
            
        Returns:
            List of n instances, each is a Dict with 'depot', 'customers', 'vehicle_capacity'
        """
        gen_prog = self.pools.get_generator(g_idx)
        if gen_prog is None:
            return self.generate_uniform_instances(n)
        
        return self.generate_from_program(gen_prog.code, gen_prog.params, n)
    
    def generate_from_program(self, code: str, params: Dict, n: int) -> List[Dict]:
        """
        Generate n complete instances from a generator program.
        
        Args:
            code: Generator code string
            params: Generator parameters (e.g., num_customers, vehicle_capacity)
            n: Number of complete instances to generate
            
        Returns:
            List of n complete CVRP instances
        """
        if code:
            try:
                gen_mod = types.ModuleType("generator_module")
                
                # Prepare namespace
                exec_namespace = prepare_exec_namespace()
                exec_namespace.update(gen_mod.__dict__)
                
                exec(code, exec_namespace)
                gen_mod.__dict__.update(exec_namespace)
                if hasattr(gen_mod, "generate_instances"):
                    seeds = np.random.randint(0, 2**31, size=n)
                    num_customers = params.get("num_customers", self.num_customers)
                    vehicle_capacity = params.get("vehicle_capacity", self.vehicle_capacity)
                    return gen_mod.generate_instances(seeds, num_customers, vehicle_capacity)
            except Exception as e:
                if self.debug_mode:
                    print(f"      Warning: Generator failed: {e}")
                return self.generate_uniform_instances(n)
        
        return self.generate_uniform_instances(n)
    
    def generate_uniform_instances(self, n: int) -> List[Dict]:
        """
        Generate n uniform random instances as fallback.
        
        Args:
            n: Number of instances to generate
            
        Returns:
            List of n uniform random instances
        """
        instances = []
        for _ in range(n):
            # Generate depot location (center of area)
            depot = np.random.uniform(0, 100, size=2).tolist()
            
            # Generate customer locations
            customer_coords = np.random.uniform(0, 100, size=(self.num_customers, 2))
            
            # Generate customer demands (between 1 and vehicle_capacity // 3)
            max_demand = max(1, self.vehicle_capacity // 3)
            demands = np.random.integers(1, max_demand + 1, size=self.num_customers)
            
            customers = []
            for i in range(self.num_customers):
                customers.append({
                    'coords': customer_coords[i].tolist(),
                    'demand': int(demands[i])
                })
            
            instance = {
                'depot': depot,
                'customers': customers,
                'vehicle_capacity': self.vehicle_capacity
            }
            
            if self.num_vehicles is not None:
                instance['num_vehicles'] = self.num_vehicles
            
            instances.append(instance)
        return instances




