"""BP Online-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

# Try to import scipy.special.erf for numpy.erf compatibility
try:
    from scipy.special import erf as scipy_erf
    HAS_SCIPY_ERF = True
except ImportError:
    HAS_SCIPY_ERF = False
    # Fallback: use math.erf if available (Python 3.7+)
    try:
        from math import erf as math_erf
        scipy_erf = math_erf
    except ImportError:
        scipy_erf = None


def prepare_exec_namespace() -> Dict[str, Any]:
    """
    Prepare a namespace for exec() that includes numpy.erf compatibility.
    
    Returns:
        Dictionary with numpy and other standard imports, including numpy.erf
    """
    namespace = {
        '__builtins__': __builtins__,
        'numpy': np,
        'np': np,
    }
    
    # Add numpy.erf compatibility
    # Since numpy doesn't have erf in newer versions, we add it directly to the namespace
    # This avoids pickling issues with wrapper classes
    if HAS_SCIPY_ERF and scipy_erf is not None:
        # Add erf as a standalone function in the namespace
        namespace['erf'] = scipy_erf
        # Also try to add it to numpy module (may not work if numpy is read-only)
        try:
            # Create a copy of numpy module attributes and add erf
            # This is safer than modifying the original numpy module
            numpy_module = types.ModuleType('numpy')
            # Copy all numpy attributes
            for attr_name in dir(np):
                if not attr_name.startswith('_'):
                    try:
                        setattr(numpy_module, attr_name, getattr(np, attr_name))
                    except (TypeError, AttributeError):
                        pass
            numpy_module.erf = scipy_erf
            namespace['numpy'] = numpy_module
            namespace['np'] = numpy_module
        except Exception:
            # If that fails, just add erf to namespace and let code use it directly
            # Code can use: erf(x) instead of numpy.erf(x)
            pass
    
    return namespace


class BPOnlineInstanceSampler:
    """BP Online-specific instance sampler."""
    
    def __init__(self, pools: StrategyPools, capacity: int, num_items: int, debug_mode: bool = False):
        """
        Initialize BP Online instance sampler.
        
        Args:
            pools: StrategyPools instance
            capacity: Bin capacity
            num_items: Number of items per instance
            debug_mode: Whether to print debug information
        """
        self.pools = pools
        self.capacity = capacity
        self.num_items = num_items
        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 'items', 'capacity', 'num_items'
        """
        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., capacity, num_items)
            n: Number of complete instances to generate
            
        Returns:
            List of n complete BP Online instances
        """
        if code:
            try:
                gen_mod = types.ModuleType("generator_module")
                
                # Prepare namespace with numpy.erf compatibility
                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)
                    capacity = params.get("capacity", self.capacity)
                    num_items = params.get("num_items", self.num_items)
                    # Generator now returns List[np.ndarray], each is items array
                    items_arrays = gen_mod.generate_instances(seeds, capacity, num_items)
                    
                    # External wrapper assembles dict format
                    instances = []
                    for items in items_arrays:
                        items = np.asarray(items, dtype=int)
                        if len(items) == num_items and items.min() >= 1 and items.max() < capacity:
                            instances.append({
                                'items': items,
                                'capacity': capacity,
                                'num_items': num_items
                            })
                    return instances if instances else self.generate_uniform_instances(n)
            except Exception as 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 random items uniformly in [1, capacity-1]
            items = np.random.randint(1, self.capacity, size=self.num_items)
            instance = {
                'items': items,
                'capacity': self.capacity,
                'num_items': self.num_items
            }
            instances.append(instance)
        return instances

