{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1322e997",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np;\n",
    "import scipy\n",
    "import random\n",
    "import numpy.random as ra;\n",
    "import numpy.linalg as la;\n",
    "import matplotlib.pyplot as plt\n",
    "import sklearn\n",
    "from sklearn import preprocessing\n",
    "from scipy.stats import bernoulli\n",
    "import kld as kl\n",
    "import theta_1_set_gen as big_theta\n",
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5780cd08",
   "metadata": {},
   "outputs": [],
   "source": [
    "random.seed(20)\n",
    "np.random.seed(21)\n",
    "\n",
    "T=5000  #Length of the time horizon \n",
    "epsilon=0.2 #epsilon parameter \n",
    "itr=5000 #total number of iterations (number of Monte Carlo runs)\n",
    "\n",
    "\n",
    "var=.5 #variance of the Gaussian noise\n",
    "path_len=5 #length of  path connected nodes\n",
    "\n",
    "d_list=[i for i in range(10,30,5)] #list of lenghts of line graph. Graph length = 10,15,20,25\n",
    "tau_list=[i for i in range(10,50,10)] #list of change points. Change point = 10,20,30,40\n",
    "\n",
    "#----------list containing candidate post change parameter (unknown to the algorithms, but common to all algorithms) \n",
    "#----------for various length of the graph------------------------------------------------------------------------\n",
    "\n",
    "theta_1_list=np.load(\"theta_1_list.npy\",allow_pickle=True).tolist()\n",
    "\n",
    "\n",
    "#list to store stopping time of algorithm \n",
    "l_stop_rand_d=[]\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1e532252",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "#_________________________________________Bandit Loop for URS______________________________________________________________\n",
    "\n",
    "beta=30 #the choice of beta for which the false alarm of URS change detector is less than 1%\n",
    "start=time.time()\n",
    "print(\"URS\")\n",
    "\n",
    "\n",
    "for (i,d) in enumerate(d_list):\n",
    "    print(\"d=\",d)\n",
    "    theta_1=theta_1_list[i]\n",
    "    theta_1=theta_1.reshape(theta_1.shape[0],)\n",
    "    \n",
    "    #post change parameter set, generated by big_theta module.\n",
    "    theta_1_set=big_theta.theta_1_set(d,path_len)\n",
    "    \n",
    "    #All post change parameters are normalised to 1.\n",
    "    theta_1_set=sklearn.preprocessing.normalize(theta_1_set,norm='l2',axis=0)\n",
    "    \n",
    "    #pointy action set\n",
    "    Action_set=np.identity(d)\n",
    "    \n",
    "    #Total number of actions \n",
    "    K=Action_set.shape[1]\n",
    "    \n",
    "    #WLOG, for synthetic experiments, we set pre-change parameter to zero.\n",
    "    theta_not=np.zeros((d,))\n",
    "    \n",
    "    #List to store stopping time of an algorithm for a fixed change point and fixed length of the graph. \n",
    "    l_stop=[]\n",
    "    \n",
    "    for (idx,tau) in enumerate(tau_list):\n",
    "        print(\"tau=\",tau)\n",
    "        \n",
    "\n",
    "\n",
    "#----------------------------- Start of Monte Carlo run----------------------------\n",
    "        \n",
    "        for j in range(itr):\n",
    "            \n",
    "            b1=0        \n",
    "            theta_hat=0\n",
    "            V=0\n",
    "            \n",
    "        #___________________________________Start of Time Horizon______________________________________________________________\n",
    "\n",
    "\n",
    "            for t in range(1,T+1):\n",
    "                \n",
    "                #Stopping criteria to check whehter the change has occurred. \n",
    "                if np.max(V) >=beta or t==T:\n",
    "                    change=t\n",
    "                    l_stop.append(change)\n",
    "                    break\n",
    "                    \n",
    "                #At time step 1, play a random action to get initial observation\n",
    "                \n",
    "                if (t<=1):\n",
    "                    \n",
    "                    # Random action at time step 1\n",
    "                    A1_idx=random.randrange(0,K,1)\n",
    "                    A=Action_set[:,A1_idx]\n",
    "                    #arm.append(A1_idx)\n",
    "                    A=A.reshape(A.shape[0],)\n",
    "                    \n",
    "                    #Observation at time step 1\n",
    "                    X=np.random.normal(0,var,1)\n",
    "                    \n",
    "                    \n",
    "                    \n",
    "\n",
    "                        #calculation of Q^{(1)} statistics -- V is Q^{(1)} -- first sample update\n",
    "                    b1=X*A\n",
    "                    temp4=np.dot((theta_1_set).T,A)                 \n",
    "                    temp3=(2*np.dot((theta_1_set).T,b1))-temp4**(2)  \n",
    "\n",
    "                    V=temp3\n",
    "                                    \n",
    "                    \n",
    "\n",
    "\n",
    "                else:\n",
    "                    \n",
    "                    #Pick action unifoormly at random \n",
    "                    play_idx=random.randrange(0,K,1)\n",
    "                    #arm.append(play_idx)\n",
    "\n",
    "                                #Play an action\n",
    "                    A=Action_set[:,play_idx]\n",
    "                    A=A.reshape(A.shape[0],)\n",
    "\n",
    "                                #Get an observation\n",
    "\n",
    "                    if t<tau:\n",
    "                        X=np.random.normal(0,var,1)\n",
    "                    else:\n",
    "                        X=np.random.normal(np.dot(A,theta_1),var,1)\n",
    "\n",
    "                        #Calculation of Q^{(1)} statistics -- V is Q^{(1)} \n",
    "                        \n",
    "#Calculating g(X_t|A_t):\n",
    "                    b1=X*A\n",
    "                    temp4=np.dot((theta_1_set).T,A)\n",
    "                    temp3=(2*np.dot((theta_1_set).T,b1))-temp4**(2)\n",
    "\n",
    "#Recursice update of Q^{(1)}:\n",
    "\n",
    "                    V=np.maximum(0,V+temp3) \n",
    "\n",
    "#Store the stopping times for each length of the graph and change point. \n",
    "#We then save stopping time as .npy file and use the .npy file in a separate .ipynb for visualisation.\n",
    "    l_stop_rand_d.append(l_stop)\n",
    "\n",
    "np.save(\"l_stop_rand_d.npy\",l_stop_rand_d)\n",
    "end=time.time()\n",
    "\n",
    "print(\"Time Taken by URS=\",end-start)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2354de54",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.8.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
