{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mRunning cells with '/opt/homebrew/bin/python3.12' requires the ipykernel package.\n",
      "\u001b[1;31mRun the following command to install 'ipykernel' into the Python environment. \n",
      "\u001b[1;31mCommand: '/opt/homebrew/bin/python3.12 -m pip install ipykernel -U --user --force-reinstall'"
     ]
    }
   ],
   "source": [
    "from preference_generator import create_all_instances, calculate_total_demand, create_noisy_model_student_list, calculate_single_student_demand,calculate_true_bundle_value\n",
    "from preference_generator_utils import load_obj\n",
    "import numpy as np"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Generating Student Instances"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Generating Some True Student Lists"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mRunning cells with '/opt/homebrew/bin/python3.12' requires the ipykernel package.\n",
      "\u001b[1;31mRun the following command to install 'ipykernel' into the Python environment. \n",
      "\u001b[1;31mCommand: '/opt/homebrew/bin/python3.12 -m pip install ipykernel -U --user --force-reinstall'"
     ]
    }
   ],
   "source": [
    "# The preferences that were fitted to the Budish and Kessler (2022) data: \n",
    "# For 6 popular courses -> mean number of favourites = 3.85\n",
    "# For 9 popular courses -> mean number of favourites = 2.6\n",
    "# Maximum budget deviation beta was set to 0.04 as in the Rubenstein 2022 paper\n",
    "# In our paper, we report results using a supply ratio of 1.1, 1.25 and 1.5\n",
    "\n",
    "true_student_lists_all_instances, capacities_all_instances, timetables_all_instances = create_all_instances(number_of_instances = 50, number_of_courses= 25, supply_ratio= 1.25, number_of_popular= 9, mean_number_of_favourites= 2.6, maximum_budget_deviation_beta= 0.04, save_results= True, save_folder= \"instances\")\n",
    "true_student_lists_all_instances, capacities_all_instances, timetables_all_instances = create_all_instances(number_of_instances = 50, number_of_courses= 25, supply_ratio= 1.25, number_of_popular= 6, mean_number_of_favourites= 3.85, maximum_budget_deviation_beta= 0.04, save_results= True, save_folder= \"instances\")\n",
    "true_student_lists_all_instances, capacities_all_instances, timetables_all_instances = create_all_instances(number_of_instances = 50, number_of_courses= 25, supply_ratio= 1.25, number_of_popular= 9, mean_number_of_favourites= 6.1, maximum_budget_deviation_beta= 0.04, additive_preferences= True,save_results= True, save_folder= \"instances\")"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Loading a problem instance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mRunning cells with '/opt/homebrew/bin/python3.12' requires the ipykernel package.\n",
      "\u001b[1;31mRun the following command to install 'ipykernel' into the Python environment. \n",
      "\u001b[1;31mCommand: '/opt/homebrew/bin/python3.12 -m pip install ipykernel -U --user --force-reinstall'"
     ]
    }
   ],
   "source": [
    "def load_instance(instance_number, supply_ratio = 1.25, number_of_popular = 9, large_grid = True, additive_preferences = False,instance_folder = \"instances\"):\n",
    "    \"\"\"\n",
    "    A simple function to load a single instance, given its number, the supply ration, and the number of pupular courses\n",
    "    \"\"\"\n",
    "\n",
    "    true_student_list = load_obj(f'{instance_folder}/true_student_lists_sr_{supply_ratio}_popular_{number_of_popular}_lg_{large_grid}_additive_preferences_{additive_preferences}')[instance_number]\n",
    "    capacities = np.load(f'{instance_folder}/capacities_all_runs_sr_{supply_ratio}_popular_{number_of_popular}_lg_{large_grid}_additive_preferences_{additive_preferences}.npy')[instance_number]\n",
    "    timetable = load_obj(f'{instance_folder}/timetables_sr_{supply_ratio}_popular_{number_of_popular}_lg_{large_grid}_additive_preferences_{additive_preferences}')[instance_number]\n",
    "\n",
    "    return true_student_list, capacities, timetable\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mRunning cells with '/opt/homebrew/bin/python3.12' requires the ipykernel package.\n",
      "\u001b[1;31mRun the following command to install 'ipykernel' into the Python environment. \n",
      "\u001b[1;31mCommand: '/opt/homebrew/bin/python3.12 -m pip install ipykernel -U --user --force-reinstall'"
     ]
    }
   ],
   "source": [
    "true_student_list, capacities, timetable = load_instance(instance_number = 2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mRunning cells with '/opt/homebrew/bin/python3.12' requires the ipykernel package.\n",
      "\u001b[1;31mRun the following command to install 'ipykernel' into the Python environment. \n",
      "\u001b[1;31mCommand: '/opt/homebrew/bin/python3.12 -m pip install ipykernel -U --user --force-reinstall'"
     ]
    }
   ],
   "source": [
    "# position i of the student list contains the true preferences, and the budget, of the i-th student \n",
    "# format: Additive preferences, list of substitute courses, list of complement courses, 3 time parameters are all set to completely off (and designed to model schedule preferences), and finally budget \n",
    "\n",
    "print(true_student_list[2])"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Creating a \"Noisy\" version of a student list"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mRunning cells with '/opt/homebrew/bin/python3.12' requires the ipykernel package.\n",
      "\u001b[1;31mRun the following command to install 'ipykernel' into the Python environment. \n",
      "\u001b[1;31mCommand: '/opt/homebrew/bin/python3.12 -m pip install ipykernel -U --user --force-reinstall'"
     ]
    }
   ],
   "source": [
    "number_of_popular = 9 \n",
    "\n",
    "# these are the noise profiles that were used in our paper and matched as closely as possible the noise profile in the Budish and Kessler 2022 paper. \n",
    "if number_of_popular == 9:\n",
    "    noise_parameter_dictionary = {'noisy_forget_base': 0.5, 'noisy_forget_adjustments': 0.48, 'noisy_base_std': 23, 'noisy_adj_std': 0.2}\n",
    "\n",
    "elif number_of_popular == 6:\n",
    "    noise_parameter_dictionary = {'noisy_forget_base': 0.5, 'noisy_forget_adjustments': 0.4825, 'noisy_base_std': 17, 'noisy_adj_std': 0.2}\n",
    "\n",
    "\n",
    "noisy_student_list = create_noisy_model_student_list(student_list = true_student_lists_all_instances[2], model_type = 'PairwiseAdjustmentsNoisy', \n",
    "                                                    seed = 1213, model_param_dictionary = noise_parameter_dictionary)"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Getting the demands of individual students for a given price vector"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mRunning cells with '/opt/homebrew/bin/python3.12' requires the ipykernel package.\n",
      "\u001b[1;31mRun the following command to install 'ipykernel' into the Python environment. \n",
      "\u001b[1;31mCommand: '/opt/homebrew/bin/python3.12 -m pip install ipykernel -U --user --force-reinstall'"
     ]
    }
   ],
   "source": [
    "price_vector = np.array([0.2 for _ in range(25)])  # A price vector for the individual courses\n",
    "credit_units = np.array([1 for _ in range(25)])  # the credit units of inidividual courses. For this problem instance, all courses are worth 1 credit unit, and each student wants up to 5 Credit Units\n",
    "\n",
    "# Demands with respect to their noisy reports in the mechanism\n",
    "total_demand, individual_demands = calculate_total_demand(prices = np.array([0.2 for _ in range(25)]), student_profiles = noisy_student_list, course_timetable = [[] for  _ in range(5)],\n",
    "                            credit_units = np.array([1 for _ in range(25)]), model_type = 'PairwiseAdjustmentsNoisy', credit_units_per_student = 5, return_individual_demands = True)\n",
    "    \n",
    "# Demands with respect to their true reports (note: these preferences are incompatible with the CM GUI language)\n",
    "total_demand_true, individual_demands_true = calculate_total_demand(prices = np.array([0.2 for _ in range(25)]), student_profiles = true_student_list, course_timetable = [[] for  _ in range(5)],\n",
    "                        credit_units = np.array([1 for _ in range(25)]), model_type = 'True', credit_units_per_student = 5, return_individual_demands = True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mRunning cells with '/opt/homebrew/bin/python3.12' requires the ipykernel package.\n",
      "\u001b[1;31mRun the following command to install 'ipykernel' into the Python environment. \n",
      "\u001b[1;31mCommand: '/opt/homebrew/bin/python3.12 -m pip install ipykernel -U --user --force-reinstall'"
     ]
    }
   ],
   "source": [
    "student_number = 7 \n",
    "\n",
    "demand_noisy = calculate_single_student_demand(prices = np.array([0.2 for _ in range(25)]), \n",
    "    student_profile = noisy_student_list[student_number],\n",
    "    course_timetable = timetable,  # the course timetable affects the feasibility constraints, as students cannot take courses that are scheduled at the same time\n",
    "    credit_units = [1 for i in range(25)],  # the credit units associated with each course \n",
    "    model_type = 'PairwiseAdjustmentsNoisy',  # this is the model type of the GUI reporting language \n",
    "    credit_units_per_student = 5,   # by changing this the maximum number of credit units that the student can take changes, and thus so does the maximum number of courses \n",
    "    budget = 1.3  # with this parameter you can change the budget of the student. \n",
    "    )\n",
    "\n",
    "demand_true = calculate_single_student_demand(prices = np.array([0.2 for _ in range(25)]), \n",
    "    student_profile = true_student_list[student_number],\n",
    "    course_timetable = timetable,\n",
    "    credit_units = [1 for i in range(25)],\n",
    "    model_type = 'True',   # this is the model type of the True student preferences\n",
    "    credit_units_per_student = 5,    \n",
    "    budget = 1.3)"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Querying the true value of a student for a bundle"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mRunning cells with '/opt/homebrew/bin/python3.12' requires the ipykernel package.\n",
      "\u001b[1;31mRun the following command to install 'ipykernel' into the Python environment. \n",
      "\u001b[1;31mCommand: '/opt/homebrew/bin/python3.12 -m pip install ipykernel -U --user --force-reinstall'"
     ]
    }
   ],
   "source": [
    "for i in range(len(individual_demands)): \n",
    "    value_bundle_noisy_preferences = calculate_true_bundle_value(bundle = individual_demands[i], student_preferences = true_student_list[i], timetable = [[] for  _ in range(5)], make_monotone = True)\n",
    "    value_bundle_true_preferences = calculate_true_bundle_value(bundle = individual_demands_true[i], student_preferences = true_student_list[i], timetable = [[] for  _ in range(5)], make_monotone = True)\n",
    "\n",
    "    print(f'For student {i}, value for the optimal bundle {value_bundle_true_preferences} and the one she got under noisy reports: {value_bundle_noisy_preferences}')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "mlca_dq",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.13"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "a13a3921ff2a036888c80ce6d9a76381b84f0c8446943c3cdab5acf61d8dcce5"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
