# main/main.py compressed, if does not work uncomment long version.
import yaml, os, json, argparse
from pathlib import Path
from main.loading.utils import set_seed, setup_logger
from main.loading.data_loader import get_data
from main.model.model import NeuralCDE, QFormerCDE, ConvCDE, ODERNN, GRUD, LogNeuralCDE
from main.model.trainer import run_experiment

def load_config(path):
    with open(path, 'r') as f: return yaml.safe_load(f)

def get_params(base, specific):
    p = base.copy(); p.update(specific); return p

def save_results(path, data):
    os.makedirs(os.path.dirname(path), exist_ok=True)
    with open(path, 'w') as f: json.dump(data, f, indent=4, default=str)

def main():
    parser = argparse.ArgumentParser(); parser.add_argument("--config", default="./main/config.yaml")
    args = parser.parse_args(); cfg = load_config(args.config); logger = setup_logger(cfg['log_dir'])
    
    logger.info("="*54 + "\n           STARTING CDE EXPERIMENT SUITE            \n" + f"           Config: {args.config}                    \n" + "="*54)
    
    time_factors = cfg.get('time_scaling_factors', []) or [cfg.get('time_scaling_factor', 'no')]
    res_root, ds_name = cfg['results_dir'], cfg['dataset_name']
    add_time_s = str(cfg.get('add_time', 'yes')).lower(); add_time_b = add_time_s == 'yes'
    logger.info(f"Include Time in Input: {add_time_b} (Config: {add_time_s})"); logger.info(f"Plotting/Diagnostics Mode: {cfg.get('plot_data', False)}")

    for ts in time_factors:
        cfg['time_scaling_factor'] = ts
        ts_no = str(ts).lower() == 'no'
        ts_suf, ts_log = ("ts-no", "NO SCALING") if ts_no else (f"ts-{ts}", str(ts))
        
        logger.info(f"\n{'#'*60}\nPROCESSING TIME SCALING FACTOR: {ts_log}\n{'#'*60}")
        data = get_data(cfg, logger); train_X, _, _, train_y, _, _, t_grid = data
        seq_len, raw_ch, out_ch = train_X.shape[1], train_X.size(-1), len(train_y.unique())
        bw_mult = seq_len
        logger.info(f"Sequence Length: {seq_len}. BW Multiplier applied: {bw_mult}")
        logger.info(f"Raw channels: {raw_ch}, Model Input Channels: {raw_ch}")

        for seed in cfg['seeds']:
            for tol in cfg['tolerances']:
                logger.info(f"\n{'='*25} SEED: {seed} | TOL: {tol} | TS: {ts_log} {'='*25}")
                set_seed(seed)
                base_p = {'dataset_name': ds_name, 'seed': seed, 'num_epochs': cfg['num_epochs'], 'batch_size': cfg['batch_size'],
                          'time_scaling_factor': ts, 'tol': tol, 'lr': cfg['lr'], 'weight_decay': cfg['weight_decay'],
                          'hidden_dim': cfg['hidden_dim'], 'bw_multiplier': bw_mult, 'add_time': add_time_s, 'plot_data': cfg.get('plot_data', False)}
                m_tgrid = t_grid if not add_time_b else None
                
                def run(cls, kw, args, name, sub, params):
                    path = Path(res_root) / ds_name / sub / f"{name}_seed-{seed}.json"
                    if path.exists(): return
                    kw.update({'input_channels': raw_ch, 'hidden_dim': cfg['hidden_dim'], 'output_channels': out_ch, 'seq_len': seq_len, 'add_time': add_time_b, 't_grid': m_tgrid})
                    if 'tol' not in kw and cls != GRUD: kw['tol'] = tol
                    res = run_experiment(cls(**kw), data, cfg, logger, name, *args)
                    res.update(params)
                    save_results(path, res)
                    if cfg.get('test_noise_robustness') and res.get('noise_robustness_results'):
                        try: rel = path.relative_to(cfg['results_dir'])
                        except: rel = Path(ds_name) / path.name
                        npath = Path(cfg['data_dir']).parent / cfg.get('noise_results_dir', 'experiment_noise') / rel
                        save_results(npath, {"experiment_name": name, "seed": seed, "model_params": params, "noise_results": res['noise_robustness_results']})
                        logger.info(f"Saving NOISE results to: {npath}")

                rl = cfg['experiments_to_run']
                if rl.get('baseline'):
                    for i in cfg['baseline_params']['interpolations']:
                        run(NeuralCDE, {'interpolation': i}, (i,), f"baseline_interp-{i}_tol-{tol}_{ts_suf}", "baseline", get_params(base_p, {'type': "baseline", 'interpolation': i}))
                
                if rl.get('log_ncde'):
                    lc = cfg.get('log_ncde_params', {'step_sizes': [5], 'depths': [1]})
                    for s in lc.get('step_sizes', [5]):
                        for d in lc.get('depths', [1]):
                            p = {'step_size': s, 'depth': d, 'tol': tol}
                            run(LogNeuralCDE, {'log_ncde_params': p}, ("log_ncde", {'kernel_params': p}), f"logncde_depth{d}_step{s}_tol-{tol}_{ts_suf}", "Log-NCDE", get_params(base_p, {'type': "log_ncde", 'interpolation': 'log_ncde', **p}))

                if rl.get('kernel'):
                    for k in cfg['kernel_params']['kernels']:
                        for br in cfg['kernel_params']['bandwidths']:
                            bw = round(br*bw_mult, 4); p = {'kernel': k, 'bandwidth': bw}
                            run(NeuralCDE, {'interpolation': "kernel", 'kernel_params': p}, ("kernel", p), f"kernel-{k}_bw-{bw}_tol-{tol}_{ts_suf}", f"kernel/{k}", get_params(base_p, {'type': "kernel", 'interpolation': 'kernel', 'bw_raw': br, **p}))

                if rl.get('gp'):
                    g = cfg.get('gp_params', {})
                    for lr in g.get('length_scales', [1.0]):
                        ls = round(lr*bw_mult, 4)
                        for n in g.get('noise_stds', [0.01]):
                            p = {'length_scale': ls, 'noise_std': n}
                            run(NeuralCDE, {'interpolation': "gp", 'kernel_params': p}, ("gp", p), f"gp_ls-{ls}_noise-{n}_tol-{tol}_{ts_suf}", "GP", get_params(base_p, {'type': "gp", 'interpolation': 'gp', 'ls_raw': lr, **p}))

                if rl.get('odernn'): run(ODERNN, {}, ("odernn", {'tol': tol}), f"odernn_tol-{tol}_{ts_suf}", "ODE-RNN", get_params(base_p, {'type': "odernn", 'interpolation': 'odernn'}))
                if rl.get('grud'): run(GRUD, {}, ("grud", {'tol': 'N/A'}), f"grud_tol-{tol}_{ts_suf}", "GRU-D", get_params(base_p, {'type': "grud", 'interpolation': 'grud'}))

                q_pc = cfg.get('qformer_params', {})
                for key, Cls, anm, ex in [('qformer', QFormerCDE, 'qformer_params', {}), ('conv', ConvCDE, 'conv_params', {'conv_kernel_size': cfg.get("conv_params", {}).get("kernel_size")})]:
                    if not rl.get(key): continue
                    bws = []
                    if 'bandwidth_lists_different' in q_pc: bws.extend([('diff', b) for b in q_pc['bandwidth_lists_different']])
                    if 'bandwidth_lists_same' in q_pc: bws.extend([('same', b) for b in q_pc['bandwidth_lists_same']])
                    for ag in q_pc.get('aggregations', [] if key == 'qformer' else ['concat']):
                        for bt, br in bws:
                            bs = [round(b * bw_mult, 4) for b in br]
                            for k in q_pc.get('kernels', [] if key == 'qformer' else ['laplacian']):
                                for n in (cfg.get('gp_params', {}).get('noise_stds', [0.01]) if k == 'gp' else [None]):
                                    par = {'kernel': k, 'bandwidths': bs, 'tol': tol, 'aggregation': ag, **ex}; spec = {'type': key.replace('_', '-'), 'interpolation': key, 'kernel': k, 'bws': bs, 'bws_raw': br, 'aggr': ag, **ex}
                                    if n: par['noise_std'] = n; spec['noise_std'] = n
                                    if key == 'conv': par['type'] = 'conv'
                                    bn = f"{key}_gp_{bt}_ls-{str(bs).replace(' ','')}_noise-{n}_tol-{tol}_aggr-{ag}" if k == 'gp' else f"{key}_{bt}_kernel-{k}_bws-{str(bs).replace(' ','')}_tol-{tol}_aggr-{ag}"
                                    if key == 'conv': bn += f"_ks-{ex['conv_kernel_size']}"
                                    run(Cls, {anm: par}, (key, par), f"{bn}_{ts_suf}", f"{key}/{bt}/{'gp' if k=='gp' else k}", get_params(base_p, spec))

    logger.info("ALL EXPERIMENTS COMPLETED")

