{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Instructions\n",
    "\n",
    "This Jupyter Notebook contains a minimal code snippet for reproducing the experimental results.\n",
    "\n",
    "The **first two cells** are used to generate plots for experiments **without noise**, and the **next two cells** are used to generate plots for experiments **with noise**. \n",
    "\n",
    "To reproduce a specific experiment, select the appropriate pair of cells based on whether noise is present, modify the sampling rate and/or instance type as needed, and then run the two cells sequentially."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 393
    },
    "id": "2VO3JIxGLRQf",
    "outputId": "e7b0578b-6b0d-4208-ae4b-0149e1469941"
   },
   "outputs": [],
   "source": [
    "# experiments without noise\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import algorithms as algo\n",
    "import instances as ins\n",
    "import math\n",
    "\n",
    "length = 20000\n",
    "time_unit = 0.005\n",
    "mean_cost = 50\n",
    "card_cost = 500\n",
    "beta_list = [0, 0.2, 0.5, 0.7, 0.9]\n",
    "supremum = 100\n",
    "xi = ((supremum/card_cost) ** (1/5))\n",
    "iterations = 100\n",
    "sample_rate = 0.1  # choose the sample rate\n",
    "\n",
    "results = {\n",
    "    \"Normal\": {\n",
    "        \"PFSUM\": [],\n",
    "        \"online_no_sample\": [],\n",
    "        \"naive_sample\": [],\n",
    "        \"three_two_sample\": [],\n",
    "        \"improved_sample\": []\n",
    "    },\n",
    "    \"Pareto\": {\n",
    "        \"PFSUM\": [],\n",
    "        \"online_no_sample\": [],\n",
    "        \"naive_sample\": [],\n",
    "        \"three_two_sample\": [],\n",
    "        \"improved_sample\": []\n",
    "    }\n",
    "}\n",
    "\n",
    "offline_opt_performance_ = []\n",
    "PFSUM_performance = []\n",
    "online_no_sample_performance = []\n",
    "naive_sample_performance = []\n",
    "three_two_sample_performance = []\n",
    "improved_sample_performance = []\n",
    "\n",
    "price_type_list = [\"Normal\", \"Pareto\"]\n",
    "\n",
    "for j in range(len(price_type_list)):\n",
    "  for beta in beta_list:\n",
    "\n",
    "    PFSUM_value = 0\n",
    "    PFSUM_unscaled_value = 0\n",
    "    online_no_sample_value = 0\n",
    "    naive_sample_value = 0\n",
    "    three_two_sample_value = 0\n",
    "    improved_sample_value = 0\n",
    "\n",
    "    for i in range(iterations):\n",
    "\n",
    "      # Choose the class of instances\n",
    "\n",
    "      instance_set = ins.instance_generator(length, \"Commuter\", 1, price_type_list[j], mean_cost, supremum) # commuter\n",
    "      # instance_set = ins.instance_generator(length, \"Exponential\", 2, price_type_list[j], mean_cost, supremum) # occasional with mean 2\n",
    "      # instance_set = ins.instance_generator(length, \"Exponential\", 40, price_type_list[j], mean_cost, supremum) # occasional with mean 40\n",
    "      # instance_set = ins.intermittent_arrivals_generator(length, 500, 50, mean_cost, price_type_list[j], supremum) # cluster\n",
    "\n",
    "      sample_set = ins.instance_sampler(sample_rate, instance_set)\n",
    "\n",
    "      instance = [sum(x) for x in instance_set]\n",
    "      sample = [sum(x) for x in sample_set]\n",
    "      scaled_sample = [x/sample_rate for x in sample]\n",
    "      OPT = algo.offline_opt(instance, beta, card_cost, time_unit)\n",
    "\n",
    "      PFSUM_value += algo.PFSUM(instance, beta, card_cost, time_unit, ins.prediction_generator(scaled_sample, math.floor(1/time_unit)))/OPT\n",
    "      online_no_sample_value += algo.online_no_sample(instance, beta, card_cost, time_unit)/OPT\n",
    "      naive_sample_value += algo.naive_sample(instance, beta, card_cost, time_unit, sample, sample_rate)/OPT\n",
    "      three_two_sample_value += algo.three_two_sample(instance, beta, card_cost, time_unit, sample, xi, sample_rate)/OPT\n",
    "      improved_sample_value += algo.improved_sample(instance, beta, card_cost, time_unit, sample, xi, sample_rate)/OPT\n",
    "\n",
    "\n",
    "    results[price_type_list[j]][\"PFSUM\"].append(PFSUM_value/100)\n",
    "    results[price_type_list[j]][\"online_no_sample\"].append(online_no_sample_value/100)\n",
    "    results[price_type_list[j]][\"naive_sample\"].append(naive_sample_value/100)\n",
    "    results[price_type_list[j]][\"three_two_sample\"].append(three_two_sample_value/100)\n",
    "    results[price_type_list[j]][\"improved_sample\"].append(improved_sample_value/100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "yu43SNZFM-NX"
   },
   "outputs": [],
   "source": [
    "# create plot\n",
    "\n",
    "x = beta_list\n",
    "fig, axes = plt.subplots(1, 2, figsize=(10, 4), sharey=True)\n",
    "\n",
    "axes[0].plot(x, results[\"Normal\"][\"PFSUM\"], marker='s',color='green', label=\"PFSUM\")\n",
    "axes[0].plot(x, results[\"Normal\"][\"online_no_sample\"], marker='x',color='purple', label=\"Counter\")\n",
    "axes[0].plot(x, results[\"Normal\"][\"three_two_sample\"], marker='^',color='blue', label=\"Algorithm 4.2\")\n",
    "axes[0].plot(x, results[\"Normal\"][\"improved_sample\"], marker='*',color='red', label=\"Algorithm 4.3\")\n",
    "axes[0].plot(x, results[\"Normal\"][\"naive_sample\"], marker='D',color='orange', label=\"Naïve\")\n",
    "\n",
    "\n",
    "axes[1].plot(x, results[\"Pareto\"][\"PFSUM\"], marker='s',color='green', label=\"PFSUM\")\n",
    "axes[1].plot(x, results[\"Pareto\"][\"online_no_sample\"], marker='x',color='purple', label=\"Counter\")\n",
    "axes[1].plot(x, results[\"Pareto\"][\"three_two_sample\"], marker='^',color='blue', label=\"Algorithm 4.2\")\n",
    "axes[1].plot(x, results[\"Pareto\"][\"improved_sample\"], marker='*',color='red', label=\"Algorithm 4.3\")\n",
    "axes[1].plot(x, results[\"Pareto\"][\"naive_sample\"], marker='D',color='orange', label=\"Naive\")\n",
    "\n",
    "handles, labels = axes[0].get_legend_handles_labels()\n",
    "fig.legend(handles, labels, loc=\"center left\", bbox_to_anchor=(1.02, 0.5))\n",
    "axes[0].set_title(\"Normal\")\n",
    "axes[1].set_title(\"Pareto\")\n",
    "axes[0].set_xlabel(\"Discount Rate\")\n",
    "axes[1].set_xlabel(\"Discount Rate\")\n",
    "axes[0].set_ylabel(\"Competitive Ratio\")\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "4CFqdQDWOBln"
   },
   "outputs": [],
   "source": [
    "# experiments with noise\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import algorithms as algo\n",
    "import instances as ins\n",
    "\n",
    "length = 20000\n",
    "time_unit = 0.005\n",
    "mean_cost = 50\n",
    "card_cost = 500\n",
    "beta = 0\n",
    "perturbation_probabilities = [0, 0.1, 0.5, 0.9]\n",
    "supremum = 100\n",
    "xi = ((supremum/card_cost) ** (1/5))\n",
    "iterations = 100\n",
    "sample_rate = 0.1  # choose the sample rate\n",
    "\n",
    "results_perturbation = {\n",
    "    \"Normal\": {\n",
    "        \"PFSUM\": [],\n",
    "        \"online_no_sample\": [],\n",
    "        \"naive_sample\": [],\n",
    "        \"three_two_sample\": [],\n",
    "        \"improved_sample\": []\n",
    "    },\n",
    "    \"Pareto\": {\n",
    "        \"PFSUM\": [],\n",
    "        \"online_no_sample\": [],\n",
    "        \"naive_sample\": [],\n",
    "        \"three_two_sample\": [],\n",
    "        \"improved_sample\": []\n",
    "    }\n",
    "}\n",
    "\n",
    "\n",
    "\n",
    "offline_opt_performance_ = []\n",
    "PFSUM_performance = []\n",
    "online_no_sample_performance = []\n",
    "naive_sample_performance = []\n",
    "three_two_sample_performance = []\n",
    "improved_sample_performance = []\n",
    "\n",
    "price_type_list = [\"Normal\", \"Pareto\"]\n",
    "\n",
    "for j in range(len(price_type_list)):\n",
    "  for prob in perturbation_probabilities:\n",
    "\n",
    "    PFSUM_value = 0\n",
    "    PFSUM_unscaled_value = 0\n",
    "    online_no_sample_value = 0\n",
    "    naive_sample_value = 0\n",
    "    three_two_sample_value = 0\n",
    "    improved_sample_value = 0\n",
    "\n",
    "    for i in range(iterations):\n",
    "      # Choose the class of instances\n",
    "\n",
    "      instance_set = ins.instance_generator(length, \"Commuter\", 1, price_type_list[j], mean_cost, supremum) # commuter\n",
    "      # instance_set = ins.instance_generator(length, \"Exponential\", 2, price_type_list[j], mean_cost, supremum) # occasional with mean 2\n",
    "      # instance_set = ins.instance_generator(length, \"Exponential\", 40, price_type_list[j], mean_cost, supremum) # occasional with mean 40\n",
    "      # instance_set = ins.intermittent_arrivals_generator(length, 500, 50, mean_cost, price_type_list[j], supremum) # cluster\n",
    "\n",
    "      perturbed_instance_set = ins.noisy_instance_generator(instance_set, price_type_list[j], mean_cost, prob)\n",
    "      sample_set = ins.instance_sampler(sample_rate, perturbed_instance_set)\n",
    "\n",
    "      instance = [sum(x) for x in instance_set]\n",
    "      sample = [sum(x) for x in sample_set]\n",
    "      scaled_sample = [x/sample_rate for x in sample]\n",
    "      OPT = algo.offline_opt(instance, beta, card_cost, time_unit)\n",
    "\n",
    "      PFSUM_value += algo.PFSUM(instance, beta, card_cost, time_unit, ins.prediction_generator(scaled_sample, math.floor(1/time_unit)))/OPT\n",
    "      online_no_sample_value += algo.online_no_sample(instance, beta, card_cost, time_unit)/OPT\n",
    "      naive_sample_value += algo.naive_sample(instance, beta, card_cost, time_unit, sample, sample_rate)/OPT\n",
    "      three_two_sample_value += algo.three_two_sample(instance, beta, card_cost, time_unit, sample, xi, sample_rate)/OPT\n",
    "      improved_sample_value += algo.improved_sample(instance, beta, card_cost, time_unit, sample, xi, sample_rate)/OPT\n",
    "\n",
    "\n",
    "    results_perturbation[price_type_list[j]][\"PFSUM\"].append(PFSUM_value/100)\n",
    "    results_perturbation[price_type_list[j]][\"online_no_sample\"].append(online_no_sample_value/100)\n",
    "    results_perturbation[price_type_list[j]][\"naive_sample\"].append(naive_sample_value/100)\n",
    "    results_perturbation[price_type_list[j]][\"three_two_sample\"].append(three_two_sample_value/100)\n",
    "    results_perturbation[price_type_list[j]][\"improved_sample\"].append(improved_sample_value/100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "CvwltNZmOooP"
   },
   "outputs": [],
   "source": [
    "x = perturbation_probabilities\n",
    "fig, axes = plt.subplots(1, 2, figsize=(10, 4), sharey=True)\n",
    "\n",
    "axes[0].plot(x, results_perturbation[\"Normal\"][\"PFSUM\"], marker='s',color='green', label=\"PFSUM\")\n",
    "axes[0].plot(x, results_perturbation[\"Normal\"][\"online_no_sample\"], marker='x',color='purple', label=\"Counter\")\n",
    "axes[0].plot(x, results_perturbation[\"Normal\"][\"three_two_sample\"], marker='^',color='blue', label=\"Algorithm 4.2\")\n",
    "axes[0].plot(x, results_perturbation[\"Normal\"][\"improved_sample\"], marker='*',color='red', label=\"Algorithm 4.3\")\n",
    "axes[0].plot(x, results_perturbation[\"Normal\"][\"naive_sample\"], marker='D',color='orange', label=\"Naïve\")\n",
    "\n",
    "\n",
    "axes[1].plot(x, results_perturbation[\"Pareto\"][\"PFSUM\"], marker='s',color='green', label=\"PFSUM (with scaled sample)\")\n",
    "axes[1].plot(x, results_perturbation[\"Pareto\"][\"online_no_sample\"], marker='x',color='purple', label=\"Counter\")\n",
    "axes[1].plot(x, results_perturbation[\"Pareto\"][\"three_two_sample\"], marker='^',color='blue', label=\"Algorithm 4.2\")\n",
    "axes[1].plot(x, results_perturbation[\"Pareto\"][\"improved_sample\"], marker='*',color='red', label=\"Algorithm 4.3\")\n",
    "axes[1].plot(x, results_perturbation[\"Pareto\"][\"naive_sample\"], marker='D',color='orange', label=\"Naïve\")\n",
    "\n",
    "handles, labels = axes[0].get_legend_handles_labels()\n",
    "fig.legend(handles, labels, loc=\"center left\", bbox_to_anchor=(1.02, 0.5))\n",
    "axes[0].set_title(\"Normal\")\n",
    "axes[1].set_title(\"Pareto\")\n",
    "axes[0].set_xlabel(\"Perturbation Probability\")\n",
    "axes[1].set_xlabel(\"Perturbation Probability\")\n",
    "axes[0].set_ylabel(\"Competitive Ratio\")\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "colab": {
   "provenance": []
  },
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
