"""Global experiment config state."""

from dataclasses import dataclass
from typing import Callable, Dict, Type


@dataclass
class ExperimentConfig:
    type: str
    seed: int
    predictor_config: dict
    dataset_config: dict
    calibrator_config: dict

    @property
    def id(self) -> str:
        import hashlib
        import json
        from dataclasses import asdict

        config_dict = asdict(self)
        config_str = json.dumps(config_dict, indent=2, sort_keys=True)
        return hashlib.md5(config_str.encode()).hexdigest()

    def to_dict(self) -> dict:
        from dataclasses import asdict

        return asdict(self)

    @classmethod
    def from_dict(cls, config_dict: dict) -> "ExperimentConfig":
        return cls(
            type=config_dict["type"],
            seed=config_dict["seed"],
            predictor_config=config_dict.get("predictor_config", {}),
            dataset_config=config_dict.get("dataset_config", {}),
            calibrator_config=config_dict.get("calibrator_config", {}),
        )


ExperimentRunFunction = Callable[[ExperimentConfig], None]


class ExperimentRegistry:
    _configs: Dict[str, Type[ExperimentConfig]] = {}
    _runners: Dict[str, ExperimentRunFunction] = {}

    @classmethod
    def register(
        cls,
        experiment_type: str,
        config_class: Type[ExperimentConfig],
        run_function: ExperimentRunFunction,
    ):
        cls._configs[experiment_type] = config_class
        cls._runners[experiment_type] = run_function

    @classmethod
    def get_config_class(cls, experiment_type: str) -> Type[ExperimentConfig]:
        if experiment_type not in cls._configs:
            raise ValueError(
                f"Experiment type '{experiment_type}' not found. "
                f"Available types: {list(cls._configs.keys())}"
            )
        return cls._configs[experiment_type]

    @classmethod
    def get_run_function(cls, experiment_type: str) -> ExperimentRunFunction:
        if experiment_type not in cls._runners:
            raise ValueError(
                f"Experiment type '{experiment_type}' not found. "
                f"Available types: {list(cls._runners.keys())}"
            )
        return cls._runners[experiment_type]

    @classmethod
    def get_available_types(cls) -> list[str]:
        return list(cls._configs.keys())


def register_experiment(
    experiment_type: str,
    config_class: Type[ExperimentConfig],
    run_function: ExperimentRunFunction,
):
    ExperimentRegistry.register(experiment_type, config_class, run_function)


def run_experiment(config_dict: dict, redo: bool = False, debug: bool = False) -> bool:
    """Run an experiment from a config dictionary.

    Args:
        config_dict: Experiment configuration dictionary.
        redo: If True, rerun even if already completed.
        debug: If True, enable debug logging.

    Returns:
        True if experiment ran successfully, False if skipped or errored.
    """
    import shutil
    from pathlib import Path

    from ..utils import (
        get_logger,
        set_debug_mode,
        set_experiment_config,
        set_global_seed,
    )

    set_debug_mode(debug)

    if "type" not in config_dict:
        print("Error: No experiment type provided in the config.")
        print(f"Available types: {ExperimentRegistry.get_available_types()}")
        return False

    experiment_type = config_dict["type"]

    try:
        config_class = ExperimentRegistry.get_config_class(experiment_type)
        run_function = ExperimentRegistry.get_run_function(experiment_type)
    except ValueError as e:
        print(f"Error: {e}")
        return False

    config = config_class.from_dict(config_dict)
    set_global_seed(config.seed)
    set_experiment_config(config)

    log_dir = Path(".logs") / config.id
    done_file = log_dir / "done"

    if done_file.exists() and not redo:
        print(f"Experiment {config.type} (id: {config.id}) has already been run.")
        print("Use --redo to rerun the experiment.")
        return False

    # Clean previous logs if rerunning
    if redo and log_dir.exists():
        shutil.rmtree(log_dir)

    logger = get_logger(__name__)
    logger.info(f"Experiment ID: {config.id}")

    run_function(config)

    logger.log_done()
    return True


__all__ = [
    "ExperimentConfig",
    "ExperimentRunFunction",
    "ExperimentRegistry",
    "register_experiment",
    "run_experiment",
]


def _register():
    from . import dummy


_register()