if __name__ == "__main__": main()

# 
# long-version
# # main/main.py
# import yaml
# import os
# import json
# import argparse
# from pathlib import Path
# from main.loading.utils import set_seed, setup_logger
# from main.loading.data_loader import get_data
# from main.model.model import NeuralCDE, QFormerCDE, ConvCDE, ODERNN, GRUD, LogNeuralCDE
# from main.model.trainer import run_experiment

# def load_config(path):
#     with open(path, 'r') as f:
#         return yaml.safe_load(f)

# def get_experiment_params(base_params, exp_specific_params):
#     params = base_params.copy()
#     params.update(exp_specific_params)
#     return params

# def save_results(result_path, result_data):
#     os.makedirs(os.path.dirname(result_path), exist_ok=True)
#     with open(result_path, 'w') as f:
#         json.dump(result_data, f, indent=4, default=str)


# def main():
#     parser = argparse.ArgumentParser(description="Run CDE Experiments")
#     parser.add_argument("--config", type=str, default="./main/config.yaml", help="Path to the config file")
#     args = parser.parse_args()
    
#     cfg = load_config(args.config)
    
#     logger = setup_logger(cfg['log_dir'])
    
#     logger.info("======================================================")
#     logger.info("           STARTING CDE EXPERIMENT SUITE            ")
#     logger.info(f"           Config: {args.config}                    ")
#     logger.info("======================================================")
    
