"""Sampling module for Expected GradCAM perturbations and baselines.

This module provides samplers for:
- Perturbations I for infidelity minimization
- Baselines D for Expected Gradients path integration

Data-aware samplers (recommended) stay on the data manifold for
theoretical correctness. Batched samplers provide ~60x speedup.

Perturbation Samplers:
    - DataAwarePerturbationSampler: Sequential data-aware sampling
    - BatchedPerturbationSampler: GPU-batched data-aware sampling (~60x faster)
    - SimpleBatchedSampler: Fast Gaussian sampling (no dataset required)

Baseline Samplers:
    - CenteredBaselineSampler: Gaussian/uniform/sphere distributions
    - DataAwareBaselineSampler: Uses pre-cached feature maps
    - DataAwareEGBaselineSampler: Extracts from real images

Utility Functions:
    - center_samples: Center samples to have zero mean
    - verify_centered: Check if samples are centered
    - compute_optimal_baseline_scale: Auto-compute scale from feature maps
    - sample_centered_baselines: Convenience function for simple baselines
    - sample_data_aware_baselines: Convenience function for data-aware baselines

Example:
    >>> from expected_gradcam.sampling import BatchedPerturbationSampler
    >>> from expected_gradcam.data import BaselineDataset
    >>>
    >>> dataset = BaselineDataset.from_directory("imagenet/train/")
    >>> sampler = BatchedPerturbationSampler(
    ...     model, target_layer, dataset, device, batch_size=64
    ... )
    >>> I_samples = sampler.sample(z0, features, M=2500)

Mathematical Background:
    For Expected GradCAM, we need:
    - Perturbations I: Used to compute M_I = E[I * I^T] and b = E[I * <I, φ>]
    - Baselines D: Used in Expected Gradients path integration

    CRITICAL: Baselines must be centered (E[z'] = 0) for the completeness axiom:
        I^T @ φ^{EG} = g(z_0) - g(z_0 - I)

    Data-aware sampling (recommended):
        I = z_0 - α * h(x')  where h(x') = GAP(A') / GAP(A_input)

    This ensures perturbations stay on the data manifold, providing
    theoretical guarantees that random Gaussian noise cannot.
"""

from __future__ import annotations

# Perturbation samplers
from expected_gradcam.sampling.perturbation import DataAwarePerturbationSampler
from expected_gradcam.sampling.batched_perturbation import (
    BatchedPerturbationSampler,
    SimpleBatchedSampler,
)

# Baseline samplers
from expected_gradcam.sampling.baseline import (
    CenteredBaselineSampler,
    DataAwareBaselineSampler,
    DataAwareEGBaselineSampler,
    sample_centered_baselines,
    sample_data_aware_baselines,
    sample_from_provider,
)

# Utility functions
from expected_gradcam.sampling.utils import (
    center_samples,
    verify_centered,
    compute_optimal_baseline_scale,
    normalize_perturbations,
    safe_divide,
)


__all__ = [
    # Perturbation samplers
    "DataAwarePerturbationSampler",
    "BatchedPerturbationSampler",
    "SimpleBatchedSampler",
    # Baseline samplers
    "CenteredBaselineSampler",
    "DataAwareBaselineSampler",
    "DataAwareEGBaselineSampler",
    # Convenience functions
    "sample_centered_baselines",
    "sample_data_aware_baselines",
    "sample_from_provider",
    # Utility functions
    "center_samples",
    "verify_centered",
    "compute_optimal_baseline_scale",
    "normalize_perturbations",
    "safe_divide",
]
