# import numpy as np
# from utils import isInvertible
# import random
# # from algorithms import (
# #     OffCLUB, OffNCLUB, OffNCLUB_estimate_gamma,
# #     OffDBSCAN_improve, OffXMeans_improve,
# #     LinUCB, LinUCB_IND, Baseline, SCLUB, CLUB
# # )
# from typing import List, Dict, Tuple, Optional
# import os
# from scipy.stats import norm
# from Off_NCLUB_estimate_gamma import OffNCLUB_estimate_gamma


import numpy as np
from .utils import isInvertible
import random
from typing import List, Dict, Tuple, Optional
import os
from scipy.stats import norm
from .Base import LinUCB, LinUCB_IND, Baseline


def compute_mean_and_ci(data: np.ndarray, confidence: float = 0.999) -> Tuple[float, float]:
    mean = np.mean(data)
    se = np.std(data, ddof=1) / np.sqrt(len(data))
    ci = se * norm.ppf((1 + confidence) / 2)
    return mean, ci

def load_regret(file_path: str) -> Optional[float]:
    if not os.path.exists(file_path):
        print(f"Warning: File {file_path} does not exist.")
        return None
    try:
        data = np.load(file_path)
        rewards = data.get('test_rewards', data.get('arr_1'))
        best_rewards = data.get('best_test_rewards', data.get('arr_2'))
        regret = best_rewards - rewards
        mean, CI  = compute_mean_and_ci(regret)
        return np.mean(regret)
    except KeyError as e:
        print(f"Error: Missing key {e} in file {file_path}.")
        return None

def load_regret_with_se(file_path: str) -> Optional[Tuple[float, float]]:
    if not os.path.exists(file_path):
        print(f"Warning: File {file_path} does not exist.")
        return None
    try:
        data = np.load(file_path)
        if "ARMUL" in file_path:
            mean = data.get('average_gap')
            CI = data.get('err')
            return mean,CI
        rewards = data.get('test_rewards', data.get('arr_1'))
        # print(f"rewards: {rewards.shape}")
        best_rewards = data.get('best_test_rewards', data.get('arr_2'))
        regret = best_rewards - rewards

        mean, CI  = compute_mean_and_ci(regret)
        return mean, CI
    except KeyError as e:
        print(f"Error: Missing key {e} in file {file_path}.")
        return None


def get_mean_regrets(seed_list: List[int], methods: Dict[str, Dict], is_empirical: int, dataset: str,
                     T_values: List[int], nu: int, d: int, m: int, L: int, pj: int, choose_gamma_alpha: float,
                     alpha: float, gamma_list: List[float],offline_learn_method = 'random', debug = False) -> Dict[str, List[Tuple[int, float, float]]]:
    results = {method: [] for method in methods}
    uniforms = ['uniform', 'half', 'arbitrary']
    for T_index, T in enumerate(T_values):
        # print(f"Processing T = {T}...")
        for method, config in methods.items():
            regrets = []
            CIs = []
            for seed in seed_list:
                filename = build_method_filename(is_empirical, dataset, config, uniforms[pj], choose_gamma_alpha, alpha,
                                        gamma_list[T_index], T, nu, d, m, L, seed, offline_learn_method, debug = debug)
                result = load_regret_with_se(filename)
                
                if result is not None:
                    regret, CI = result
                    regrets.append(regret)
                    CIs.append(CI)
            if regrets:
                mean = np.mean(np.array(regrets))
                ci = np.mean(np.array(CIs))
                results[method].append((T, mean, ci))
            else:
                print(f"No valid data for method {method} at T={T}.")
    return results


