from cleanup import courses, make_grid, generate_student_preferences, timetable_generator, calculate_bundle_value_by_hand
import random
import numpy as np
import math


def student(bundle, additive_prefs, substitutes, complements, timetable, overload_penalty = -7,
            free_days_marginal_values = [22, 14, 6, 0, 0], credit_units = [1 for i in range(30)], make_monotone = False):
    """
    This function simply calls the calculate_bundle_value_by_hand function to avoid having to parse all the parameters each time.
    """

    def sub_lists(l):
        lists = [[]]
        for i in range(len(l) + 1):
            for j in range(i):
                lists.append(l[j: i])
        return lists

    if (not make_monotone):

        return calculate_bundle_value_by_hand(bundle, timetable, credit_units = credit_units, additive_preferences= additive_prefs,
                                              timegap_penalty= -5, complements_sets= complements, substitutes_sets= substitutes, overload_penalty= overload_penalty, ignore_timegaps= True,
                                              free_days_marginal_values= free_days_marginal_values, verbose = False)
    else:
        # get the position of all the ones
        one_positions = [i for i in range(len(bundle)) if bundle[i] == 1]
        # get the indices of all the elements
        subsets = sub_lists(one_positions)
        max_subset_value = 0
        for subset in subsets:
            x = [0 for i in range(len(bundle))]
            for i in subset:
                x[i] = 1   # get the subset bundle in binary form

            subset_value = calculate_bundle_value_by_hand(x, timetable, credit_units = credit_units, additive_preferences= additive_prefs,
                                                          timegap_penalty= -5, complements_sets= complements, substitutes_sets= substitutes, overload_penalty= overload_penalty, ignore_timegaps= True,
                                                          free_days_marginal_values= free_days_marginal_values, verbose = False)

            max_subset_value = max(max_subset_value, subset_value)
        return max_subset_value


def create_initial_dataset(additive_prefs, substitutes, complements, timetable, credit_units = [1 for i in range(30)], overload_penalty=-7, free_days_marginal_values=[22, 14, 6, 0, 0],
                           n_samples_train=50, n_samples_test=400, n_courses=30, make_monotone=False, seed=None,
                                value_range = (250, math.inf), n_samples_train_range = 0, n_samples_test_range = 0, courses_per_sample = 5):
    """
    A function creating a dataset for a student of the specific size.

    Parameters:
    --------------------
    n_samples_train: int
        The number of samples in the training set
    n_samples_test: int
        The number of samples in the test set
    n_courses: int
        The number of courses
    make_monotone: True
        If True: It will force the student's preferences to be monotone, by calculating all subsets of any bundle.
    seed: int or None
        The seed to be used for all randomness
    value_range: (float, float) tuple
        The target value range for a portion of the samples.
    n_samples_train_range:  int
         That many train samples will be withing the value_range
    n_samples_test_range: int
        The number of test samples that will be within the value_range
    courses_per_sample: Int
        How many courses each query will have

    Returns:
    --------------------
    X_train: numpy matrix of 0-1s of shape (n_samples_train, n_courses)
        X_train[i]: A binary encoding of the i-th bundle in the training set
    y_train: numpy matrix of floats of shape (n_samples_train)
        y_train[i]: The float value of the i-th bundle in the training set
    X_test: numpy matrix of 0-1s of shape (n_samples_test, n_courses)
        X_test[i]: A binary encoding of the i-th bundle in the test set
    y_test: numpy matrix of floats of shape (n_samples_test)
        y_test[i]: The float value of the i-th bundle in the test set
    """

    if (seed):
        random.seed(seed)
        np.random.seed(seed)

    X = []
    y = []
    for i in range(n_samples_train):
        x = [0 for i in range(n_courses)]
        courses_in_sample = random.sample(
            [i for i in range(n_courses)], courses_per_sample)
        for j in courses_in_sample:
            x[j] = 1
        X.append(x)
        y.append(student(x, additive_prefs, substitutes, complements, timetable, overload_penalty,
                         free_days_marginal_values, credit_units, make_monotone))

    samples_in_range = 0
    samples_tested_train = 0
    while(samples_in_range < n_samples_train_range):
        samples_tested_train += 1
        x = [0 for i in range(n_courses)]
        courses_in_sample = random.sample(
            [i for i in range(n_courses)], courses_per_sample)
        for j in courses_in_sample:
            x[j] = 1
        y_candidate = student(x, additive_prefs, substitutes, complements, timetable, overload_penalty,
                              free_days_marginal_values, credit_units, make_monotone)
        if (y_candidate >= value_range[0] and y_candidate <= value_range[1]):
            y.append(y_candidate)
            X.append(x)
            samples_in_range += 1

    X_train = np.array(X)
    y_train = np.array(y)

    X = []
    y = []
    for i in range(n_samples_test):
        x = [0 for i in range(n_courses)]
        courses_in_sample = random.sample(
            [i for i in range(n_courses)], courses_per_sample)
        for j in courses_in_sample:
            x[j] = 1
        X.append(x)
        y.append(student(x, additive_prefs, substitutes, complements, timetable, overload_penalty,
                         free_days_marginal_values, credit_units, make_monotone))

    samples_in_range = 0
    samples_tested_test = 0
    while(samples_in_range < n_samples_test_range):
        samples_tested_test += 1
        x = [0 for i in range(n_courses)]
        courses_in_sample = random.sample(
            [i for i in range(n_courses)], courses_per_sample)
        for j in courses_in_sample:
            x[j] = 1
        y_candidate = student(x, additive_prefs, substitutes, complements, timetable, overload_penalty,
                              free_days_marginal_values, credit_units, make_monotone)
        if (y_candidate >= value_range[0] and y_candidate <= value_range[1]):
            y.append(y_candidate)
            X.append(x)
            samples_in_range += 1

    X_test = np.array(X)
    y_test = np.array(y)

