from pipelines import *
from tasks import *
from pipelines import *


def get_config():
    lr = 0.01
    scale_reg = 1e-1
    option_dataset = {'name_scaler': 'minmax'}
    num_epochs = 1000
    name_dataset = 'adult_fix'
    input_dim = 14
    device = 'cuda'
    n_train = 5000
    n_val = 5000
    batch_size = 1
    depth = 10000
    scale = 1.
    name_hyper_model = "multi"
    option_model = {}
    dtype = 'float'
    kwargs_method = {}
    option_oracle = {}
    # lr = params['alpha'] / (params['gamma'] + t + 1)
    d_configs = {}

    d_configs.update({f'neumann': ('neumann', {'scale': scale})})
    d_configs.update({f'unroll': ('unroll', {'scale': scale})})

    for a in [1e-3, 2e-3, 5e-3, 1e-2, 2e-2, 5e-2, 1e-1, 2e-1, 5e-1]:
        d_configs.update({f'vr_no_km(a={a})': ('vr', {'a': a, 'scale': scale})})

    scheduler = 'const'
    for lr_ in [1e-3, 2e-3, 5e-3, 1e-2, 2e-2, 5e-2, 1e-1, 2e-1, 5e-1]:
        d_configs.update({f'grazzi_{scheduler}(lr={lr_})': ('grazzi', {'lr': lr_, 'scheduler': scheduler, 'scale': scale})})

    for a in [1e-3, 2e-3, 5e-3, 1e-2, 2e-2, 5e-2, 1e-1, 2e-1, 5e-1]:
        for lr_ in [1e-3, 2e-3, 5e-3, 1e-2, 2e-2, 5e-2, 1e-1, 2e-1, 5e-1]:
            d_configs.update({f'vr_km_{scheduler}(a={a},lr={lr_})': ('vr_km', {'a': a, 'lr': lr_, 'scheduler': scheduler, 'scale': scale})})

    scheduler = 'linear'
    for gamma in [1e1, 2e1, 5e1, 1e2, 2e2, 5e2, 1e3, 2e3, 5e3, 1e4]:
        for alpha in [1e1, 2e1, 5e1, 1e2, 2e2, 5e2, 1e3, 2e3, 5e3, 1e4]:
            if alpha <=  gamma:
                d_configs.update({f'grazzi_{scheduler}(alpha={alpha},gamma={gamma})': ('grazzi', {'gamma': gamma, 'alpha': alpha, 'scheduler': scheduler, 'scale': scale})})

    for a in [1e-4, 2e-4, 5e-4, 1e-3, 2e-3, 5e-3, 1e-2, 2e-2, 5e-2, 1e-1, 2e-1, 5e-1]:
        for gamma in [1e1, 2e1, 5e1, 1e2, 2e2, 5e2, 1e3, 2e3, 5e3, 1e4]:
            for alpha in [1e1, 2e1, 5e1, 1e2, 2e2, 5e2, 1e3, 2e3, 5e3, 1e4]:
                if alpha <= gamma:
                    d_configs.update({f'vr_km_{scheduler}(a={a},alpha={alpha},gamma={gamma})': ('vr_km', {'a': a, 'gamma': gamma, 'alpha': alpha, 'scheduler': scheduler, 'scale': scale})})

    scheduler = 'const'
    for lr_ in [1e0]:
        for scale in [1e-3, 2e-3, 5e-3, 1e-2, 2e-2, 5e-2, 1e-1, 2e-1, 5e-1, 1e0]:
            d_configs.update({f'grazzi_{scheduler}(lr={lr_},scale={scale})': ('grazzi', {'lr': lr_, 'scheduler': scheduler, 'scale': scale})})

    xlabel = 'sample'
    labels = list(d_configs.keys())
    methods = [method for method, kwargs in d_configs.values()]
    params = [kwargs for method, kwargs in d_configs.values()]

    seeds = list(range(10))

    param_dict = {
        gokart.TaskOnKart.task_family: {
            "fix_random_seed_value": 0
        },
        DatasetPreparation.task_family: {
            "name_dataset": name_dataset,
            "option_dataset": option_dataset,
            "n_train": n_train,
            "n_val": n_val,
            "input_dim": input_dim,
            "dtype": dtype
        },
        ModelInitialization.task_family: {
            "name_hyper_model": name_hyper_model,
            "kwargs_hyper": {"scale_reg": scale_reg},
            "name_dataset": name_dataset,
            "input_dim": input_dim,
            "option_model": option_model,
            "device": device,
            "dtype": dtype
        },
        FullBatchInnerOptimization.task_family: {
            "lr": lr,
            "num_epochs": num_epochs,
            "name_dataset": name_dataset,
            "input_dim": input_dim,
            "device": device,
            "dtype": dtype
        },
        CreateGradientOracle.task_family: {
            "dtype": dtype,
            "input_dim": input_dim,
            "batch_size": batch_size,
            "depth": depth,
            "kwargs_method": kwargs_method,
            "option_oracle": option_oracle,
            "device": device,
        },
        BatchHyperGradEstimations.task_family: {
            "dtype": dtype,
            "depth": depth,
            "kwargs_method": kwargs_method,
        },
        MakeOutputDict.task_family: {
        },
        PlotErrors.task_family: {
            "xlabel": xlabel,
            "xscale": "linear",
            "yscale": "log",
        },
        MainPipeline.task_family: {
            "labels": labels,
            "methods": methods,
            "params": params,
            "seeds": seeds
        },
        GetGoodResults.task_family: {
            "n_methods": {
                "neumann": 1,
                "unroll": 1,
                "vr_no_km": 1,
                "grazzi_const": 1,
                "vr_km_const": 1,
                "grazzi_linear": 1,
                "vr_km_linear": 1,
            },
            "ave_over_last": int(depth / 10)
        },
        SampleHessians.task_family: {
            "dtype": dtype,
        }
    }

    return param_dict