#     time_factors = cfg.get('time_scaling_factors', [])
#     if not time_factors:
#         single_factor = cfg.get('time_scaling_factor', 'no')
#         time_factors = [single_factor]
    
#     dataset_results_dir = os.path.join(cfg['results_dir'], cfg['dataset_name'])

#     add_time_str = str(cfg.get('add_time', 'yes')).lower()
#     add_time_bool = (add_time_str == 'yes')
#     logger.info(f"Include Time in Input: {add_time_bool} (Config: {add_time_str})")
    
#     plot_data_bool = cfg.get('plot_data', False)
#     logger.info(f"Plotting/Diagnostics Mode: {plot_data_bool}")

#     for ts_factor in time_factors:
#         cfg['time_scaling_factor'] = ts_factor
        
#         if isinstance(ts_factor, str) and ts_factor.lower() == 'no':
#             ts_suffix = "ts-no"
#             ts_log_str = "NO SCALING"
#         else:
#             ts_suffix = f"ts-{ts_factor}"
#             ts_log_str = str(ts_factor)

#         logger.info(f"\n{'#'*60}")
#         logger.info(f"PROCESSING TIME SCALING FACTOR: {ts_log_str}")
#         logger.info(f"{'#'*60}")

#         # Unpack 7 values (including timestamps)
#         data = get_data(cfg, logger)
#         train_X, _, _, train_y, _, _, t_grid = data
        
#         seq_len = train_X.shape[1] 
#         bw_multiplier = seq_len
#         logger.info(f"Sequence Length: {seq_len}. BW Multiplier applied: {bw_multiplier}")
        
#         raw_channels = train_X.size(-1)
#         input_channels = raw_channels
            
#         output_channels = len(train_y.unique())
        
#         logger.info(f"Raw channels: {raw_channels}, Model Input Channels: {input_channels}")

#         for seed in cfg['seeds']:
#             for tol in cfg['tolerances']:
#                 logger.info(f"\n{'='*25} SEED: {seed} | TOL: {tol} | TS: {ts_log_str} {'='*25}")
#                 set_seed(seed)
                
#                 base_params = {
#                     'dataset_name': cfg['dataset_name'],
#                     'seed': seed,
#                     'num_epochs': cfg['num_epochs'],
#                     'batch_size': cfg['batch_size'],
#                     'time_scaling_factor': ts_factor, 
#                     'tol': tol,
#                     'lr' : cfg['lr'],
#                     'weight_decay': cfg['weight_decay'],
#                     'hidden_dim' : cfg['hidden_dim'],
#                     'bw_multiplier': bw_multiplier,
#                     'add_time': add_time_str,
#                     'plot_data': plot_data_bool 
#                 }

#                 model_t_grid = t_grid if not add_time_bool else None