#     print(f'In order to create {n_samples_train_range} train samples in the range ({value_range[0]},{value_range[1]}), {samples_tested_train} samples had to be checked!')

    return (X_train, y_train, X_test, y_test)


def create_multiple_datasets(timetable, credit_units=[1 for i in range(30)], n_courses=30, n_types=1, n_A_pop=8, n_A_npop=22,
                            number_of_favourites = 2, number_of_centers = 2, gr_height=6, gr_length=5, num_datasets=50, train_size=50,
                            validation_size=40, test_size=400,
                            ordinal_train=None, ordinal_validation=None, ordinal_test=None, ordinal_pool_size=1000, seed=42,
                            substitute_range=1, complement_range=1, complements_expected_value=0.4,
                            substitutes_expected_value=0.4, fixed_comp_values = True,
                            favourites_value_range=(40, 60), non_favourites_value_range=(80, 120), make_monotone=True,
                            same_student=False, disable_time_prefs=True, sample_value_range = (250, math.inf),
                                     train_size_range = 0, validation_size_range = 0, test_size_range = 0, courses_per_sample = 5):
    """
    A function generating multiple datasets, from the same or different students.

    Parameters:
    -----------------
    timetable: list of lists of lists
        timetable[i][j] The courses taught in the j-th timeslot of the i-th day
    credit_units: list of length number_of_courses
        credit_units[i]: The credit units of the i-th course
    n_courses: int
        The number of courses
    n_types: int
        The types of different students
    n_A_pop: int
        The number of popular courses
    n_A_npop: int
        The number of non-popular courses
    number_of_favourites: int
        The number of favourite courses for each student, drawn from the popular courses!
    number_of_centers: int
        The number of complement/substitute centers for each student, drawn from her favourite courses!
    gr_height: int
        The height of the course grid
    gr_length: int
        The length of the course grid
    num_datasets: int
        The number of datasets to generate
    train_size: int
        The number of samples in the training set
    validataion_size: int
        The nubmer of samples in the validation set of each dataset
    test_size: int
        The number of test samples in each dataset
    ordinal_train: int or None
        If it is not None: The number of ordinal samples in each training dataset
    ordinal_validation: int or None
        If it is not None: The number of ordinal samples in each validation dataset
    ordinal_test: int or None
        If it is not None: The number of ordinal samples in each test dataset
    ordinal_pool_size: int
        The pool of samples to create in order to draw the ordinal questions
    seed: int
        The random seed to use
    substitute_range: int
        The range of substitutes in the grid
    complement_range: int
        The range of complements in the grid
    complements_expected_value: float
        The expected bonus of having a second item in a complement set
    substitutes_expected_value: float
        The expected penalty of having a second item in a substitute set
    fixed_comp_values: Bool
        if True: The students' complements/substitutes values will be exactly equal to their expected values.
    favourites_value_range: (int, int) tuple
        The value of a favourite course for each student is drawn uniformly from that range
    non_favourites_value_range: (int, int) tuple
        The value of a favourite course for each student is drawn uniformly from that range
    make_monotone: True
        If True: It will force the student's preferences to be monotone, by calculating all subsets of any bundle.
    same_student: Bool
        If True: All the datasets will be drawn from the preferences of a single student.
    disable_time_prefs: Bool
        If true: Students have no time preferences.
    seed: int or None
        The seed to be used for all randomness
    value_range: (float, float) tuple
        The target value range for a portion of the samples.
    n_samples_train_range:  int
         That many train samples will be withing the value_range
    n_samples_test_range: int
        The number of test samples that will be within the value_range
    courses_per_sample: Int
        How many courses each query will have
    """
    X_train_list = []
    y_train_list = []
    X_test_list = []
    y_test_list = []
    X_validation_list = []
    y_validation_list = []

    X_train_list_ordinal = []
    y_train_list_ordinal = []
    X_test_list_ordinal = []
    y_test_list_ordinal = []
    X_validation_list_ordinal = []
    y_validation_list_ordinal = []

    course_buckets = courses(n_types=n_types, n_courses=n_courses,
                             n_A_pop=n_A_pop, n_A_npop=n_A_npop, seed=seed)
    grid = make_grid(n_courses=n_courses, height=gr_height,
                     width=gr_length, seed=seed)
    if (same_student):
        (additive_prefs, substitutes, complements, overload_penalty, timegap_penalty,
         free_days_marginal_values) = generate_student_preferences(course_buckets,
                                                                   grid, complement_range=complement_range,
                                                                   substitute_range=substitute_range,
                                                                   complements_expected_value=complements_expected_value,
                                                                   substitutes_expected_value=substitutes_expected_value,
                                                                   seed=seed, disable_time_prefs=disable_time_prefs,
                                                                   favourites_value_range=favourites_value_range,
                                                                   non_favourites_value_range=non_favourites_value_range)

    for i in range(num_datasets):
        if (not same_student):
            (additive_prefs, substitutes, complements, overload_penalty, timegap_penalty,
             free_days_marginal_values) = generate_student_preferences(course_buckets,
                                                                       grid, complement_range=complement_range,
                                                                       substitute_range=substitute_range,
                                                                       complements_expected_value=complements_expected_value,
                                                                       substitutes_expected_value=substitutes_expected_value,
                                                                       disable_time_prefs=disable_time_prefs,
                                                                       seed=seed + i,
                                                                       favourites_value_range=favourites_value_range,
                                                                       non_favourites_value_range=non_favourites_value_range)

        (X_train_val, y_train_val, X_test, y_test) = create_initial_dataset(additive_prefs, substitutes, complements,
                                                                            timetable, credit_units, overload_penalty,
                                                                            free_days_marginal_values,
                                                                            n_samples_train=train_size + validation_size,
                                                                            n_samples_test=test_size, seed=seed + i,
                                                                            n_courses=n_courses,
                                                                            make_monotone=make_monotone,
                                                                            value_range= sample_value_range,
                                                                            n_samples_train_range= train_size_range + validation_size_range,
                                                                            n_samples_test_range = test_size_range, courses_per_sample = courses_per_sample)

        if (ordinal_train is not None):  # construct the ordinal set of queries
            (X_ordinal_pool, y_ordinal_pool, _, _) = create_initial_dataset(additive_prefs, substitutes, complements,
                                                                            timetable, credit_units,
                                                                            overload_penalty, free_days_marginal_values,
                                                                            n_samples_train=ordinal_pool_size,
                                                                            n_samples_test=0, seed=None,
                                                                            n_courses=n_courses,
                                                                            make_monotone=make_monotone, courses_per_sample = courses_per_sample)

            np.random.seed(seed)
            ordinal_indexes_train = np.random.randint(
                low=0, high=ordinal_pool_size, size=2 * (ordinal_train))
            ordinal_indexes_test = np.random.randint(
                low=0, high=ordinal_pool_size, size=2 * (ordinal_test))
            ordinal_indexes_val = np.random.randint(
                low=0, high=ordinal_pool_size, size=2 * (ordinal_validation))

            X_train_ordinal = []
            y_train_ordinal = []
            X_test_ordinal = []
            y_test_ordinal = []
            X_val_ordinal = []
            y_val_ordinal = []

            for j in range(ordinal_train):
                X_1 = X_ordinal_pool[ordinal_indexes_train[2 * j]]
                X_2 = X_ordinal_pool[ordinal_indexes_train[2 * j + 1]]

                y_1 = y_ordinal_pool[ordinal_indexes_train[2 * j]]
                y_2 = y_ordinal_pool[ordinal_indexes_train[2 * j + 1]]

                X_train_ordinal.append((X_1, X_2))
                if (y_1 >= y_2):
                    y_train_ordinal.append(1)
                else:
                    y_train_ordinal.append(0)

            for j in range(ordinal_test):
                X_1 = X_ordinal_pool[ordinal_indexes_test[2 * j]]
                X_2 = X_ordinal_pool[ordinal_indexes_test[2 * j + 1]]
                X_test_ordinal.append((X_1, X_2))
                y_1 = y_ordinal_pool[ordinal_indexes_test[2 * j]]
                y_2 = y_ordinal_pool[ordinal_indexes_test[2 * j + 1]]
                if (y_1 >= y_2):
                    y_test_ordinal.append(1)
                else:
                    y_test_ordinal.append(0)

            for j in range(ordinal_validation):
                X_1 = X_ordinal_pool[ordinal_indexes_val[2 * j]]
                X_2 = X_ordinal_pool[ordinal_indexes_val[2 * j + 1]]

                y_1 = y_ordinal_pool[ordinal_indexes_val[2 * j]]
                y_2 = y_ordinal_pool[ordinal_indexes_val[2 * j + 1]]

                X_val_ordinal.append((X_1, X_2))
                if (y_1 >= y_2):
                    y_val_ordinal.append(1)
                else:
                    y_val_ordinal.append(0)

            X_train_list_ordinal.append(X_train_ordinal)
            y_train_list_ordinal.append(y_train_ordinal)
            X_test_list_ordinal.append(X_test_ordinal)
            y_test_list_ordinal.append(y_test_ordinal)
            X_validation_list_ordinal.append(X_val_ordinal)
            y_validation_list_ordinal.append(y_val_ordinal)

        X_train = X_train_val[:(train_size + train_size_range)]
        y_train = y_train_val[:(train_size + train_size_range)]
        X_val = X_train_val[(train_size + train_size_range):]
        y_val = y_train_val[(train_size + train_size_range):]

        X_train_list.append(X_train)
        y_train_list.append(y_train)
        X_validation_list.append(X_val)
        y_validation_list.append(y_val)
        X_test_list.append(X_test)
        y_test_list.append(y_test)

    if (ordinal_train is not None):
        X_train_list_ordinal = np.array(X_train_list_ordinal)
        X_test_list_ordinal = np.array(X_test_list_ordinal)
        X_validation_list_ordinal = np.array(X_validation_list_ordinal)
        y_train_list_ordinal = np.array(y_train_list_ordinal)
        y_test_list_ordinal = np.array(y_test_list_ordinal)
        y_validation_list_ordinal = np.array(y_validation_list_ordinal)

        return (X_train_list, y_train_list, X_validation_list, y_validation_list, X_test_list, y_test_list,
                X_train_list_ordinal, y_train_list_ordinal, X_validation_list_ordinal, y_validation_list_ordinal,
                X_test_list_ordinal, y_test_list_ordinal)
    return (X_train_list, y_train_list, X_validation_list, y_validation_list, X_test_list, y_test_list)


