import numpy as np
import torch
import os
import sys
import argparse
import rpy2.robjects as ro

sys.path.append(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
from CoxCP import CoxCP
from env import *

#model_list = ['linear', 'loglinear', 'PH']
model_list = ['PH']
cdf_type_list = ['gaussian', 'MoU']
context_type_list = ['gaussian', 'uniform', 'binary']

algo_name = 'CoxCP'

l0_set = [16, 32, 64, 128, 256]
ngrid = 100

if __name__=='__main__':
    ro.r('install.packages("icenReg")')
    parser = argparse.ArgumentParser()
    parser.add_argument('--cuda', type=str, default='0')
    parser.add_argument('--search', action='store_true')
    args = parser.parse_args()

    os.environ["CUDA_VISIBLE_DEVICES"]= args.cuda

    device = "cuda" if torch.cuda.is_available() else "cpu"
    env_generator = torch.Generator(device=device)

    for model in model_list:
        for cdf_type in cdf_type_list:
            for context_type in context_type_list:

                d = 5
                T = 2000
                rep = 5
                env_seed = algo_seed = 1234

                basedir = f'search/sim_d={d}_T={T}/model={model}/cdf={cdf_type}/context={context_type}/{algo_name}'
                if not os.path.exists(basedir):
                    os.makedirs(basedir)

                if context_type=='gaussian':
                    context_dist = GaussianContext(d=d, sigma=1, generator=env_generator, device=device)
                elif context_type=='uniform':
                    context_dist = UniformContext(d=d, device=device, generator=env_generator)
                elif context_type=='binary':
                    context_dist = BinaryContext(d=d, device=device, generator=env_generator)
                else:
                    raise NotImplementedError
                
                if cdf_type=='gaussian':
                    cdf = gaussian_cdf
                elif cdf_type=='MoU':
                    cdf = MoU_cdf
                else:
                    raise NotImplementedError

                if model=='linear':
                    valuation_model = LinearModel(d=d, cdf=cdf, device=device)
                elif model=='loglinear':
                    valuation_model = LogLinearModel(d=d, cdf=cdf, device=device)
                elif model=='PH':
                    valuation_model = PHModel(d=d, cdf=cdf, device=device)
                else:
                    raise NotImplementedError
                env = Env(generator=env_generator, context_dist=context_dist, valuation_model=valuation_model)
                
                if args.search:
                    for l0 in l0_set:
                        print(f'l0={l0}')
                        env_generator.manual_seed(env_seed)
                        algo_generator = np.random.default_rng(algo_seed)

                        algo = CoxCP(l0=l0, d=d, T=T, ngrid=ngrid, generator=algo_generator)
                        algo.run(rep=rep, env=env, basedir=f'{basedir}/l0={l0}')
                
                best = 1e9
                for l0 in l0_set:
                    dir=f'{basedir}/l0={l0}'
                    reward = np.load(dir+'/reward.npy')
                    optimal_reward = np.load(dir+'/optimal_reward.npy')
                    mean_regret = np.mean(optimal_reward-reward)
                    if mean_regret < best:
                        best = mean_regret
                        best_param = l0
                    print(f'l0={l0}: {np.mean(optimal_reward-reward)}')
                
                # run with the best parameter
                print(f'best parameter : l0={best_param}')

                l0 = best_param
                d = 5
                T = 5000
                rep = 5
                env_seed = algo_seed = 123
                basedir = f'results/sim_d={d}_T={T}/model={model}/cdf={cdf_type}/context={context_type}/{algo_name}'

                env_generator.manual_seed(env_seed)
                algo_generator = np.random.default_rng(algo_seed)
                algo = CoxCP(l0=l0, d=d, T=T, ngrid=ngrid, generator=algo_generator)
                algo.run(rep=rep, env=env, basedir=basedir)