def get_gamma_regrets(seed_list: List[int], methods: Dict[str, Dict], is_empirical: int, dataset: str,
                     T: int, nu: int, d: int, m: int, L: int, pj: int, choose_gamma_alpha: float,
                     alpha: float, gamma_list: List[float]) -> Dict[str, List[Tuple[int, float, float]]]:
    results = {method: [] for method in methods}
    uniforms = ['uniform', 'half', 'arbitrary']
    for method, config in methods.items():
        regrets = []
        if config['gamma_required']:
            for gamma in gamma_list:
                tmp_regrets = 0
                for seed in seed_list:
                    filename = build_method_filename(is_empirical, dataset, config, uniforms[pj], choose_gamma_alpha, alpha, gamma, T, nu, d, m, L, seed)
                    regret = load_regret(filename)
                    if regret is not None:
                        tmp_regrets += regret
                regrets.append(tmp_regrets/len(seed_list))
        else:
            tmp_regrets = 0
            for seed in seed_list:
                filename = build_method_filename(is_empirical, dataset, config, uniforms[pj], choose_gamma_alpha, alpha, gamma, T, nu,
                                          d, m, L, seed)
                regret = load_regret(filename)
                if regret is not None:
                    tmp_regrets += regret
            regrets.append(tmp_regrets/len(seed_list))

        results[method] = regrets
    return results

# def build_filename(output_folder, dataset, method, uniforms_str, choose_gamma_alpha, alpha, gamma, T, nu, d, m, L, seed, offline_learn_method="random"):
#     if T == 0:
#         return f'{output_folder}/{dataset}_Baseline_{uniforms_str}_random_T100000nu{nu}d{d}m{m}L{L}_{seed}.npz'
#     prefix = method['prefix']
#     alpha_varying = method['alpha_varying']
#     gamma_required = method['gamma_required']
#     choose_gamma_alpha_varying = method['choose_gamma_alpha_required']
#     empirical = method['empirical']
#     file_name = f'{output_folder}/{dataset}_{prefix}_{uniforms_str}'

#     if empirical and dataset != "synthetic":
#         file_name += f'empirical'
#     file_name += f'_{offline_learn_method}'

#     if choose_gamma_alpha_varying:
#         file_name += f'_choose_gamma_alpha{choose_gamma_alpha:.3f}'
#     else:
#         pass
#     if alpha_varying:
#         file_name += f'_alpha{alpha:.2f}'
#     else:
#         pass
#     if gamma_required:
#         file_name += f'_gamma{gamma:.2f}'
#     else:
#         pass
#     file_name += f'_T{T}nu{nu}d{d}m{m}L{L}_{seed}.npz'
#     return file_name

def build_method_filename(is_empirical, dataset, method, uniforms_str, choose_gamma_alpha, alpha, gamma, T, nu, d, m, L, seed, offline_learn_method="random", debug = False):
    # print(f"method: {method}")
    if is_empirical:
        output_folder = 'artifacts/output_data_real'
    else:
        output_folder = 'artifacts/output_data'
    if debug:
        output_folder = 'artifacts/output_data_debug'
        
    if method["prefix"] in ['sclub']:
        output_folder = 'artifacts/output_data_debug'
    if not os.path.exists(output_folder):
        os.makedirs(output_folder)
    if T == 0:
        return f'{output_folder}/{dataset}_Baseline_{uniforms_str}_random_T100000nu{nu}d{d}m{m}L{L}_1.npz'
    prefix = method['prefix']
    alpha_varying = method['alpha_varying']
    gamma_required = method['gamma_required']

    choose_gamma_alpha_varying = method['choose_gamma_alpha_required']
    empirical = method['empirical']
    file_name = f'{output_folder}/{dataset}_{prefix}_{uniforms_str}'

    if empirical and dataset != "synthetic":
        file_name += f'empirical'

    file_name += f'_{offline_learn_method}'
    if choose_gamma_alpha_varying:
        file_name += f'_choose_gamma_alpha{choose_gamma_alpha:.3f}'
    else:
        pass
    if alpha_varying:
        file_name += f'_alpha{alpha:.2f}'
    else:
        pass
    if gamma_required:
        file_name += f'_gamma{gamma:.2f}'
    else:
        pass
    file_name += f'_T{T}nu{nu}d{d}m{m}L{L}_{seed}.npz'
    return file_name

