import os
import torch
import pandas as pd
import numpy as np

PACKAGE_DIR = os.path.dirname(os.path.realpath(__file__))
DATA_DIR = PACKAGE_DIR + '/multivariate'


class CausalDataset(object):
    pass


class Sachs(CausalDataset):
    n_datasets = 1

    def __init__(self,
                 pair_id=None,
                 path=DATA_DIR,
                 double=False,
                 preprocessor=None):
        assert pair_id in list(range(1, self.n_datasets + 1))
        directory = DATA_DIR + '/' + self.folder_name

        data = np.load(directory + f'/data{pair_id}.npy')
        graph = np.load(directory + f'/DAG{pair_id}.npy')

        if preprocessor is not None:
            data = preprocessor.fit_transform(data)

        data, graph = torch.from_numpy(data), torch.from_numpy(graph)
        if double:  # default double
            data, graph = data.double(), graph.double()
        else:
            data, graph = data.float(), graph.float()
        self.data, self.graph = data, graph

    @property
    def folder_name(self):
        return 'Sachs/continuous'


class Syntren(CausalDataset):
    n_datasets = 10

    def __init__(self,
                 pair_id=None,
                 path=DATA_DIR,
                 double=False,
                 preprocessor=None):
        assert pair_id in list(range(1, self.n_datasets + 1))
        directory = DATA_DIR + '/' + self.folder_name

        data = np.load(directory + f'/data{pair_id}.npy')
        graph = np.load(directory + f'/DAG{pair_id}.npy')

        if preprocessor is not None:
            data = preprocessor.fit_transform(data)

        data, graph = torch.from_numpy(data), torch.from_numpy(graph)
        if double:  # default double
            data, graph = data.double(), graph.double()
        else:
            data, graph = data.float(), graph.float()
        self.data, self.graph = data, graph

    @property
    def folder_name(self):
        return 'Syntren_p20/syntren'


class Simulated(CausalDataset):
    n_datasets = 50

    def __init__(self,
                 pair_id=None,
                 path=DATA_DIR,
                 double=False,
                 preprocessor=None):
        assert pair_id in list(range(1, self.n_datasets + 1))
        directory = DATA_DIR + '/' + self.folder_name

        data = np.load(directory + f'/data{pair_id}.npy')
        graph = np.load(directory + f'/DAG{pair_id}.npy')

        if preprocessor is not None:
            data = preprocessor.fit_transform(data)

        data, graph = torch.from_numpy(data), torch.from_numpy(graph)
        if double:  # default double
            data, graph = data.double(), graph.double()
        else:
            data, graph = data.float(), graph.float()
        self.data, self.graph = data, graph

    @property
    def folder_name(self):
        raise NotImplementedError


class Simulated_ANM_ER_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-er-5'


class Simulated_ANM_SF_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-sf-5'


class Simulated_ANM_SF_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-sf-10'


class Simulated_ANM_ER_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-er-10'


class Simulated_ANM_SF_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-sf-20'


class Simulated_ANM_ER_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-er-20'


class Simulated_ANM_SF_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-sf-50'


class Simulated_ANM_ER_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-er-50'


class Simulated_ANM_SF_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-sf-100'


class Simulated_ANM_ER_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-er-100'


class Simulated_LSNM_ER_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-5'


class Simulated_LSNM_SF_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-5'


class Simulated_LSNM_SF_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-10'


class Simulated_LSNM_ER_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-10'


class Simulated_LSNM_SF_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-20'


class Simulated_LSNM_ER_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-20'


class Simulated_LSNM_SF_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-50'


class Simulated_LSNM_ER_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-50'


class Simulated_LSNM_SF_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-100'


class Simulated_LSNM_ER_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-100'


class Simulated_ANM_C_ER_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-er-5'


class Simulated_ANM_C_SF_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-sf-5'


class Simulated_ANM_C_SF_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-sf-10'


class Simulated_ANM_C_ER_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-er-10'


class Simulated_ANM_C_SF_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-sf-20'


class Simulated_ANM_C_ER_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-er-20'


class Simulated_ANM_C_SF_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-sf-50'


class Simulated_ANM_C_ER_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-er-50'


class Simulated_ANM_C_SF_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-sf-100'


class Simulated_ANM_C_ER_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/anm-c-er-100'


class Simulated_LSNM_C_ER_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-er-5'


class Simulated_LSNM_C_SF_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-sf-5'


class Simulated_LSNM_C_SF_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-sf-10'


class Simulated_LSNM_C_ER_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-er-10'


class Simulated_LSNM_C_SF_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-sf-20'


class Simulated_LSNM_C_ER_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-er-20'


class Simulated_LSNM_C_SF_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-sf-50'


class Simulated_LSNM_C_ER_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-er-50'


class Simulated_LSNM_C_SF_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-sf-100'


class Simulated_LSNM_C_ER_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-c-er-100'


class Simulated_GC_ER_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-er-5'


class Simulated_GC_SF_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-sf-5'


class Simulated_GC_SF_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-sf-10'


class Simulated_GC_ER_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-er-10'


class Simulated_GC_SF_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-sf-20'


class Simulated_GC_ER_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-er-20'


class Simulated_GC_SF_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-sf-50'


class Simulated_GC_ER_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-er-50'


class Simulated_GC_SF_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-sf-100'


class Simulated_GC_ER_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/gcastle-er-100'


class Simulated_LINGAM_ER_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-er-5'


class Simulated_LINGAM_SF_5(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-sf-5'


class Simulated_LINGAM_SF_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-sf-10'


class Simulated_LINGAM_ER_10(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-er-10'


class Simulated_LINGAM_SF_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-sf-20'


class Simulated_LINGAM_ER_20(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-er-20'


class Simulated_LINGAM_SF_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-sf-50'


class Simulated_LINGAM_ER_50(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-er-50'


class Simulated_LINGAM_SF_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-sf-100'


class Simulated_LINGAM_ER_100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lingam-er-100'


class Simulated_LSNM_SF_100_N100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-100[100]'


class Simulated_LSNM_ER_100_N100(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-100[100]'


class Simulated_LSNM_SF_100_N200(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-100[200]'


class Simulated_LSNM_ER_100_N200(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-100[200]'


class Simulated_LSNM_SF_100_N500(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-100[500]'


class Simulated_LSNM_ER_100_N500(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-100[500]'


class Simulated_LSNM_SF_100_N800(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-100[800]'


class Simulated_LSNM_ER_100_N800(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-100[800]'


class Simulated_LSNM_SF_100_N1000(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-100[1000]'


class Simulated_LSNM_ER_100_N1000(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-100[1000]'


class Simulated_LSNM_SF_100_N2000(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-100[2000]'


class Simulated_LSNM_ER_100_N2000(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-100[2000]'


class Simulated_LSNM_SF_100_N3000(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-sf-100[3000]'


class Simulated_LSNM_ER_100_N3000(Simulated):
    n_datasets = 50

    @property
    def folder_name(self):
        return 'Simulated/lsnm-er-100[3000]'
