#!/usr/bin/env python3
"""
This is the CLI script that is executed when the user runs the `fusion_bench` command.
The script is responsible for parsing the command-line arguments, loading the configuration file, and running the fusion algorithm.
"""
import logging
from typing import TYPE_CHECKING

import hydra
from omegaconf import DictConfig, OmegaConf

from fusion_bench.utils import instantiate
from fusion_bench.utils.hydra_utils import get_default_config_path

if TYPE_CHECKING:
    from fusion_bench.programs import BaseHydraProgram

log = logging.getLogger(__name__)


def main(cfg: DictConfig) -> None:
    """
    Main entry point for the FusionBench command-line interface.

    This function serves as the primary entry point for the `fusion_bench` CLI command.
    It is decorated with Hydra's main decorator to handle configuration management,
    command-line argument parsing, and configuration file loading.

    The function performs the following operations:
    1. Resolves any interpolations in the configuration using OmegaConf
    2. Instantiates the appropriate program class based on the configuration
    3. Executes the program's run method to perform the fusion task

    Args:
        cfg (DictConfig): The Hydra configuration object containing all settings
            for the fusion task. This includes method configuration, model pool
            configuration, task pool configuration, and other runtime parameters.
            The configuration is automatically loaded by Hydra from the specified
            config files and command-line overrides.

    Returns:
        None: This function doesn't return a value but executes the fusion
            program which may save results, log outputs, or perform other
            side effects as configured.

    Example:
        This function is typically called automatically when running:
        ```bash
        fusion_bench method=... modelpool=... taskpool=...
        ```

        The Hydra decorator handles parsing these command-line arguments and
        loading the corresponding configuration files to populate the cfg parameter.
    """
    OmegaConf.resolve(cfg)
    program: "BaseHydraProgram" = instantiate(cfg)

    # Validate that instantiation succeeded and returned an object with 'run' method
    if not hasattr(program, "run") or not callable(getattr(program, "run")):
        err_msg = (
            f"Expected an object with a callable 'run' method, but got {type(program).__name__}. "
            "Ensure that the configuration specifies a concrete program class with '_target_'."
        )
        if "_target_" not in cfg:
            err_msg += "\nThe '_target_' field is missing from the root configuration."
        else:
            err_msg += f"\nFound '_target_': {cfg._target_}"
        err_msg += f"\n\nConfiguration content:\n{cfg}"
        raise TypeError(err_msg)

    try:
        program_result = program.run()
        return program_result
    except Exception as e:
        # Log the exception before exiting
        log.error(e, exc_info=True)
        raise e


@hydra.main(
    config_path=get_default_config_path(),
    config_name="fabric_model_fusion",
    version_base=None,
)
def _hydra_main(cfg: DictConfig) -> None:
    main(cfg)


if __name__ == "__main__":
    _hydra_main()
