"""Architecture-related exceptions."""

from __future__ import annotations

from typing import TYPE_CHECKING

from expected_gradcam.exceptions.base import ExpectedGradCAMError


if TYPE_CHECKING:
    from collections.abc import Sequence


class ArchitectureError(ExpectedGradCAMError):
    """Base exception for architecture-related errors."""

    pass


class UnsupportedArchitectureError(ArchitectureError):
    """Raised when a model architecture is not supported.

    This error occurs when:
    - No registered plugin supports the given model
    - The model structure cannot be automatically analyzed

    Example:
        >>> raise UnsupportedArchitectureError(
        ...     "CustomModel",
        ...     supported=["ResNet", "VGG", "DenseNet"]
        ... )
    """

    def __init__(
        self,
        model_name: str,
        *,
        supported: Sequence[str] | None = None,
    ) -> None:
        """Initialize the exception.

        Args:
            model_name: Name or type of the unsupported model.
            supported: List of supported architecture names.
        """
        self.model_name = model_name
        self.supported = list(supported) if supported else []

        message = f"Architecture '{model_name}' is not supported."
        if self.supported:
            supported_str = ", ".join(self.supported)
            suggestion = (
                f"Supported architectures: {supported_str}. "
                "You can provide a custom ArchitecturePlugin or pass "
                "classifier_head and target_layer explicitly."
            )
        else:
            suggestion = (
                "Provide a custom ArchitecturePlugin or pass "
                "classifier_head and target_layer explicitly."
            )

        super().__init__(message, suggestion=suggestion)


class ClassifierHeadExtractionError(ArchitectureError):
    """Raised when the classifier head cannot be extracted from a model.

    This error occurs when:
    - The model structure is unexpected for the detected architecture
    - Required layers (GAP, FC) cannot be found
    - Layer dimensions don't match expected values

    Example:
        >>> raise ClassifierHeadExtractionError(
        ...     "ResNet50",
        ...     reason="Could not find fully-connected layer"
        ... )
    """

    def __init__(
        self,
        model_name: str,
        *,
        reason: str | None = None,
    ) -> None:
        """Initialize the exception.

        Args:
            model_name: Name of the model.
            reason: Specific reason for the extraction failure.
        """
        self.model_name = model_name
        self.reason = reason

        if reason:
            message = f"Failed to extract classifier head from '{model_name}': {reason}"
        else:
            message = f"Failed to extract classifier head from '{model_name}'."

        suggestion = (
            "Ensure the model has a standard structure (GAP + FC layers after target layer). "
            "Alternatively, pass classifier_head explicitly to ExpectedGradCAM."
        )

        super().__init__(message, suggestion=suggestion)


class TargetLayerNotFoundError(ArchitectureError):
    """Raised when the target layer cannot be found in the model.

    This error occurs when:
    - The specified layer name doesn't exist in the model
    - Auto-detection fails to find a suitable target layer

    Example:
        >>> raise TargetLayerNotFoundError(
        ...     "layer5",
        ...     available=["layer1", "layer2", "layer3", "layer4"]
        ... )
    """

    def __init__(
        self,
        layer_name: str,
        *,
        available: Sequence[str] | None = None,
    ) -> None:
        """Initialize the exception.

        Args:
            layer_name: Name of the layer that was not found.
            available: List of available layer names in the model.
        """
        self.layer_name = layer_name
        self.available = list(available) if available else []

        message = f"Target layer '{layer_name}' not found in model."
        if self.available:
            available_str = ", ".join(self.available[:10])
            if len(self.available) > 10:
                available_str += f", ... ({len(self.available) - 10} more)"
            suggestion = f"Available layers: {available_str}"
        else:
            suggestion = "Use model.named_modules() to see available layer names."

        super().__init__(message, suggestion=suggestion)