#                 # --- 1. BASELINE ---
#                 if cfg['experiments_to_run'].get('baseline', False):
#                     for interp in cfg['baseline_params']['interpolations']:
#                         exp_specifics = {'type': "baseline", 'interpolation': interp}
#                         exp_params = get_experiment_params(base_params, exp_specifics)
#                         exp_name = f"baseline_interp-{interp}_tol-{tol}_{ts_suffix}"
#                         results_dir = Path(dataset_results_dir) / "baseline"
#                         result_file = results_dir / f"{exp_name}_seed-{seed}.json"

#                         if result_file.exists(): continue

#                         model = NeuralCDE(input_channels, cfg['hidden_dim'], output_channels, seq_len, interpolation=interp, tol=tol, add_time=add_time_bool, t_grid=model_t_grid)
#                         result = run_experiment(model, data, cfg, logger, exp_name, interp)
#                         result.update(exp_params)
#                         save_results(result_file, result)


#                         # 2. НОВОЕ: Если есть результаты по шуму, сохраняем их отдельно
#                         if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                             noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
                            
#                             # Формируем структуру папок аналогично основной, но в noise папке
#                             # Например: experiment_noise/CharacterTrajectories/Log-NCDE/...
                            
#                             # Относительный путь от cfg['results_dir'] до result_file
#                             # result_file - это полный путь Path object
#                             try:
#                                 rel_path = result_file.relative_to(cfg['results_dir'])
#                             except ValueError:
#                                 # Fallback, если пути не совпадают (редко)
#                                 rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                             noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                            
#                             # Можно оставить только данные по шуму в этом файле, чтобы не дублировать всё
#                             noise_only_payload = {
#                                 "experiment_name": exp_name,
#                                 "seed": seed,
#                                 "model_params": exp_params,
#                                 "noise_results": result['noise_robustness_results']
#                             }
                            
#                             logger.info(f"Saving NOISE results to: {noise_save_path}")
#                             save_results(noise_save_path, noise_only_payload)
#                 # --- NEW: Log-NCDE ---
#                 if cfg['experiments_to_run'].get('log_ncde', False):
#                     # Параметры по умолчанию, если не заданы в конфиге
#                     lncde_cfg = cfg.get('log_ncde_params', {'step_sizes': [5], 'depths': [1]})
                    
#                     steps = lncde_cfg.get('step_sizes', [5])
#                     depths = lncde_cfg.get('depths', [1])

#                     for step in steps:
#                         for depth in depths:
#                             log_params = {'step_size': step, 'depth': depth, 'tol': tol}
#                             exp_specifics = {'type': "log_ncde", 'interpolation': 'log_ncde', **log_params}
#                             exp_params = get_experiment_params(base_params, exp_specifics)
                            
#                             exp_name = f"logncde_depth{depth}_step{step}_tol-{tol}_{ts_suffix}"
#                             results_dir = Path(dataset_results_dir) / "Log-NCDE"
#                             result_file = results_dir / f"{exp_name}_seed-{seed}.json"

#                             if result_file.exists(): continue

#                             # Log-NCDE работает с сырыми данными (не cubic coeffs), 
#                             # но data_loader уже подготовил X как тензор.
#                             # Передаем add_time из конфига.
#                             model = LogNeuralCDE(input_channels, cfg['hidden_dim'], output_channels, seq_len, 
#                                                  log_ncde_params=log_params, tol=tol, add_time=add_time_bool, t_grid=model_t_grid)
                            
#                             result = run_experiment(model, data, cfg, logger, exp_name, "log_ncde", kernel_params=log_params)
#                             result.update(exp_params)
#                             save_results(result_file, result)

#                             # 2. НОВОЕ: Если есть результаты по шуму, сохраняем их отдельно
#                             if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                                 noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
                                
#                                 # Формируем структуру папок аналогично основной, но в noise папке
#                                 # Например: experiment_noise/CharacterTrajectories/Log-NCDE/...
                                
#                                 # Относительный путь от cfg['results_dir'] до result_file
#                                 # result_file - это полный путь Path object
#                                 try:
#                                     rel_path = result_file.relative_to(cfg['results_dir'])
#                                 except ValueError:
#                                     # Fallback, если пути не совпадают (редко)
#                                     rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                                 noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                                
#                                 # Можно оставить только данные по шуму в этом файле, чтобы не дублировать всё
#                                 noise_only_payload = {
#                                     "experiment_name": exp_name,
#                                     "seed": seed,
#                                     "model_params": exp_params,
#                                     "noise_results": result['noise_robustness_results']
#                                 }
                                
