"""
Unified API for CoBET methods.
"""
import pandas as pd
from .methods import CoBET, dCoBET, wa_dCoBET
from .simulation import run_power_analysis, export_results, print_summary


def run_test(
    method,
    n_list,
    theta=2,
    K=4,
    d=2,
    b_config_by_n=None,
    transforms=("trigU", "expquad", "linear", "logquad"),
    R_eval=1000,
    alpha=0.05,
    seed=123,
    output=None,
    unbiased=True,
    print_results=True,
    **method_kwargs
):
    """
    Unified entry point for running CoBET method simulations.
    
    This function orchestrates power analysis and Type I error simulations
    for any CoBET method (CoBET, dCoBET, or wa_dCoBET).
    
    Parameters
    ----------
    method : str or BaseCoBET
        Method to use: 'cobet', 'dcobet', 'wa_dcobet', or method instance
    n_list : list of int
        Sample sizes to test, e.g., [250, 500, 1000]
    theta : float, default=2
        Clayton copula parameter for dependence structure
    K : int, default=4
        Dyadic depth (number of bits in binary expansion)
    d : int, default=2
        Dimension of data
    b_config_by_n : dict
        Mapping from sample size to transform-specific b values.
        Example: {250: {"trigU": [0.05, 0.1], "linear": [0.1, 0.2]}, ...}
    transforms : tuple of str, default=("trigU", "expquad", "linear", "logquad")
        Transform families to test
    R_eval : int, default=1000
        Number of Monte Carlo replications per setting
    alpha : float, default=0.05
        Significance level
    seed : int, default=123
        Random seed for reproducibility
    output : str, optional
        Output file path (.xlsx or .csv). If None, generates timestamped filename.
    unbiased : bool, default=True
        Use unbiased variance estimator
    print_results : bool, default=True
        Print summary of results to console
    **method_kwargs
        Additional keyword arguments passed to method constructor
        (e.g., n_folds=10 for wa_dCoBET, reuse_J=True for dCoBET)
    
    Returns
    -------
    results : pd.DataFrame
        Simulation results with columns:
        - method: Method name
        - n: Sample size
        - transform: Transform family
        - b: Dependence parameter
        - metric: 'typeI' or 'power'
        - value: Rejection rate
        - d, K, theta, alpha, R_eval, seed: Configuration parameters
        - Z_mean, Z_std: Test statistic summaries
    
    Examples
    --------
    Run CoBET (identity weights):
    
    >>> from cobet import run_test
    >>> 
    >>> b_config = {
    ...     250: {"trigU": [0.05, 0.1], "linear": [0.1, 0.2]},
    ...     500: {"trigU": [0.03, 0.05], "linear": [0.05, 0.1]}
    ... }
    >>> 
    >>> results = run_test(
    ...     method='cobet',
    ...     n_list=[250, 500],
    ...     K=4, d=5, theta=2,
    ...     b_config_by_n=b_config,
    ...     R_eval=500,
    ...     output='cobet_results.xlsx'
    ... )
    
    Run dCoBET (J weights):
    
    >>> results = run_test(
    ...     method='dcobet',
    ...     n_list=[250, 500],
    ...     K=4, d=5,
    ...     b_config_by_n=b_config,
    ...     R_eval=500,
    ...     reuse_J=True  # method-specific parameter
    ... )
    
    Run wa_dCoBET (adaptive weights):
    
    >>> results = run_test(
    ...     method='wa_dcobet',
    ...     n_list=[250, 500],
    ...     K=4, d=5,
    ...     b_config_by_n=b_config,
    ...     R_eval=500,
    ...     n_folds=10  # method-specific parameter
    ... )
    """
    # Validate inputs
    if b_config_by_n is None:
        raise ValueError("b_config_by_n must be provided")
    
    for n in n_list:
        if n not in b_config_by_n:
            raise ValueError(f"Missing configuration for n={n} in b_config_by_n")
    
    # Create method instance if string provided
    if isinstance(method, str):
        method_map = {
            'cobet': CoBET,
            'dcobet': dCoBET,
            'wa_dcobet': wa_dCoBET,
        }
        method_class = method_map.get(method.lower())
        if method_class is None:
            raise ValueError(
                f"Unknown method: {method}. "
                f"Available: {list(method_map.keys())}"
            )
        
        # Create instance with provided parameters
        method_instance = method_class(
            K=K, d=d, theta=theta, alpha=alpha, 
            seed=seed, unbiased=unbiased,
            **method_kwargs
        )
        method_name = method.lower()
    else:
        # Method instance provided directly
        method_instance = method
        method_name = method_instance.method_name.lower()
    
    print(f"\n{'='*60}")
    print(f"Running {method_instance.method_name} simulations")
    print(f"{'='*60}")
    print(f"Configuration:")
    print(f"  Sample sizes: {n_list}")
    print(f"  Dimension: {d}")
    print(f"  Dyadic depth: {K}")
    print(f"  Theta: {theta}")
    print(f"  Alpha: {alpha}")
    print(f"  Replications: {R_eval}")
    print(f"  Transforms: {transforms}")
    print(f"{'='*60}\n")
    
    # Run power analysis
    results = run_power_analysis(
        method=method_instance,
        n_list=n_list,
        theta=theta,
        K=K,
        d=d,
        alpha=alpha,
        R_eval=R_eval,
        seed=seed,
        transforms=transforms,
        b_config_by_n=b_config_by_n,
        unbiased=unbiased
    )
    
    # Export results
    if output is not None:
        export_results(results, output, method_name=method_name)
    
    # Print summary
    if print_results:
        print_summary(results)
    
    return results


# Convenience functions for each method
def run_cobet(n_list, b_config_by_n, theta=2, K=4, d=2, R_eval=1000, 
              alpha=0.05, seed=123, output=None, **kwargs):
    """Run CoBET (identity weights) simulations."""
    return run_test(
        method='cobet',
        n_list=n_list,
        theta=theta,
        K=K,
        d=d,
        b_config_by_n=b_config_by_n,
        R_eval=R_eval,
        alpha=alpha,
        seed=seed,
        output=output,
        **kwargs
    )


def run_dcobet(n_list, b_config_by_n, theta=2, K=4, d=2, R_eval=1000,
               alpha=0.05, seed=123, output=None, reuse_J=True, **kwargs):
    """Run dCoBET (J weights) simulations."""
    return run_test(
        method='dcobet',
        n_list=n_list,
        theta=theta,
        K=K,
        d=d,
        b_config_by_n=b_config_by_n,
        R_eval=R_eval,
        alpha=alpha,
        seed=seed,
        output=output,
        reuse_J=reuse_J,
        **kwargs
    )


def run_wa_dcobet(n_list, b_config_by_n, theta=2, K=4, d=2, R_eval=1000,
                  alpha=0.05, seed=123, output=None, n_folds=10, 
                  reuse_J=True, **kwargs):
    """Run wa_dCoBET (adaptive weights) simulations."""
    return run_test(
        method='wa_dcobet',
        n_list=n_list,
        theta=theta,
        K=K,
        d=d,
        b_config_by_n=b_config_by_n,
        R_eval=R_eval,
        alpha=alpha,
        seed=seed,
        output=output,
        n_folds=n_folds,
        reuse_J=reuse_J,
        **kwargs
    )