def create_multiple_students(number_of_students = 100, number_of_courses = 30,  number_of_popular = 8, mean_number_of_favourites = 4.87, number_of_centers = 2, gr_height = 5, gr_width = 6, seed = 42,
                            base_budget = 1, complement_range = 1, substitute_range = 1, complements_expected_value = 0.4,
                             substitutes_expected_value = 0.4, favourites_value_range = (80, 120), non_favourites_value_range = (40, 60), complements_decay = 0.8, substitutes_decay = 0.8,
                             disable_time_prefs = True, fixed_comp_values = True):
    """
    A function generating a complete student list for a run of Course Match, including the student preferences and their budgets.

    Parameters:
    --------------
    number_of_students: int
        The number of students
    number_of_courses: int
        The number of courses
    number_of_popular: int
        The number of popular courses
    nubmer_of_favourites: int
        The number of favourite courses. They will be drawn exclusively from the popular courses!
    number_of_centers: int
        The number of complement/substitute centers. They will be drawn exclusively from the student's favourite courses!
    gr_height: int
        The height of the grid of the thematic map
    gr_width: int
        The width of the grid of the thempatic map
    seed: int
        The random seeed to set
    base_budget: float
        The mean budget of a student
    budget_spread: float
        The spread of the students' budgets
    complement_range: int
        Defines the range within which the courses are complements to the center
    substitute_range: int
        Defines the range within which the courses are substitutes to the center
    complements_expected_value: float
        The expected bonus for getting more courses from the same set of complements
    substitutes_expected_value: float
        The expected penalty for getting more courses from the same set of substitutes
    favourites_value_range: (int, int) tuple
        The value of a favourite course for each student is drawn uniformly from that range
    non_favourites_value_range: (int, int) tuple
        The value of a favourite course for each student is drawn uniformly from that range
    disable_time_prefs: Bool
        If true: Students have no time preferences.
    fixed_comp_values: Bool
        If true: The students complement bonus/substitute penalty is always equal to its expected value.


    """
    if(seed is not None):  # just so that we don't by mistake set a random random seed
        random.seed(seed)
        np.random.seed(seed)

    course_buckets = courses(n_types = 1, n_courses= number_of_courses,
                n_A_pop = number_of_popular, n_A_npop = number_of_courses - number_of_popular, seed = seed)  # create the course buckets (same for all students)
    grid = make_grid(n_courses = number_of_courses, height = gr_height, width = gr_width, seed = seed)   # create the course grid (same for all students)
    beta = min(1 / (5-1), 1 / number_of_students)
    budget_step = beta / number_of_students

    student_list = []
    for i in range(number_of_students):
        random_number = np.random.uniform(low = 0, high = 1)
        if random_number <= mean_number_of_favourites - math.floor(mean_number_of_favourites):
            number_of_favourites = int(math.ceil(mean_number_of_favourites))
        else:
            number_of_favourites = int(math.floor(mean_number_of_favourites))

        (additive_prefs, substitutes, complements, overload_penalty, timegap_penalty, free_days_marginal_values) = generate_student_preferences(course_buckets,
                        grid, complement_range = complement_range, substitute_range = substitute_range, favourite_courses_number= [number_of_favourites],
                        non_favourite_courses_number= [number_of_courses - number_of_favourites], non_favourite_courses_probabilities= [1],
                        complements_expected_value = complements_expected_value, substitutes_expected_value = substitutes_expected_value,
                        seed = seed + i, disable_time_prefs= disable_time_prefs, favourites_value_range= favourites_value_range, center_numbers= [number_of_centers], center_number_probabilities= [1],
                        non_favourites_value_range = non_favourites_value_range, complements_decay= complements_decay, substitutes_decay= substitutes_decay, fixed_comp_values= fixed_comp_values)

        budget = 1 + i * budget_step

        student_list.append((additive_prefs, substitutes, complements, overload_penalty, timegap_penalty, free_days_marginal_values, budget))

    return student_list