#                                 logger.info(f"Saving NOISE results to: {noise_save_path}")
#                                 save_results(noise_save_path, noise_only_payload)
#                 # --- 2. KERNEL (BW Scaled) ---
#                 if cfg['experiments_to_run'].get('kernel', False):
#                     for kernel in cfg['kernel_params']['kernels']:
#                         for bw_raw in cfg['kernel_params']['bandwidths']:
#                             bw = round(bw_raw * bw_multiplier, 4)
                            
#                             kernel_p = {'kernel': kernel, 'bandwidth': bw}
#                             exp_specifics = {'type': "kernel", 'interpolation': 'kernel', 'bw_raw': bw_raw, **kernel_p}
#                             exp_params = get_experiment_params(base_params, exp_specifics)
                            
#                             exp_name = f"kernel-{kernel}_bw-{bw}_tol-{tol}_{ts_suffix}"
#                             results_dir = Path(dataset_results_dir) / "kernel" / kernel
#                             result_file = results_dir / f"{exp_name}_seed-{seed}.json"

#                             if result_file.exists(): continue

#                             model = NeuralCDE(input_channels, cfg['hidden_dim'], output_channels, seq_len, "kernel", kernel_params=kernel_p, tol=tol, add_time=add_time_bool, t_grid=model_t_grid)
#                             result = run_experiment(model, data, cfg, logger, exp_name, "kernel", kernel_p)
#                             result.update(exp_params)
#                             save_results(result_file, result)

#                             # 2. НОВОЕ: Если есть результаты по шуму, сохраняем их отдельно
#                             if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                                 noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
                                
#                                 # Формируем структуру папок аналогично основной, но в noise папке
#                                 # Например: experiment_noise/CharacterTrajectories/Log-NCDE/...
                                
#                                 # Относительный путь от cfg['results_dir'] до result_file
#                                 # result_file - это полный путь Path object
#                                 try:
#                                     rel_path = result_file.relative_to(cfg['results_dir'])
#                                 except ValueError:
#                                     # Fallback, если пути не совпадают (редко)
#                                     rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                                 noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                                
#                                 # Можно оставить только данные по шуму в этом файле, чтобы не дублировать всё
#                                 noise_only_payload = {
#                                     "experiment_name": exp_name,
#                                     "seed": seed,
#                                     "model_params": exp_params,
#                                     "noise_results": result['noise_robustness_results']
#                                 }
                                
#                                 logger.info(f"Saving NOISE results to: {noise_save_path}")
#                                 save_results(noise_save_path, noise_only_payload)

#                 # --- 3. GP (Gaussian Process) ---
#                 if cfg['experiments_to_run'].get('gp', False):
#                     gp_cfg = cfg.get('gp_params', {})
#                     ls_list = gp_cfg.get('length_scales', [1.0])
#                     noise_list = gp_cfg.get('noise_stds', [0.01])
                    
#                     for ls_raw in ls_list:
#                         ls = round(ls_raw * bw_multiplier, 4)
#                         for noise in noise_list:
                            
#                             gp_p = {'length_scale': ls, 'noise_std': noise}
#                             exp_specifics = {'type': "gp", 'interpolation': 'gp', 'ls_raw': ls_raw, **gp_p}
#                             exp_params = get_experiment_params(base_params, exp_specifics)
                            
#                             exp_name = f"gp_ls-{ls}_noise-{noise}_tol-{tol}_{ts_suffix}"
#                             results_dir = Path(dataset_results_dir) / "GP"
#                             result_file = results_dir / f"{exp_name}_seed-{seed}.json"

#                             if result_file.exists(): continue

#                             model = NeuralCDE(input_channels, cfg['hidden_dim'], output_channels, seq_len, "gp", kernel_params=gp_p, tol=tol, add_time=add_time_bool, t_grid=model_t_grid)
#                             result = run_experiment(model, data, cfg, logger, exp_name, "gp", gp_p)
#                             result.update(exp_params)
#                             save_results(result_file, result)

#                             if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                                 noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
#                                 try:
#                                     rel_path = result_file.relative_to(cfg['results_dir'])
#                                 except ValueError:
#                                     rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                                 noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                                
#                                 noise_only_payload = {
#                                     "experiment_name": exp_name,
#                                     "seed": seed,
#                                     "model_params": exp_params,
#                                     "noise_results": result['noise_robustness_results']
#                                 }
                                
