from util_fdp_alt import *
import ray

"""
Main code: Identical distribution
"""

ray.init(runtime_env={"working_dir": "."})  # Set working directory
os.makedirs("output", exist_ok=True)

dist_type = 'normal'
gene_process = 'homo' # 'hete' / 'hete_d'
mode='federated' # federated / global
T_mode='samples'  # samples / rounds
n_sim = 1000
seed = 42 if gene_process == 'homo' else 2025
a=0.51; b=100; c=20

Ts = [10000, 50000] if T_mode == 'samples' else [5000,10000]
tauss = [[0.5]*10,np.linspace(0.3, 0.5, 10).tolist(),
        np.linspace(0.5, 0.8, 10).tolist()]
rs = [0.25,0.9]

n_clients = 10
client_rss = generate_lists(rs[0], rs[1], n_clients)
lr_ratio = np.mean(client_rss,axis=1)

rs_names = [rs[0],'hetero',rs[1]]


Es = [1]
nn_ct = len(Ts)*len(tauss)*len(client_rss)*len(Es)

# Initialize result storage dictionaries
cvgdict = {}
maedict = {}

# Federated simulation
ct = 0
for T in Ts:
    # Initialize dictionary level for current T
    cvgdict[T] = {};maedict[T] = {}
    for i,taus in enumerate(tauss):
        # Initialize dictionary level for current quantiles
        cvgdict[T][i] = {};maedict[T][i] = {}
        for name_idx, client_rs in enumerate(client_rss):
            c_rs = c * lr_ratio[name_idx]
            name = rs_names[name_idx]
            cvgdict[T][i][name] = {};maedict[T][i][name] = {}
            for E in Es:
                E_typ = 'log' if E == 'log' else 'cons'
                t1 = time.time()
                fed_results = run_federated_simulation(
                    dist_type=dist_type,taus=taus,
                    client_rs=client_rs,n_clients=n_clients,
                    T=T,E_typ=E_typ,E_cons=E,gene_process=gene_process,
                    mode=mode,
                    n_sim=n_sim,base_seed=seed,a=a, b=b,c=c_rs,
                T_mode=T_mode)
                # Analyze results
                z_score = 6.753 if E == 'log' else 6.74735
                output = analyze_results(fed_results,z_score=z_score)
                cvg = output['coverage'];mae = output['mae']

                # Store results
                cvgdict[T][i][name][E] = cvg;maedict[T][i][name][E] = mae
                t2 = time.time()
                ct += 1
                save_pickle(cvgdict, f'output/case_dpsgd_{dist_type}_{mode}_{T_mode}_{gene_process}_cvg.pkl')
                save_pickle(maedict, f'output/case_dpsgd_{dist_type}_{mode}_{T_mode}_{gene_process}_mae.pkl')
                print(f'Ts:{T} taus:{taus} name:{name} E:{E} TC:{(t2-t1)/60:.2f}min LTC:{(t2-t1)*(nn_ct-ct)/60:.2f}min')
                
ray.shutdown()