def create_multiple_datasets_enhanced(timetable, credit_units=[1 for i in range(30)], number_of_courses = 30, n_types=1, number_of_popular = 8,
                            number_of_favourites = 2, number_of_centers = 2, gr_height=6, gr_width=5, num_datasets=50, train_size=50,
                            validation_size=40, test_size=400,
                            ordinal_train=None, ordinal_validation=None, ordinal_test=None, ordinal_pool_size=1000, seed=42,
                            substitute_range=1, complement_range=1, complements_expected_value=0.4, complements_decay = 0.8, substitutes_decay = 0.8,
                            substitutes_expected_value=0.4, fixed_comp_values = True,
                            favourites_value_range=(40, 60), non_favourites_value_range=(80, 120), make_monotone=True,
                            disable_time_prefs=True, sample_value_range = (250, math.inf),
                            train_size_range = 0, validation_size_range = 0, test_size_range = 0, courses_per_sample = 5):
    X_train_list = []
    y_train_list = []
    X_test_list = []
    y_test_list = []
    X_validation_list = []
    y_validation_list = []

    X_train_list_ordinal = []
    y_train_list_ordinal = []
    X_test_list_ordinal = []
    y_test_list_ordinal = []
    X_validation_list_ordinal = []
    y_validation_list_ordinal = []

    student_list = create_multiple_students(number_of_students= num_datasets, number_of_courses= number_of_courses, number_of_popular= number_of_popular, number_of_favourites= number_of_favourites,
        number_of_centers= number_of_centers, gr_height= gr_height, gr_width= gr_width, seed = seed, base_budget= 1, budget_spread= 0.1, complement_range= complement_range, substitute_range= substitute_range,
        complements_expected_value= complements_expected_value, substitutes_expected_value= substitutes_expected_value, favourites_value_range= favourites_value_range, non_favourites_value_range= non_favourites_value_range,
        complements_decay= complements_decay, substitutes_decay= substitutes_decay, disable_time_prefs= disable_time_prefs, fixed_comp_values= fixed_comp_values)  # create the students for the whole dataset

    # (additive_prefs, substitutes, complements, overload_penalty, timegap_penalty, free_days_marginal_values, budget)

    for (i, (additive_prefs, substitutes, complements, overload_penalty, timegap_penalty, free_days_marginal_values, budget)) in enumerate(student_list):
        (X_train_val, y_train_val, X_test, y_test) = create_initial_dataset(additive_prefs, substitutes, complements,
                                                                            timetable, credit_units, overload_penalty,
                                                                            free_days_marginal_values,
                                                                            n_samples_train=train_size + validation_size,
                                                                            n_samples_test=test_size, seed=seed + i,
                                                                            n_courses=number_of_courses,
                                                                            make_monotone=make_monotone,
                                                                            value_range= sample_value_range,
                                                                            n_samples_train_range= train_size_range + validation_size_range,
                                                                            n_samples_test_range = test_size_range, courses_per_sample = courses_per_sample)

        if (ordinal_train is not None):  # construct the ordinal set of queries
            (X_ordinal_pool, y_ordinal_pool, _, _) = create_initial_dataset(additive_prefs, substitutes, complements,
                                                                            timetable, credit_units,
                                                                            overload_penalty, free_days_marginal_values,
                                                                            n_samples_train=ordinal_pool_size,
                                                                            n_samples_test=0, seed=None,
                                                                            n_courses=number_of_courses,
                                                                            make_monotone=make_monotone, courses_per_sample = courses_per_sample)

            np.random.seed(seed)
            ordinal_indexes_train = np.random.randint(
                low=0, high=ordinal_pool_size, size=2 * (ordinal_train))
            ordinal_indexes_test = np.random.randint(
                low=0, high=ordinal_pool_size, size=2 * (ordinal_test))
            ordinal_indexes_val = np.random.randint(
                low=0, high=ordinal_pool_size, size=2 * (ordinal_validation))

            X_train_ordinal = []
            y_train_ordinal = []
            X_test_ordinal = []
            y_test_ordinal = []
            X_val_ordinal = []
            y_val_ordinal = []

            for j in range(ordinal_train):
                X_1 = X_ordinal_pool[ordinal_indexes_train[2 * j]]
                X_2 = X_ordinal_pool[ordinal_indexes_train[2 * j + 1]]

                y_1 = y_ordinal_pool[ordinal_indexes_train[2 * j]]
                y_2 = y_ordinal_pool[ordinal_indexes_train[2 * j + 1]]

                X_train_ordinal.append((X_1, X_2))
                if (y_1 >= y_2):
                    y_train_ordinal.append(1)
                else:
                    y_train_ordinal.append(0)

            for j in range(ordinal_test):
                X_1 = X_ordinal_pool[ordinal_indexes_test[2 * j]]
                X_2 = X_ordinal_pool[ordinal_indexes_test[2 * j + 1]]
                X_test_ordinal.append((X_1, X_2))
                y_1 = y_ordinal_pool[ordinal_indexes_test[2 * j]]
                y_2 = y_ordinal_pool[ordinal_indexes_test[2 * j + 1]]
                if (y_1 >= y_2):
                    y_test_ordinal.append(1)
                else:
                    y_test_ordinal.append(0)

            for j in range(ordinal_validation):
                X_1 = X_ordinal_pool[ordinal_indexes_val[2 * j]]
                X_2 = X_ordinal_pool[ordinal_indexes_val[2 * j + 1]]

                y_1 = y_ordinal_pool[ordinal_indexes_val[2 * j]]
                y_2 = y_ordinal_pool[ordinal_indexes_val[2 * j + 1]]

                X_val_ordinal.append((X_1, X_2))
                if (y_1 >= y_2):
                    y_val_ordinal.append(1)
                else:
                    y_val_ordinal.append(0)

            X_train_list_ordinal.append(X_train_ordinal)
            y_train_list_ordinal.append(y_train_ordinal)
            X_test_list_ordinal.append(X_test_ordinal)
            y_test_list_ordinal.append(y_test_ordinal)
            X_validation_list_ordinal.append(X_val_ordinal)
            y_validation_list_ordinal.append(y_val_ordinal)

        X_train = X_train_val[:(train_size + train_size_range)]
        y_train = y_train_val[:(train_size + train_size_range)]
        X_val = X_train_val[(train_size + train_size_range):]
        y_val = y_train_val[(train_size + train_size_range):]

        X_train_list.append(X_train)
        y_train_list.append(y_train)
        X_validation_list.append(X_val)
        y_validation_list.append(y_val)
        X_test_list.append(X_test)
        y_test_list.append(y_test)

    if (ordinal_train is not None):
        X_train_list_ordinal = np.array(X_train_list_ordinal)
        X_test_list_ordinal = np.array(X_test_list_ordinal)
        X_validation_list_ordinal = np.array(X_validation_list_ordinal)
        y_train_list_ordinal = np.array(y_train_list_ordinal)
        y_test_list_ordinal = np.array(y_test_list_ordinal)
        y_validation_list_ordinal = np.array(y_validation_list_ordinal)

        return (X_train_list, y_train_list, X_validation_list, y_validation_list, X_test_list, y_test_list,
                X_train_list_ordinal, y_train_list_ordinal, X_validation_list_ordinal, y_validation_list_ordinal,
                X_test_list_ordinal, y_test_list_ordinal)
    return (X_train_list, y_train_list, X_validation_list, y_validation_list, X_test_list, y_test_list)


if __name__ == '__main__':
    timetable = timetable_generator()
    a = create_multiple_datasets(timetable=timetable, num_datasets=10)
    print(a)
