import yaml
import os

from dataclasses import dataclass, fields


@dataclass(frozen=True)
class Config:
    """ Config class encapsulates configuration parameters for the endogenous contexts simulation study.
    Attributes:
        N (int): Number of nodes.
        density (float): Graph sparsity.
        max_lag (int): Maximum lag in the time series.
        pc_alpha (float): Significance level for PC algorithm.
        sample_size (int): Number of samples in total.
        reg_children_known (str): Whether context-system links are known (link assumptions) or not.
        nb_changed_links (int): Number of changed links between contexts.
        nb_regimes (int): Number of contexts.
        nb_repeats (int): Number of repeats per configuration.
        cycles_only (bool): Whether to include only graphs with cycles.
        remove_only (bool): Whether to only remove links.
        use_cmiknnmixed (bool): Whether to use CMIkNN mixed method (not supported/tested here).
        imbalance_factor (float): Factor controlling context imbalance.
        contemp_fraction (float): Fraction of contemporaneous links.
        regime_autocorr (float): Auto-lag coefficient for the context variable.
        endo_regime (bool): Whether the context is endogenous.
        contemp_context (bool): Whether context is contemporaneous.
        save_folder (str): Folder to save results.
    Methods:
        from_tuple(cls, t): Create a Config instance from a legacy 17-element tuple.
        as_tuple_with(sample_size=None, regime_children_value=None, autocorr=None): 
            Return a legacy 16-field tuple with optional overrides.
        items(): Yield (field_name, value) pairs for all fields.
        __getitem__(index): Get field value by index.
        check_type(key, value): Type-check and convert value for a given key."""
    N: int
    density: float
    max_lag: int
    pc_alpha: float
    sample_size: int
    reg_children_known: str 
    nb_changed_links: int
    nb_regimes: int
    nb_repeats: int
    cycles_only: bool
    remove_only: bool
    use_cmiknnmixed: bool
    imbalance_factor: float
    contemp_fraction: float
    regime_autocorr: float
    endo_regime: bool
    contemp_context: bool
    save_folder: str

    @classmethod
    def from_tuple(cls, t):
        """Create a Config from the legacy 17-element tuple."""
        return cls(*t)

    def as_tuple_with(self, *, sample_size=None, regime_children_value=None, autocorr=None):
        """Return the legacy 16-field tuple (without save_folder) with optional overrides
        for fields that were varied in filenames.
        """
        return (
            self.N,
            self.density,
            self.max_lag,
            self.pc_alpha,
            self.sample_size if sample_size is None else sample_size,
            self.reg_children_known if regime_children_value is None else regime_children_value,
            self.nb_changed_links,
            self.nb_regimes,
            self.nb_repeats,
            self.cycles_only,
            self.remove_only,
            self.use_cmiknnmixed,
            self.imbalance_factor,
            self.contemp_fraction,
            self.regime_autocorr if autocorr is None else autocorr,
            self.endo_regime,
            self.contemp_context
        )
    
    def items(self):
        for field in fields(self):
            yield field.name, getattr(self, field.name)

    def __getitem__(self, index: int):
        flds = list(self.__dataclass_fields__)   # field names
        n = len(flds)
        if not -n <= index < n:
            raise IndexError("Index out of range")
        return getattr(self, flds[index])
    
    def check_type(self, key, value):
        if key == 'sample_size':
            return int(value)
        if key == 'regime_autocorr':
            return float(value)
        

def generate_name_from_params(params):
    """
    Generates a hyphen-separated string from the given parameters except the last one.

    Parameters:
    - params (Config): A Config object containing parameters.

    Returns:
    - str: A string representation of the parameters separated by hyphens.
    """
    # Join all parameters except the last one into a string separated by hyphens
    params = [str(param) for param in params]
    params_str = "-".join(str(p) for p in params[:-1])

    return params_str

def generate_string_from_params(params):
    """
    Generates a hyphen-separated string from the given parameters.

    Parameters:
    - params (Config): A Config object containing parameters.

    Returns:
    - str: A string representation of the parameters separated by hyphens.
    """
    # Join all parameters into a string separated by hyphens
    p = []
    for key, value in params.items():
        p.append(str(value))
    params_str = '-'.join(p)

    return params_str
    

def load_configurations(path):
    """
    Loads configurations from a YAML file.

    Parameters:
    - path (str): The file path to the YAML configuration file.

    Returns:
    - dict: The configuration dictionary.
    """
    # Open the YAML configuration file and load it
    with open(path, 'r') as file:
        config = yaml.safe_load(file)
    return config

def generate_configurations(config_path):
    """
    Generates a list of configurations based on the parameters specified in a YAML configuration file.

    Parameters:
    - config_path (str): The file path to the YAML configuration file.

    Returns:
    - str: The directory path where results will be stored.
    - list: A list of configuration tuples.
    """
    # Load configurations from the YAML file
    config = load_configurations(config_path)
    configurations = []

    # Compute the base path relative to this script's location
    base_path = os.path.dirname(os.path.abspath(__file__))
     # Compute the absolute path to the directory where results will be stored
    data_directory = os.path.abspath(os.path.join(base_path, '..', config['results_folder']))

    # Generate configurations by iterating over all combinations of parameters
    for sample_size in config['sample_size']:
        for N in config['N_values']:
            for density in config['densities']:
                for max_lag in config['max_lags']:
                    for pc_alpha in config['pc_alphas']:
                        for regime_child_known in config['regime_children_known']:
                            for nb_changed_link in config['nb_changed_links']:
                                for nb_regime in config['nb_regimes']:
                                    for cycle_only in config['cycles_only']:
                                        for remove in config['remove_only']:
                                            for cmiknnmixed in config['use_cmiknnmixed']:
                                                for imbalance_factor in config['imbalance_factor']:
                                                    for contemp_fraction in config['contemp_fraction']:
                                                        for regime_autocorr in config['regime_autocorr']:
                                                            for endo_regime in config['endo_regime']:
                                                                for contemp_context in config['contemp_context']:
                                                                    # Prepare the configuration tuple with all parameters
                                                                    config_tuple = Config(N, density, max_lag, pc_alpha, sample_size, regime_child_known,
                                                                                    nb_changed_link, nb_regime, config['nb_repeats'], cycle_only,
                                                                                    remove, cmiknnmixed, imbalance_factor, contemp_fraction, 
                                                                                    regime_autocorr, bool(endo_regime), bool(contemp_context),
                                                                                    data_directory)
                                                                    configurations.append(config_tuple)
    # Return the directory and the list of configurations
    return data_directory, configurations
