{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "import os\n",
    "os.chdir(\"..\")\n",
    "import matplotlib.pyplot as plt\n",
    "import sys\n",
    "import numpy as np\n",
    "import os\n",
    "from math import log\n",
    "import random\n",
    "import time\n",
    "import pickle\n",
    "from numpy.linalg import pinv\n",
    "from arm_class import ArmGaussian\n",
    "from LinUCB_class import PolicyLinUCB\n",
    "from dLinUCB_class import DynamicLinUCB\n",
    "from D_LinUCB_class import DLinUCB\n",
    "from SW_LinUCB_class import SWLinUCB\n",
    "from environment_class import Environment\n",
    "from simulator_class import Simulator\n",
    "from utils import plot_regret, scatter_abrupt, scatter_smooth, action_check, get_B_T_smooth, detection_sorted"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# General parameters for all the experiments\n",
    "delta = 0.01 # Probability of being outside the confidence interval\n",
    "s = 1  # Bound on the theta_star parameter\n",
    "lambda_ = 0.1 # Regularisation parameter\n",
    "q = 5 # Diplaying the quantile (in %)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "def experiment_policies(n_mc, option):\n",
    "    mab = Environment(d, theta, sigma_noise, verbose)\n",
    "    simulator = Simulator(mab, theta, policies, k, d, steps, bp, verbose)\n",
    "    if option == \"par\":\n",
    "        avgRegret, qRegret, QRegret = simulator.run_multiprocessing(n_process, steps, n_mc, q, t_saved)\n",
    "        return avgRegret, qRegret, QRegret\n",
    "    else:\n",
    "        avgRegret, qRegret, QRegret, timedic, theta_hat, action_check = simulator.run(steps, \n",
    "                                                               n_mc, q, n_scat, t_saved)\n",
    "        return avgRegret, qRegret, QRegret, timedic, theta_hat, action_check\n",
    "\n",
    "def data_from_experiment(n_mc, option):\n",
    "    if option == \"par\":\n",
    "        avgRegret, qRegret, QRegret = experiment_policies(n_mc, option)\n",
    "        data = [[policy, avgRegret[policy], qRegret[policy],\n",
    "                QRegret[policy]] for policy in avgRegret]\n",
    "        return data\n",
    "    else:\n",
    "        avgRegret, qRegret, QRegret, timedic, theta_hat, action_check = experiment_policies(n_mc, '')\n",
    "        data = [[policy, avgRegret[policy], qRegret[policy],\n",
    "                QRegret[policy]] for policy in avgRegret]\n",
    "        return data, theta_hat, action_check, timedic"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Creating the folder out_ABRUPT_NEW\n",
      "out_ABRUPT_NEW already exists\n",
      "Creating the folder detection_out\n"
     ]
    }
   ],
   "source": [
    "# The saved files should be located in the saved/ folder\n",
    "# The output images would be located in the out/ folder\n",
    "# If the folders are not created they must be created to run the code without bugs\n",
    "\n",
    "path = os.getcwd()\n",
    "out_dir = 'out_ABRUPT_NEW'\n",
    "saved_data_path = 'out_ABRUPT_NEW'\n",
    "detection_folder = 'detection_out'\n",
    "\n",
    "if not os.path.exists(path + '/' + out_dir):\n",
    "    os.mkdir(path + '/' + out_dir)\n",
    "    print('Creating the folder %s' %out_dir)\n",
    "else:\n",
    "    print(\"%s already exists\" %out_dir)\n",
    "    \n",
    "if not os.path.exists(path + '/' + saved_data_path):\n",
    "    os.mkdir(path + '/' + saved_data_path)\n",
    "    print('Creating the folder %s' %saved_data_path)\n",
    "else:\n",
    "    print(\"%s already exists\" %saved_data_path)\n",
    "    \n",
    "if not os.path.exists(path + '/' + detection_folder):\n",
    "    os.mkdir(path + '/' + detection_folder)\n",
    "    print('Creating the folder %s' %detection_folder)\n",
    "else:\n",
    "    print(\"%s already exists\" %detection_folder)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def save_file(filename, var):\n",
    "    with open(saved_data_path + '/' + str(filename) + '.pkl', 'wb') as f:\n",
    "        pickle.dump(var, f)\n",
    "\n",
    "def load_file(filename):\n",
    "    with open(saved_data_path + '/' + str(filename) + '.pkl', 'rb') as f:\n",
    "        res = pickle.load(f)\n",
    "        return res\n",
    "\n",
    "def save_file_from_folder(folder, filename, var):\n",
    "    with open(str(folder) + '/' + str(filename) + '.pkl', 'wb') as f:\n",
    "        pickle.dump(var, f)\n",
    "        \n",
    "def load_file_from_folder(folder, filename):\n",
    "    with open(str(folder) + '/' + str(filename) + '.pkl', 'rb') as f:\n",
    "        res = pickle.load(f)\n",
    "        return res"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "B_T value: 5.414213562373095\n",
      "Sigma value for the experimenxt: 1\n",
      "gamma: 0.9941173985691094\n",
      "tau: 169.99281895061446\n"
     ]
    }
   ],
   "source": [
    "# PARAMETERS\n",
    "d = 2  # dimension of the problem\n",
    "k = 50  # number of arms at each step\n",
    "steps = 6000  # number of steps for the experiment\n",
    "n_scat = 1000 # How frequently should we plot the true parameter\n",
    "t_saved = None\n",
    "\n",
    "# When saving only a part of the trajectory rather than all the points uncomment the 2 following lines\n",
    "# number_t_saved = steps//10\n",
    "# t_saved = np.int_(np.linspace(0, steps - 1, number_t_saved))\n",
    "\n",
    "alpha = 1\n",
    "sigma_noise = 1 # Square root of the variance of the noise\n",
    "verbose = False\n",
    "q = 5 # 5 percent quantiles used\n",
    "\n",
    "### Description of the experiment\n",
    "theta = np.array([1,  0]) # Starting point of the true unknown parameter\n",
    "bp = {1001: np.array([-1,  0]), 2001: np.array([0,  1]), 3001: np.array([0,  -1])} # Descrition of the breakpoints\n",
    "\n",
    "B_T = 4 + np.sqrt(2) # Computed manually\n",
    "print('B_T value:', B_T)\n",
    "print('Sigma value for the experimenxt:', sigma_noise)\n",
    "\n",
    "gamma  = 1 - (B_T/(d*steps))**(2/3) # Optimal Value to minimize the asymptotical regret\n",
    "tau = (d*steps/B_T)**(2/3) # Optimal Value to minimize the asymptotical regret\n",
    "print('gamma:', gamma)\n",
    "print('tau:', tau)\n",
    "\n",
    "policies = [DLinUCB(d, delta, alpha, lambda_, s, gamma, '', sm = False, sigma_noise = sigma_noise, verbose=verbose),\n",
    "            SWLinUCB(d, delta, alpha, lambda_, s, tau, '', sm=False, sigma_noise=sigma_noise, verbose=verbose),\n",
    "            DynamicLinUCB(d, delta, alpha, lambda_, s, tau=200, name='', filename = 'abrupt_exp_final',\n",
    "                          sm=True, sigma_noise = sigma_noise, delta_2=0.01, tilde_delta=0.002, verbose=False, \n",
    "                          omniscient=False),\n",
    "            PolicyLinUCB(d, delta, alpha, lambda_, s, '', sm = True, sigma_noise = sigma_noise, verbose=verbose),\n",
    "            PolicyLinUCB(d, delta, alpha, lambda_, s, '-OR', sm = True, sigma_noise = sigma_noise, verbose=verbose,\n",
    "                        omniscient = True)\n",
    "           ]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Unparallelized experiments"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  0%|          | 0/2 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0 D-LinUCB [0. 0.] [1 0]\n",
      "1000 D-LinUCB [0.90407206 0.1054755 ] [1 0]\n",
      "2000 D-LinUCB [-1.02211949 -0.14260033] [-1  0]\n",
      "3000 D-LinUCB [-0.08412444  0.95132123] [0 1]\n",
      "4000 D-LinUCB [-0.15844839 -0.95974805] [ 0 -1]\n",
      "5000 D-LinUCB [ 0.09348665 -1.04377718] [ 0 -1]\n",
      "0 SW-LinUCB [0. 0.] [1 0]\n",
      "1000 SW-LinUCB [0.89756872 0.03494319] [1 0]\n",
      "2000 SW-LinUCB [-1.04402242 -0.0985694 ] [-1  0]\n",
      "3000 SW-LinUCB [0.14258225 1.00912332] [0 1]\n",
      "4000 SW-LinUCB [-0.00712314 -0.95596182] [ 0 -1]\n",
      "5000 SW-LinUCB [-0.00182952 -1.00290445] [ 0 -1]\n",
      "0 dLinUCB [0. 0.] [1 0]\n",
      "1000 dLinUCB [ 0.97233228 -0.06281797] [1 0]\n",
      "2000 dLinUCB [ 0.15324808 -0.01428524] [-1  0]\n",
      "3000 dLinUCB [0.27400385 0.45069433] [0 1]\n",
      "4000 dLinUCB [0.11762648 0.05913625] [ 0 -1]\n",
      "5000 dLinUCB [ 0.20261139 -0.08633962] [ 0 -1]\n",
      "0 LinUCB [0. 0.] [1 0]\n",
      "1000 LinUCB [ 0.9625944  -0.11897597] [1 0]\n",
      "2000 LinUCB [ 0.11132243 -0.03088358] [-1  0]\n",
      "3000 LinUCB [0.28036457 0.46675616] [0 1]\n",
      "4000 LinUCB [0.0894421  0.02300438] [ 0 -1]\n",
      "5000 LinUCB [ 0.22275791 -0.14055766] [ 0 -1]\n",
      "0 LinUCB-OR [0. 0.] [1 0]\n",
      "1000 LinUCB-OR [1.03826766 0.01634168] [1 0]\n",
      "2000 LinUCB-OR [-1.04460851  0.03357497] [-1  0]\n",
      "3000 LinUCB-OR [0.00814999 0.97247328] [0 1]\n",
      "4000 LinUCB-OR [ 0.01895529 -0.99568921] [ 0 -1]\n",
      "5000 LinUCB-OR [ 0.01701812 -1.00448521] [ 0 -1]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 50%|█████     | 1/2 [00:51<00:51, 51.31s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Creation of a new model at time:1198\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 2/2 [01:41<00:00, 50.98s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "-- Building data out of the experiments ---\n",
      "--- Data built ---\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "data, theta_hat, action_check_val, timedic = data_from_experiment(n_mc=2, option = '')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'D-LinUCB': 25.196383476257324, 'SW-LinUCB': 20.035295009613037, 'dLinUCB': 19.525344371795654, 'LinUCB': 18.756324529647827, 'LinUCB-OR': 18.44198441505432}\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Plot on a single experiment\n",
    "print(timedic) # Computation time for the different policies\n",
    "action_check(action_check_val, 0) # To see the available actions at one step\n",
    "plot_regret(data, t_saved, filename = None, log=False, qtl=True, loc=2, font=13, bp = bp, bp_2 = {})\n",
    "scatter_abrupt(theta_hat, None, theta, bp, loc=0, font=10, circle=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 100 indepent experiments without parallelization, should take around half an hour.\n",
    "data_100, theta_hat_100, action_check_val_100, timedic_100 = data_from_experiment(n_mc=100, option = '')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plot average on 100 independent experiments\n",
    "print(timedic_100) # Computation time for the different policies\n",
    "plot_regret(data_100, t_saved, filename = None, log=False, qtl=True, loc=2, font=13, bp = bp, bp_2 = {1157:''})\n",
    "scatter_abrupt(theta_hat_100, None, theta, bp, loc=0, font=10, circle=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# For simplifying the plot and removing unwanted policies\n",
    "theta_hat_100_partial = {}\n",
    "theta_hat_100_partial['D-LinUCB'] = theta_hat_100['D-LinUCB']\n",
    "theta_hat_100_partial['SW-LinUCB'] = theta_hat_100['SW-LinUCB']\n",
    "theta_hat_100_partial['dLinUCB'] = theta_hat_100['dLinUCB']\n",
    "theta_hat_100_partial['LinUCB'] = theta_hat_100['LinUCB']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_regret(data_100, t_saved, filename = None, log=False, qtl=True, loc=2, font=13, bp = bp, bp_2 = {1157:''})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# when removing the LinUCB-OR policy\n",
    "scatter_abrupt(theta_hat_100_partial, None, theta, bp, loc=0, font=13, circle=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Parallelized experiments"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {
    "collapsed": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Creation of a new model at time:1104\n",
      "Creation of a new model at time:1152\n",
      "Creation of a new model at time:1125\n",
      "Creation of a new model at time:1191\n",
      "Creation of a new model at time:1136\n",
      "Creation of a new model at time:1095\n",
      "Creation of a new model at time:1123\n",
      "Creation of a new model at time:1091\n",
      "Creation of a new model at time:1122\n",
      "Creation of a new model at time:1136\n",
      "Creation of a new model at time:1150\n",
      "Creation of a new model at time:1179\n",
      "Creation of a new model at time:1064\n",
      "Creation of a new model at time:1135\n",
      "Creation of a new model at time:1179\n",
      "Creation of a new model at time:1114\n",
      "Creation of a new model at time:1167\n",
      "Creation of a new model at time:1174\n",
      "Creation of a new model at time:1094\n",
      "Creation of a new model at time:1139\n",
      "Creation of a new model at time:1185\n",
      "Creation of a new model at time:1198\n",
      "Creation of a new model at time:3205\n",
      "Creation of a new model at time:1145\n",
      "Creation of a new model at time:1105\n",
      "Creation of a new model at time:1090\n",
      "Creation of a new model at time:1139\n",
      "Creation of a new model at time:1068\n",
      "Creation of a new model at time:1139\n",
      "Creation of a new model at time:1116\n",
      "Creation of a new model at time:1104\n",
      "Creation of a new model at time:1181\n",
      "Creation of a new model at time:1171\n",
      "Creation of a new model at time:1186\n",
      "Creation of a new model at time:3257\n",
      "Creation of a new model at time:1170\n",
      "Creation of a new model at time:1311\n",
      "Creation of a new model at time:1168\n",
      "Creation of a new model at time:1108\n",
      "Creation of a new model at time:1197\n",
      "Creation of a new model at time:1123\n",
      "Creation of a new model at time:1125\n",
      "Creation of a new model at time:1153\n",
      "Creation of a new model at time:1217\n",
      "Creation of a new model at time:1113\n",
      "Creation of a new model at time:1162\n",
      "Creation of a new model at time:1143\n",
      "Creation of a new model at time:1105\n",
      "Creation of a new model at time:1138\n",
      "Creation of a new model at time:1135\n",
      "Creation of a new model at time:1159\n",
      "Creation of a new model at time:1098\n",
      "Creation of a new model at time:1162\n",
      "Creation of a new model at time:1152\n",
      "Creation of a new model at time:1136\n",
      "Creation of a new model at time:1156\n",
      "Creation of a new model at time:1172\n",
      "Creation of a new model at time:1119\n",
      "Creation of a new model at time:1160\n",
      "Creation of a new model at time:1189\n",
      "Creation of a new model at time:1174\n",
      "Creation of a new model at time:1149\n",
      "Creation of a new model at time:1179\n",
      "Creation of a new model at time:1169\n",
      "Creation of a new model at time:1164\n",
      "Creation of a new model at time:1161\n",
      "Creation of a new model at time:1129\n",
      "Creation of a new model at time:1135\n",
      "Creation of a new model at time:1143\n",
      "Creation of a new model at time:1172\n",
      "Creation of a new model at time:1136\n",
      "Creation of a new model at time:1431\n",
      "Creation of a new model at time:3213\n",
      "Creation of a new model at time:1095\n",
      "Creation of a new model at time:1180\n",
      "Creation of a new model at time:1083\n",
      "Creation of a new model at time:1122\n",
      "Creation of a new model at time:1132\n",
      "Creation of a new model at time:3277\n",
      "Creation of a new model at time:1137\n",
      "Creation of a new model at time:1148\n",
      "Creation of a new model at time:1187\n",
      "Creation of a new model at time:1092\n",
      "Creation of a new model at time:1196\n",
      "Creation of a new model at time:1103\n",
      "Creation of a new model at time:1217\n",
      "Creation of a new model at time:1164\n",
      "Creation of a new model at time:3163\n",
      "Creation of a new model at time:1166\n",
      "Creation of a new model at time:1119\n",
      "Creation of a new model at time:1215\n",
      "Creation of a new model at time:1166\n",
      "Creation of a new model at time:3130\n",
      "Creation of a new model at time:1185\n",
      "Creation of a new model at time:1190\n",
      "Creation of a new model at time:1211\n",
      "Creation of a new model at time:1121\n",
      "Creation of a new model at time:1224\n",
      "Creation of a new model at time:1117\n",
      "Creation of a new model at time:1153\n",
      "Creation of a new model at time:1106\n",
      "Total Time Experiment:  299.2663106918335\n",
      "Conversion time: 1.1676139831542969\n",
      "Post-Processing Time: 0.027048110961914062\n"
     ]
    }
   ],
   "source": [
    "n_process = 11\n",
    "\n",
    "file = 'abrupt_par_exp_final'\n",
    "policies = [DLinUCB(d, delta, alpha, lambda_, s, gamma, '', sm = False, sigma_noise = sigma_noise, verbose=verbose),\n",
    "            SWLinUCB(d, delta, alpha, lambda_, s, tau, '', sm=False, sigma_noise=sigma_noise, verbose=verbose),\n",
    "            DynamicLinUCB(d, delta, alpha, lambda_, s, tau=200, name='', filename = file,\n",
    "                          sm=True, sigma_noise = sigma_noise, delta_2=0.01, tilde_delta=0.002, verbose=False, \n",
    "                          omniscient=False),\n",
    "            PolicyLinUCB(d, delta, alpha, lambda_, s, '', sm = True, sigma_noise = sigma_noise, verbose=verbose),\n",
    "            PolicyLinUCB(d, delta, alpha, lambda_, s, '-OR', sm = True, sigma_noise = sigma_noise, verbose=verbose,\n",
    "                        omniscient = True)\n",
    "           ]\n",
    "\n",
    "n_expe = 100\n",
    "data_par = data_from_experiment(n_expe,'par')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "95\n",
      "0\n",
      "6\n"
     ]
    }
   ],
   "source": [
    "with open(\"detection_out/\" + file, 'r') as f:\n",
    "    detect = list(map(lambda x:x.strip(),f.readlines()))\n",
    "\n",
    "detect = [int(el) for el in detect]\n",
    "list_1 = [el for el in detect if el < 2000]\n",
    "list_2 = [el for el in detect if el < 3000 and el >= 2000]\n",
    "list_3 = [el for el in detect if  el >= 3000]\n",
    "\n",
    "# Number of detections\n",
    "print(len(list_1))\n",
    "print(len(list_2))\n",
    "print(len(list_3))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Percentage of detection first breakpoint: 95.0\n",
      "Percentage of detection second breakpoint: 0.0\n",
      "Percentage of detection third breakpoint: 6.0\n"
     ]
    }
   ],
   "source": [
    "print('Percentage of detection first breakpoint:', len(list_1)/n_expe*100)\n",
    "print('Percentage of detection second breakpoint:',len(list_2)/n_expe*100)\n",
    "print('Percentage of detection third breakpoint:',len(list_3)/n_expe*100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1150.93684211\n",
      "3207.5\n"
     ]
    }
   ],
   "source": [
    "print(np.mean(list_1)) # Average detection step of the first breakpoint\n",
    "print(np.mean(list_3)) # Average detection step of the third breakpoint"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
