"""Metrics computation for Expected GradCAM.

This package provides metrics for evaluating and monitoring Expected GradCAM
computations, including infidelity, condition number, effective rank, and more.

The metrics system features:
- Automatic registration via metaclass
- Validated configuration via descriptors
- Streaming metrics for real-time visualization
- Comprehensive error handling with suggestions

Example:
    >>> from expected_gradcam.metrics import (
    ...     InternalInfidelity,
    ...     ConditionNumber,
    ...     EffectiveRank,
    ...     MetricConfig,
    ... )
    >>>
    >>> # Create metrics
    >>> infidelity = InternalInfidelity()
    >>> condition = ConditionNumber()
    >>>
    >>> # Compute metrics
    >>> inf_value = infidelity.compute(
    ...     alpha=alpha,
    ...     I_samples=I_samples,
    ...     g_z0=g_z0,
    ...     g_perturbed=g_perturbed,
    ... )
    >>> cond_value = condition.compute(M_I=M_I)
    >>>
    >>> # Access via registry
    >>> from expected_gradcam.metrics import MetricRegistry
    >>> MetricRegistry.list_metrics()
    ['condition_number', 'effective_rank', 'internal_infidelity', ...]

Metric Categories:
    - **infidelity**: Explanation quality metrics (internal, batched)
    - **solver**: Linear system metrics (condition, rank, residual)
    - **heatmap**: Output quality metrics (weight_norm, entropy)
"""

from __future__ import annotations

# Base classes and decorators
from expected_gradcam.metrics.base import (
    BaseMetric,
    MetricProtocol,
    StreamingMetric,
    StreamingMetricProtocol,
    cached,
    check_finite,
    gpu_sync,
    inference_mode,
    no_grad,
    require_grad,
    timed,
    validate_inputs,
)

# Configuration
from expected_gradcam.metrics.config import (
    DEFAULT_METRIC_CONFIG,
    FAST_METRIC_CONFIG,
    RESEARCH_METRIC_CONFIG,
    MetricConfig,
    VisualizationMetricConfig,
)

# Exceptions
from expected_gradcam.metrics.exceptions import (
    InfidelityComputationError,
    InsufficientSamplesError,
    InvalidMetricInputError,
    MetricComputationError,
    MetricError,
    MetricNotFoundError,
    NumericalInstabilityError,
)

# Registry
from expected_gradcam.metrics.registry import (
    MetricRegistry,
    MetricRegistryMeta,
    register_metric,
)


def __getattr__(name: str):  # noqa: ANN202
    """Lazy import of metric classes to avoid circular imports."""
    # Infidelity metrics
    if name == "InternalInfidelity":
        from expected_gradcam.metrics.infidelity.internal import InternalInfidelity

        return InternalInfidelity
    if name == "BatchedInfidelity":
        from expected_gradcam.metrics.infidelity.batched import BatchedInfidelity

        return BatchedInfidelity

    # Solver metrics
    if name == "ConditionNumber":
        from expected_gradcam.metrics.solver.condition import ConditionNumber

        return ConditionNumber
    if name == "EffectiveRank":
        from expected_gradcam.metrics.solver.condition import EffectiveRank

        return EffectiveRank
    if name == "EigenvalueSpread":
        from expected_gradcam.metrics.solver.condition import EigenvalueSpread

        return EigenvalueSpread
    if name == "ResidualNorm":
        from expected_gradcam.metrics.solver.residual import ResidualNorm

        return ResidualNorm

    # Heatmap metrics
    if name == "WeightNorm":
        from expected_gradcam.metrics.heatmap.weight_norm import WeightNorm

        return WeightNorm
    if name == "HeatmapEntropy":
        from expected_gradcam.metrics.heatmap.entropy import HeatmapEntropy

        return HeatmapEntropy

    raise AttributeError(f"module {__name__!r} has no attribute {name!r}")


__all__ = [
    # Base classes
    "BaseMetric",
    "StreamingMetric",
    # Protocols
    "MetricProtocol",
    "StreamingMetricProtocol",
    # Decorators
    "timed",
    "validate_inputs",
    "cached",
    "check_finite",
    "require_grad",
    "no_grad",
    "inference_mode",
    "gpu_sync",
    # Registry
    "MetricRegistry",
    "MetricRegistryMeta",
    "register_metric",
    # Configuration
    "MetricConfig",
    "VisualizationMetricConfig",
    "DEFAULT_METRIC_CONFIG",
    "FAST_METRIC_CONFIG",
    "RESEARCH_METRIC_CONFIG",
    # Exceptions
    "MetricError",
    "MetricComputationError",
    "InfidelityComputationError",
    "MetricNotFoundError",
    "InsufficientSamplesError",
    "NumericalInstabilityError",
    "InvalidMetricInputError",
    # Metric classes (lazy loaded)
    "InternalInfidelity",
    "BatchedInfidelity",
    "ConditionNumber",
    "EffectiveRank",
    "EigenvalueSpread",
    "ResidualNorm",
    "WeightNorm",
    "HeatmapEntropy",
]