#                                 logger.info(f"Saving NOISE results to: {noise_save_path}")
#                                 save_results(noise_save_path, noise_only_payload)

#                 if cfg['experiments_to_run'].get('odernn', False):
#                     exp_specifics = {'type': "odernn", 'interpolation': 'odernn'}
#                     exp_params = get_experiment_params(base_params, exp_specifics)
                    
#                     exp_name = f"odernn_tol-{tol}_{ts_suffix}"
#                     results_dir = Path(dataset_results_dir) / "ODE-RNN"
#                     result_file = results_dir / f"{exp_name}_seed-{seed}.json"

#                     if not result_file.exists():
#                         model = ODERNN(input_channels, cfg['hidden_dim'], output_channels, seq_len, tol=tol, add_time=add_time_bool, t_grid=model_t_grid)
#                         result = run_experiment(model, data, cfg, logger, exp_name, "odernn", kernel_params={'tol': tol})
#                         result.update(exp_params)
#                         save_results(result_file, result)

#                         if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                             noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
                            
#                             try:
#                                 rel_path = result_file.relative_to(cfg['results_dir'])
#                             except ValueError:
#                                 rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                             noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                            
#                             noise_only_payload = {
#                                 "experiment_name": exp_name,
#                                 "seed": seed,
#                                 "model_params": exp_params,
#                                 "noise_results": result['noise_robustness_results']
#                             }
                            
#                             logger.info(f"Saving NOISE results to: {noise_save_path}")
#                             save_results(noise_save_path, noise_only_payload)
                
#                 if cfg['experiments_to_run'].get('grud', False):
#                     exp_specifics = {'type': "grud", 'interpolation': 'grud'}
#                     exp_params = get_experiment_params(base_params, exp_specifics)
                    
#                     exp_name = f"grud_tol-{tol}_{ts_suffix}"
#                     results_dir = Path(dataset_results_dir) / "GRU-D"
#                     result_file = results_dir / f"{exp_name}_seed-{seed}.json"

#                     if not result_file.exists():
#                         model = GRUD(input_channels, cfg['hidden_dim'], output_channels, seq_len, add_time=add_time_bool, t_grid=model_t_grid)
#                         result = run_experiment(model, data, cfg, logger, exp_name, "grud", kernel_params={'tol': 'N/A'})
#                         result.update(exp_params)
#                         save_results(result_file, result)

#                         if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                             noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
                            
#                             try:
#                                 rel_path = result_file.relative_to(cfg['results_dir'])
#                             except ValueError:
#                                 rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                             noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                            
#                             noise_only_payload = {
#                                 "experiment_name": exp_name,
#                                 "seed": seed,
#                                 "model_params": exp_params,
#                                 "noise_results": result['noise_robustness_results']
#                             }
                            
#                             logger.info(f"Saving NOISE results to: {noise_save_path}")
#                             save_results(noise_save_path, noise_only_payload)
                            

#                 if cfg['experiments_to_run'].get('qformer', False):
#                     q_cfg = cfg['qformer_params']
#                     gp_cfg = cfg.get('gp_params', {})
#                     noise_list = gp_cfg.get('noise_stds', [0.01])

#                     all_bw_configs = []
#                     if 'bandwidth_lists_different' in q_cfg:
#                         all_bw_configs.extend([('diff', bw) for bw in q_cfg['bandwidth_lists_different']])
#                     if 'bandwidth_lists_same' in q_cfg:
#                         all_bw_configs.extend([('same', bw) for bw in q_cfg['bandwidth_lists_same']])

#                     for aggregation in q_cfg.get('aggregations', []):
#                         for (bw_type, bws_raw) in all_bw_configs:
#                             bws_scaled = [round(b * bw_multiplier, 4) for b in bws_raw]
                            
#                             for kernel in q_cfg.get('kernels', []):
#                                 if kernel == 'gp':
#                                     for noise in noise_list:
#                                         q_params = {
#                                             'kernel': 'gp', 
#                                             'bandwidths': bws_scaled, 
#                                             'noise_std': noise, 
#                                             'tol': tol, 
#                                             'aggregation': aggregation
#                                         }
#                                         exp_name_base = f"qformer_gp_{bw_type}_ls-{str(bws_scaled).replace(' ','')}_noise-{noise}_tol-{tol}_aggr-{aggregation}_{ts_suffix}"
#                                         folder_name = "gp"
                                        