def compute_max_value(C1, lambda_a, d, delta, gamma, U):
    term1 = (64 / (C1 ** 2 * lambda_a ** 2)) * np.log(
        32 * d / (C1 ** 2 * lambda_a ** 2 * delta)
    )
    term2 = (256 * d / ((1 - C1) * gamma ** 2 * lambda_a)) * np.log(2 * U / delta)
    return max(term1, term2)

def create_algorithm_instance(method_name, nu, d, T, L, choose_gamma_alpha, alpha, gamma):
    if  method_name == 'OffDBSCAN_improve':
        from ..algorithms.clustering.DBSCAN_improve import OffDBSCAN_improve
        return OffDBSCAN_improve(nu=nu, d=d, T=T, ni=L, eps=gamma * 0.5, min_samples=5)
    
    elif method_name == 'OffXMeans_improve':
        from ..algorithms.clustering.XMeans_improve import OffXMeans_improve
        return OffXMeans_improve(nu=nu, d=d, T=T, ni=L)

    elif method_name == 'CLUB':
        from ..algorithms.bandits.CLUB import CLUB
        return CLUB(nu=nu, d=d, T=T, ni=L, alpha=alpha, edge_probability=1)

    elif method_name == 'OffCLUB':
        from ..algorithms.bandits.Off_CLUB import OffCLUB
        return OffCLUB(nu=nu, d=d, T=T, ni=L, alpha=alpha, delta=0.01, C1=0.5, edge_probability=1)

    elif method_name == 'LinUCB':
        return LinUCB(nu=nu, d=d, T=T, ni=L)

    elif method_name == 'LinUCB_IND':
        return LinUCB_IND(nu=nu, d=d, T=T, ni=L)

    elif method_name == 'SCLUB':
        from ..algorithms.bandits.Off_SCLUB import SCLUB
        return SCLUB(nu=nu, d=d, T=T, ni=L)

    elif method_name == 'OffNCLUB':
        from ..algorithms.bandits.Off_NCLUB import OffNCLUB
        return OffNCLUB(nu=nu, d=d, T=T, ni=L, alpha=alpha, delta=0.01, C1=0.5, gamma=gamma)

    elif method_name == 'OffNCLUB_estimate_gamma_Pessimistic':
        from ..algorithms.bandits.Off_NCLUB_estimate_gamma import OffNCLUB_estimate_gamma
        return OffNCLUB_estimate_gamma(
            nu=nu, d=d, T=T, ni=L, choose_gamma_alpha= choose_gamma_alpha,
            alpha=alpha, delta=0.01, C1=0.5, model="Pessimistic"
        )

    elif method_name == 'OffNCLUB_estimate_gamma_Optimistic':
        from ..algorithms.bandits.Off_NCLUB_estimate_gamma import OffNCLUB_estimate_gamma
        return OffNCLUB_estimate_gamma(
            nu=nu, d=d, T=T, ni=L, choose_gamma_alpha= choose_gamma_alpha,
            alpha=alpha, delta=0.01, C1=0.5, model="Optimistic"
        )
    
    elif method_name == 'OffNCUCB_estimate_gamma_Pessimistic':
        from ..algorithms.bandits.Off_NCUCB_estimate_gamma import OffNCUCB_estimate_gamma
        return OffNCUCB_estimate_gamma(
            nu=nu, d=d, T=T, ni=L, choose_gamma_alpha= choose_gamma_alpha,
            alpha=alpha, delta=0.01, C1=0.5, model="Pessimistic"
        )
    
    elif method_name == 'OffNCUCB_estimate_gamma_Optimistic':
        from ..algorithms.bandits.Off_NCUCB_estimate_gamma import OffNCUCB_estimate_gamma
        return OffNCUCB_estimate_gamma(
            nu=nu, d=d, T=T, ni=L, choose_gamma_alpha= choose_gamma_alpha,
            alpha=alpha, delta=0.01, C1=0.5, model="Optimistic"
        )

    elif method_name == 'Baseline':
        return Baseline(nu=nu, d=d, T=T, ni=L)
    else:
        raise ValueError(f"Method {method_name} not recognized.")
