"""Base configuration dataclasses."""

# pylint: disable=all
# Temporarily disable all pylint checkers during AST traversal to prevent crash.
# The imports checker crashes when resolving fwh_core package imports due to a bug
# in pylint/astroid: https://github.com/pylint-dev/pylint/issues/10185
# pylint: enable=all
# Re-enable all pylint checkers for the checking phase. This allows other checks
# (code quality, style, undefined names, etc.) to run normally while bypassing
# the problematic imports checker that would crash during AST traversal.

from dataclasses import dataclass

from omegaconf import DictConfig

from fwh_core.exceptions import ConfigValidationError
from fwh_core.logger import FWH_CORE_LOGGER
from fwh_core.structured_configs.mlflow import MLFlowConfig, validate_mlflow_config
from fwh_core.structured_configs.validation import (
    validate_mapping,
    validate_non_negative_int,
    validate_nonempty_str,
    validate_path,
)
from fwh_core.utils.config_utils import dynamic_resolve


@dataclass
class BaseConfig:
    """Base configuration for all components."""

    device: str | None = None
    seed: int | None = None
    tags: dict[str, str] | None = None
    logging_config_path: str | None = None
    mlflow: MLFlowConfig | None = None


def validate_base_config(cfg: DictConfig) -> None:
    """Validate a BaseConfig.

    Args:
        cfg: A DictConfig with seed, tags, and mlflow fields (from Hydra).
    """
    device = cfg.get("device")
    seed = cfg.get("seed")
    tags = cfg.get("tags")
    logging_config_path = cfg.get("logging_config_path")
    mlflow = cfg.get("mlflow")

    validate_nonempty_str(device, "BaseConfig.device", is_none_allowed=True)
    allowed_devices = ("auto", "cpu", "gpu", "cuda")
    if device is not None and device not in allowed_devices:
        raise ConfigValidationError(f"BaseConfig.device must be one of: {allowed_devices}")
    validate_non_negative_int(seed, "BaseConfig.seed", is_none_allowed=True)
    validate_mapping(tags, "BaseConfig.tags", key_type=str, value_type=str, is_none_allowed=True)
    validate_path(logging_config_path, "BaseConfig.logging_config_path", is_none_allowed=True, must_exist=True)
    if mlflow is not None:
        if not isinstance(mlflow, DictConfig):
            raise ConfigValidationError("BaseConfig.mlflow must be a MLFlowConfig")
        validate_mlflow_config(mlflow)


@dynamic_resolve
def resolve_base_config(cfg: DictConfig, *, strict: bool, seed: int | None = None, device: str | None = None) -> None:
    """Resolve the BaseConfig by setting default values and logging mismatches.

    This function sets default seed and strict tag values if not present in the config.
    If values are already set but don't match the provided parameters, it logs
    a warning and overrides them.

    Args:
        cfg: A DictConfig with seed and tags fields (from Hydra).
        strict: Whether strict mode is enabled. Used to set tags.strict.
        seed: The random seed to use. If None, defaults to 42 when config has no seed.
        device: The device to use. If None, defaults to "auto" when config has no device.
    """
    device_tag = cfg.get("device")
    if device_tag is None:
        cfg.device = device or "auto"
    elif device and device_tag != device:
        FWH_CORE_LOGGER.warning(
            "Device tag set to '%s', but device is '%s'. Overriding device tag.", device_tag, device
        )
        cfg.device = device

    seed_tag = cfg.get("seed")
    if seed_tag is None:
        cfg.seed = seed if seed is not None else 42
    elif seed is not None and seed_tag != seed:
        FWH_CORE_LOGGER.warning("Seed tag set to '%s', but seed is '%s'. Overriding seed tag.", seed_tag, seed)
        cfg.seed = seed

    if cfg.get("tags") is None:
        cfg.tags = DictConfig({"strict": str(strict).lower()})
    else:
        tags: DictConfig = cfg.get("tags")
        strict_value = str(strict).lower()
        if tags.get("strict") is None:
            tags.strict = strict_value
        else:
            strict_tag: str = tags.get("strict")
            if strict_tag.lower() != strict_value:
                FWH_CORE_LOGGER.warning(
                    "Strict tag set to '%s', but strict mode is '%s'. Overriding strict tag.", strict_tag, strict_value
                )
                tags.strict = strict_value