#                                         exp_specifics = {
#                                             'type': "q-former", 'interpolation': 'qformer', 
#                                             'kernel': kernel, 'bws': bws_scaled, 'bws_raw': bws_raw, 
#                                             'aggr': aggregation, 'noise_std': noise 
#                                         }
#                                         exp_params = get_experiment_params(base_params, exp_specifics)
                                        
#                                         result_file = Path(dataset_results_dir) / "qformer" / bw_type / folder_name / f"{exp_name_base}_seed-{seed}.json"

#                                         if result_file.exists(): continue

#                                         model = QFormerCDE(input_channels, cfg['hidden_dim'], output_channels, seq_len, qformer_params=q_params, add_time=add_time_bool, t_grid=model_t_grid)
#                                         result = run_experiment(model, data, cfg, logger, exp_name_base, "qformer", q_params)
#                                         result.update(exp_params)
#                                         save_results(result_file, result)

#                                         if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                                             noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
                                            
                                            
#                                             try:
#                                                 rel_path = result_file.relative_to(cfg['results_dir'])
#                                             except ValueError:
#                                                 rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                                             noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                                            
#                                             noise_only_payload = {
#                                                 "experiment_name": exp_name,
#                                                 "seed": seed,
#                                                 "model_params": exp_params,
#                                                 "noise_results": result['noise_robustness_results']
#                                             }
                                            
#                                             logger.info(f"Saving NOISE results to: {noise_save_path}")
#                                             save_results(noise_save_path, noise_only_payload)
#                                 else:
#                                     q_params = {'kernel': kernel, 'bandwidths': bws_scaled, 'tol': tol, 'aggregation': aggregation}
#                                     exp_name_base = f"qformer_{bw_type}_kernel-{kernel}_bws-{str(bws_scaled).replace(' ','')}_tol-{tol}_aggr-{aggregation}_{ts_suffix}"
#                                     folder_name = kernel

#                                     exp_specifics = {'type': "q-former", 'interpolation': 'qformer', 'kernel': kernel, 'bws': bws_scaled, 'bws_raw': bws_raw, 'aggr': aggregation}
#                                     exp_params = get_experiment_params(base_params, exp_specifics)
                                    
#                                     result_file = Path(dataset_results_dir) / "qformer" / bw_type / folder_name / f"{exp_name_base}_seed-{seed}.json"

#                                     if result_file.exists(): continue

#                                     model = QFormerCDE(input_channels, cfg['hidden_dim'], output_channels, seq_len, qformer_params=q_params, add_time=add_time_bool, t_grid=model_t_grid)
#                                     result = run_experiment(model, data, cfg, logger, exp_name_base, "qformer", q_params)
#                                     result.update(exp_params)
#                                     save_results(result_file, result)


#                                     if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                                         noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
                                        
#                                         try:
#                                             rel_path = result_file.relative_to(cfg['results_dir'])
#                                         except ValueError:
#                                             rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                                         noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                                        
#                                         noise_only_payload = {
#                                             "experiment_name": exp_name,
#                                             "seed": seed,
#                                             "model_params": exp_params,
#                                             "noise_results": result['noise_robustness_results']
#                                         }
                                        
#                                         logger.info(f"Saving NOISE results to: {noise_save_path}")
#                                         save_results(noise_save_path, noise_only_payload)

#                 if cfg['experiments_to_run'].get('conv', False):
#                     c_ksize = cfg["conv_params"]["kernel_size"]
#                     q_cfg = cfg['qformer_params'] 
                    
#                     gp_cfg = cfg.get('gp_params', {})
#                     noise_list = gp_cfg.get('noise_stds', [0.01])

#                     all_bw_configs = []
#                     if 'bandwidth_lists_different' in q_cfg:
#                         all_bw_configs.extend([('diff', bw) for bw in q_cfg['bandwidth_lists_different']])
#                     if 'bandwidth_lists_same' in q_cfg:
#                         all_bw_configs.extend([('same', bw) for bw in q_cfg['bandwidth_lists_same']])

#                     for aggr in q_cfg.get('aggregations', ['concat']):
#                         for (bw_type, bws_raw) in all_bw_configs:
#                             bws_scaled = [round(b * bw_multiplier, 4) for b in bws_raw]
                            
#                             for kernel in q_cfg.get('kernels', ['laplacian']): 
                                
