"""Baseline provider system for data-aware Expected GradCAM.

This module provides a flexible, modular system for managing baseline
data sources used in Expected GradCAM's data-aware perturbation mode.

Quick Start
-----------
The simplest way to create a provider is with :func:`baseline_from`::

    from expected_gradcam.baselines import baseline_from

    # From directory
    provider = baseline_from("/data/imagenet/train")

    # From HuggingFace dataset
    provider = baseline_from("imagenet-1k", split="train")

    # From cached features
    provider = baseline_from("/cache/features.npy")

Builder Pattern
---------------
For complex configurations, use :class:`BaselineProviderBuilder`::

    from expected_gradcam.baselines import BaselineProviderBuilder

    provider = (
        BaselineProviderBuilder()
        .from_huggingface("imagenet-1k")
        .with_split("train")
        .with_max_images(1000)
        .with_cache("/cache/features.npy")
        .build()
    )

Custom Providers
----------------
Register custom providers with the :func:`baseline_provider` decorator::

    from expected_gradcam.baselines import baseline_provider, BaseProvider

    @baseline_provider("my_custom", aliases=("custom",))
    class MyCustomProvider(BaseProvider):
        def __init__(self, my_param: str) -> None:
            super().__init__()
            self.my_param = my_param

        @property
        def provider_type(self) -> str:
            return "my_custom"

        def _do_initialize(self) -> None:
            # Load data and extract features
            ...

        def _get_raw_samples(self, n: int) -> Tensor:
            # Return n raw samples (centering applied automatically)
            ...

        def __len__(self) -> int:
            return self._n_samples

Available Providers
-------------------
- ``directory``: Load from directory of images
- ``imagenet``: ImageNet directory structure (ILSVRC format)
- ``huggingface``: HuggingFace datasets (requires ``datasets`` package)
- ``torch_dataset``: Wrap existing PyTorch datasets
- ``cached``: Load pre-extracted features from .npy/.npz files

Classes
-------
.. autosummary::
    :toctree: generated/

    BaselineProvider
    CacheableProvider
    StreamingProvider
    BaseProvider
    CacheableBaseProvider
    BaselineProviderBuilder
    ProviderRegistry

Functions
---------
.. autosummary::
    :toctree: generated/

    baseline_from
    baseline_provider
    get_registry
"""

from __future__ import annotations

# Protocols
from expected_gradcam.baselines.protocols import (
    BaselineProvider,
    CacheableProvider,
    StreamingProvider,
)

# Registry
from expected_gradcam.baselines.registry import (
    ProviderMetadata,
    ProviderRegistry,
    baseline_provider,
    get_registry,
)

# Base classes
from expected_gradcam.baselines.base import (
    BaseProvider,
    CacheableBaseProvider,
)

# Builder and factory
from expected_gradcam.baselines.builder import BaselineProviderBuilder
from expected_gradcam.baselines.factory import baseline_from

# Concrete providers (imported for registration side effects)
from expected_gradcam.baselines.providers import (
    CachedFeatureProvider,
    DirectoryProvider,
    ImageNetProvider,
    TorchDatasetProvider,
)

# Conditional HuggingFace import
try:
    from expected_gradcam.baselines.providers import HuggingFaceProvider

    _HAS_HF = True
except ImportError:
    HuggingFaceProvider = None  # type: ignore[misc, assignment]
    _HAS_HF = False


def __getattr__(name: str):
    """Lazy attribute access for optional dependencies."""
    if name == "HuggingFaceProvider":
        if not _HAS_HF:
            raise ImportError(
                "HuggingFaceProvider requires the 'datasets' package. "
                "Install with: pip install expected-gradcam[hf]"
            )
        return HuggingFaceProvider
    raise AttributeError(f"module {__name__!r} has no attribute {name!r}")


__all__ = [
    # Protocols
    "BaselineProvider",
    "CacheableProvider",
    "StreamingProvider",
    # Registry
    "ProviderRegistry",
    "ProviderMetadata",
    "baseline_provider",
    "get_registry",
    # Base classes
    "BaseProvider",
    "CacheableBaseProvider",
    # Builder and factory
    "BaselineProviderBuilder",
    "baseline_from",
    # Concrete providers
    "DirectoryProvider",
    "ImageNetProvider",
    "TorchDatasetProvider",
    "CachedFeatureProvider",
    "HuggingFaceProvider",
]
