"""Configuration-related exceptions."""

from __future__ import annotations

from typing import Any

from expected_gradcam.exceptions.base import ExpectedGradCAMError


class ConfigurationError(ExpectedGradCAMError):
    """Base exception for configuration errors."""

    pass


class InvalidParameterError(ConfigurationError):
    """Base exception for invalid parameter values."""

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

        Args:
            param_name: Name of the invalid parameter.
            value: The invalid value that was provided.
            reason: Reason why the value is invalid.
            suggestion: How to fix the issue.
        """
        self.param_name = param_name
        self.value = value
        self.reason = reason

        if reason:
            message = f"Invalid value for '{param_name}': {value!r}. {reason}"
        else:
            message = f"Invalid value for '{param_name}': {value!r}"

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


class InvalidParameterRangeError(InvalidParameterError):
    """Raised when a parameter value is outside its valid range.

    Example:
        >>> raise InvalidParameterRangeError(
        ...     "M", -5, bounds=(1, 10000)
        ... )
    """

    def __init__(
        self,
        param_name: str,
        value: Any,
        *,
        bounds: tuple[Any, Any],
    ) -> None:
        """Initialize the exception.

        Args:
            param_name: Name of the parameter.
            value: The invalid value.
            bounds: (min, max) valid bounds for the parameter.
        """
        self.bounds = bounds
        min_val, max_val = bounds

        super().__init__(
            param_name,
            value,
            reason=f"Must be in range [{min_val}, {max_val}]",
            suggestion=f"Use a value between {min_val} and {max_val}.",
        )


class InvalidParameterChoiceError(InvalidParameterError):
    """Raised when a parameter value is not one of the allowed choices.

    Example:
        >>> raise InvalidParameterChoiceError(
        ...     "solver_method", "invalid",
        ...     choices=("pinv", "adaptive_reg", "subspace")
        ... )
    """

    def __init__(
        self,
        param_name: str,
        value: Any,
        *,
        choices: tuple[Any, ...],
    ) -> None:
        """Initialize the exception.

        Args:
            param_name: Name of the parameter.
            value: The invalid value.
            choices: Tuple of valid choices.
        """
        self.choices = choices
        choices_str = ", ".join(repr(c) for c in choices)

        super().__init__(
            param_name,
            value,
            reason=f"Must be one of: {choices_str}",
            suggestion=f"Choose from: {choices_str}",
        )


class MissingParameterError(ConfigurationError):
    """Raised when a required parameter is not provided.

    Example:
        >>> raise MissingParameterError(
        ...     "baseline_dataset",
        ...     context="ExpectedGradCAM requires a baseline dataset"
        ... )
    """

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

        Args:
            param_name: Name of the missing parameter.
            context: Additional context about why the parameter is needed.
        """
        self.param_name = param_name
        self.context = context

        if context:
            message = f"Required parameter '{param_name}' is missing. {context}"
        else:
            message = f"Required parameter '{param_name}' is missing."

        suggestion = f"Provide a value for '{param_name}'."

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