#                                 if kernel == 'gp':
#                                     for noise in noise_list:
#                                         conv_p = {
#                                             'type': "conv", 'kernel': 'gp', 
#                                             'bandwidths': bws_scaled, 
#                                             'noise_std': noise, 
#                                             'tol': tol, 'aggregation': aggr, 'conv_kernel_size': c_ksize
#                                         }
#                                         exp_name_base = f"conv_gp_{bw_type}_ls-{str(bws_scaled).replace(' ','')}_noise-{noise}_tol-{tol}_aggr-{aggr}_ks-{c_ksize}_{ts_suffix}"
#                                         folder_name = "gp"
                                        
#                                         exp_specifics = {
#                                             'type': "conv", 'interpolation': 'conv', 
#                                             'kernel': kernel, 'bws': bws_scaled, 'bws_raw': bws_raw, 
#                                             'aggr': aggr, 'conv_kernel_size': c_ksize,
#                                             'noise_std': noise 
#                                         }
#                                         exp_params = get_experiment_params(base_params, exp_specifics)
                                        
#                                         result_file = Path(dataset_results_dir) / "conv" / bw_type / folder_name / f"{exp_name_base}_seed-{seed}.json"

#                                         if result_file.exists(): continue
                                        
#                                         model = ConvCDE(input_channels, cfg['hidden_dim'], output_channels, seq_len, conv_params=conv_p, add_time=add_time_bool, t_grid=model_t_grid)
#                                         result = run_experiment(model, data, cfg, logger, exp_name_base, "conv", conv_p)
#                                         result.update(exp_params)
#                                         save_results(result_file, result)

                                       
#                                         if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                                             noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
                                            
#                                             try:
#                                                 rel_path = result_file.relative_to(cfg['results_dir'])
#                                             except ValueError:
                                                
#                                                 rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                                             noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                                            
                                            
#                                             noise_only_payload = {
#                                                 "experiment_name": exp_name,
#                                                 "seed": seed,
#                                                 "model_params": exp_params,
#                                                 "noise_results": result['noise_robustness_results']
#                                             }
                                            
#                                             logger.info(f"Saving NOISE results to: {noise_save_path}")
#                                             save_results(noise_save_path, noise_only_payload)
#                                 else:
                                    
#                                     conv_p = {'type': "conv", 'kernel': kernel, 'bandwidths': bws_scaled, 'tol': tol, 'aggregation': aggr, 'conv_kernel_size': c_ksize}
#                                     exp_name_base = f"conv_{bw_type}_kernel-{kernel}_bws-{str(bws_scaled).replace(' ','')}_tol-{tol}_aggr-{aggr}_ks-{c_ksize}_{ts_suffix}"
#                                     folder_name = kernel

#                                     exp_specifics = {'type': "conv", 'interpolation': 'conv', 'kernel': kernel, 'bws': bws_scaled, 'bws_raw': bws_raw, 'aggr': aggr, 'conv_kernel_size': c_ksize}
#                                     exp_params = get_experiment_params(base_params, exp_specifics)
                                    
#                                     result_file = Path(dataset_results_dir) / "conv" / bw_type / folder_name / f"{exp_name_base}_seed-{seed}.json"

#                                     if result_file.exists(): continue
                                    
#                                     model = ConvCDE(input_channels, cfg['hidden_dim'], output_channels, seq_len, conv_params=conv_p, add_time=add_time_bool, t_grid=model_t_grid)
#                                     result = run_experiment(model, data, cfg, logger, exp_name_base, "conv", conv_p)
#                                     result.update(exp_params)
#                                     save_results(result_file, result)

#                                     if cfg.get('test_noise_robustness', False) and result.get('noise_robustness_results'):
#                                         noise_dir_name = cfg.get('noise_results_dir', 'experiment_noise')
                                        
#                                         try:
#                                             rel_path = result_file.relative_to(cfg['results_dir'])
#                                         except ValueError:
#                                             rel_path = Path(cfg['dataset_name']) / f"{exp_name}_seed-{seed}.json"

#                                         noise_save_path = Path(cfg['data_dir']).parent / noise_dir_name / rel_path
                                        
#                                         noise_only_payload = {
#                                             "experiment_name": exp_name,
#                                             "seed": seed,
#                                             "model_params": exp_params,
#                                             "noise_results": result['noise_robustness_results']
#                                         }
                                        
#                                         logger.info(f"Saving NOISE results to: {noise_save_path}")
#                                         save_results(noise_save_path, noise_only_payload)
                                        
#     logger.info("ALL EXPERIMENTS COMPLETED")

# if __name__ == "__main__":
#     main()