{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "129dfc82",
   "metadata": {},
   "outputs": [],
   "source": [
    "import librosa\n",
    "import librosa.display\n",
    "import numpy as np\n",
    "import numpy.linalg as la;\n",
    "import matplotlib.pyplot as plt\n",
    "%matplotlib inline\n",
    "plt.style.use('Solarize_Light2')\n",
    "prop_cycle = plt.rcParams['axes.prop_cycle']\n",
    "colors = prop_cycle.by_key()['color']\n",
    "blue = colors[1]\n",
    "red = colors[5]\n",
    "import tensorflow as tf\n",
    "import os, sys\n",
    "import sound_tools\n",
    "import utils\n",
    "from sklearn.model_selection import train_test_split\n",
    "import pickle\n",
    "from scipy.stats import bernoulli\n",
    "import itertools \n",
    "from tqdm import tqdm\n",
    "import time "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fee06dac",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "model_trained=[tf.keras.models.load_model('**path_to_Slider_Trained**/Slider_Trained/model/%d'%i) for i in range(0,8,2)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "93ea1825",
   "metadata": {},
   "outputs": [],
   "source": [
    "import glob\n",
    "import kld_1 as kl\n",
    "import Matrix_Norm as mn\n",
    "import random\n",
    "\n",
    "random.seed(20)\n",
    "\n",
    "Action_set=np.identity(10)\n",
    "K=4 #No of arms\n",
    "\n",
    "#Note: Here, action set is an identity matrix of dimension 4x4. \n",
    "\n",
    "#------------------------Each column corresponds to a particular machine id.--------------------\n",
    "#------------------------Column 0 corresponds to id_00, column 1 corresponds to id_02,------------------\n",
    "#------------------------column 1 corresponds to id_02 , and column 1 corresponds to id_02---------------------\n",
    "\n",
    "\n",
    "\n",
    "var=0.5 #Variance of the Gaussian distribution.\n",
    "beta=250 #Value of beta chosen such that false alarm is less than 1%\n",
    "T=60 #Total length of the audio clip. Each time step, t \\in [1,...,T] is of 10s duration.\n",
    "epsilon=0.2 #epsilon parameter\n",
    "\n",
    "\n",
    "#Parameters used for feature extraction (Spectrogram) from a 10s audio clip.\n",
    "\n",
    "n_mels = 64 #number of mel-bands\n",
    "frames = 5 #number of frames\n",
    "n_fft = 1024 #FFT used\n",
    "hop_length = 512 #Hop length\n",
    "power = 2.0 #Power \n",
    "\n",
    "\n",
    "theta_1_set=np.load(\"theta_1_set_new.npy\") #Estimated post-change parameter set\n",
    "theta_not=np.load(\"theta_0_new.npy\") #Estimated pre-change parameter\n",
    "\n",
    "\n",
    "theta_not=theta_not.reshape(theta_not.shape[0],1) \n",
    "\n",
    "theta_1=theta_1_set[:,0] #True post-change parameter.\n",
    "theta_1=theta_1.reshape(theta_1.shape[0],1)\n",
    "\n",
    "stop_EG1=[] #list to store stopping time statistics of Epsilon-Greedy change detector\n",
    "action_EG1=[] #list to store the frequency of action until stopping time\n",
    "\n",
    "iterations=50 #total number of iterations per audio clip. There are 20 different 600s audio clip. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c95a624c",
   "metadata": {},
   "outputs": [],
   "source": [
    "start=time.time()\n",
    "\n",
    "#Loop over 20 different audio files, each audio is of 600s duration\n",
    "for j in range(1,21):\n",
    "    print(\"file=\",j)\n",
    "    test_files_00=[]\n",
    "    test_files_02=[]\n",
    "    test_files_04=[]\n",
    "    test_files_06=[]\n",
    "    \n",
    "    test_files_00.append('/path_to_jth_audio_file_machine_ID_00')\n",
    "    test_files_02.append('/path_to_jth_audio_file_machine_ID_02')\n",
    "    test_files_04.append('/path_to_jth_audio_file_machine_ID_04')\n",
    "    test_files_06.append('/path_to_jth_audio_file_machine_ID_06')\n",
    "    \n",
    "    \n",
    "    test_file_new=[test_files_00[0],test_files_02[0],test_files_04[0],test_files_06[0]]\n",
    "   \n",
    "    \n",
    "     #Epsilon-Greedy Change Detector \n",
    "  #----------------------Loop over each audio clip-----------------------------------------------------------------\n",
    "    for itr in range(iterations):\n",
    "        \n",
    "        flag=0\n",
    "        b1=0\n",
    "        theta_hat=0\n",
    "        V=0\n",
    "        arm_played=[]\n",
    "        \n",
    "        \n",
    "        for t in range(T):\n",
    "            #Condition to check whether the change has occurred. \n",
    "            #If there is no change has been detected by the algorithm, then we manually stop at time horizon.\n",
    "            #flag variable. If flag=1, then we stop and report change\n",
    "            \n",
    "            if flag==1 or t==T-1:\n",
    "                change=t+1\n",
    "                break\n",
    "\n",
    "        #At time step 0, play a random action to get initial observation \n",
    "\n",
    "            if(t<1):\n",
    "                #randomly picked action at time step 0\n",
    "                A1_idx=random.randrange(0,K,1) \n",
    "                a=Action_set[:,A1_idx]\n",
    "                a=a.reshape(a.shape[0],) \n",
    "                \n",
    "                #We access the test file corresponding to action picked at time 0. \n",
    "                #For example, if action 0 is picked, then we access test file corresponding to machine id_00.\n",
    "                # If action 1 is picked, then we access test file corresponding to machine id_02.\n",
    "                # If action 2 is picked, then we access test file corresponding to machine id_04.\n",
    "                # If action 6 is picked, then we access test file corresponding to machine id_06.\n",
    "                \n",
    "                eval_filename=test_file_new[A1_idx]\n",
    "                signal, sr = sound_tools.load_sound_file(eval_filename)\n",
    "                \n",
    "                #Note: Each autoencoder corresponding to a machine, can only take 10s audio clip as the input.\n",
    "                # So, given an input audio clip of 600s, we incrementally access 10s clip at each time step. \n",
    "                signal1=signal[int(10*(t)*sr):int(10*(t+1)*sr)]\n",
    "\n",
    "            #Extract features from this 10s signal:\n",
    "                eval_features = sound_tools.extract_signal_features(\n",
    "                signal1, sr, \n",
    "            n_mels=n_mels, \n",
    "            frames=frames, \n",
    "            n_fft=n_fft, \n",
    "            hop_length=hop_length\n",
    "            )\n",
    "    \n",
    "\n",
    "                feature=eval_features\n",
    "                prediction=model_trained[A1_idx].predict(feature)\n",
    "            \n",
    "                #reconstruction error corresponding to machine id, where machine id number = action number\n",
    "                X=np.mean(np.mean(np.square(eval_features - prediction), axis=1)) \n",
    "    \n",
    "        \n",
    "\n",
    "        \n",
    "        \n",
    "#----------------------Calculation of Q^{(1)} statistics -- V is Q^{(1)} -- One sample update ----------------\n",
    "#-------------This is an one sample update and below calculation is for time step 0-----------------------\n",
    "\n",
    "\n",
    "#Calculating g(X_t|A_t):\n",
    "\n",
    "                temp4=(theta_1_set+theta_not).T[A1_idx,:]\n",
    "                temp5=(theta_1_set-theta_not).T[A1_idx,:]\n",
    "                temp6=temp4*temp5  \n",
    "                temp3=(2*X*((theta_1_set-theta_not).T[A1_idx,:])-temp6)/(2*var)\n",
    "        \n",
    "\n",
    "        \n",
    "#Q^{(1)}:\n",
    "\n",
    "                V=temp3\n",
    "    \n",
    "                theta_hat_idx=np.argmax(V)\n",
    "                theta_hat=theta_1_set[:,theta_hat_idx]\n",
    "\n",
    "    \n",
    "            else:\n",
    "                    #Exploration Phase\n",
    "\n",
    "                if np.random.random()<=epsilon:\n",
    "                    #play action uniformly at random\n",
    "                    play_idx=random.randrange(0,K,1)\n",
    "                    \n",
    "                #Exploitation Phase\n",
    "                else:\n",
    "                    #Calculation of KL Divergence for different set of actions \n",
    "                    mu1=theta_not\n",
    "                    mu2=theta_hat\n",
    "                    a_list=kl.kld(mu2,mu1,var**2)\n",
    "                    #pick action that maximizes the KL divergence\n",
    "                    play_idx=np.argmax(a_list)\n",
    "                \n",
    "                a=Action_set[:,play_idx]\n",
    "                a=a.reshape(a.shape[0],)\n",
    "                arm_played.append(play_idx)\n",
    "                \n",
    "                #We access the test file corresponding to action picked at time t. \n",
    "                #For example, if action 0 is picked, then we access test file corresponding to machine id_00.\n",
    "                # If action 1 is picked, then we access test file corresponding to machine id_02.\n",
    "                # If action 2 is picked, then we access test file corresponding to machine id_04.\n",
    "                # If action 6 is picked, then we access test file corresponding to machine id_06.\n",
    "                \n",
    "                    \n",
    "                eval_filename=test_file_new[play_idx]\n",
    "                signal, sr = sound_tools.load_sound_file(eval_filename)\n",
    "                \n",
    "                #Note: Each autoencoder corresponding to a machine, can only take 10s audio clip as the input.\n",
    "                # So, given an input audio clip of 600s, we incrementally access 10s clip at each time step.\n",
    "                signal1=signal[int(10*(t)*sr):int(10*(t+1)*sr)]\n",
    "\n",
    "    # Extract features from this signal:\n",
    "                eval_features = sound_tools.extract_signal_features(\n",
    "                signal1, sr, \n",
    "                    n_mels=n_mels, \n",
    "                    frames=frames, \n",
    "                    n_fft=n_fft, \n",
    "                    hop_length=hop_length\n",
    "                        )\n",
    "                feature=eval_features\n",
    "                prediction=model_trained[play_idx].predict(feature)\n",
    "                \n",
    "                # reconstruction error corresponding to machine id, where machine id number = action number\n",
    "                X=np.mean(np.mean(np.square(eval_features - prediction), axis=1)) \n",
    "                \n",
    "                #Calculation of Q^{(1)} statistics -- V is Q^{(1)} -- One sample update\n",
    "                \n",
    "#Calculating g(X_t|A_t):\n",
    "\n",
    "                temp4=(theta_1_set+theta_not).T[play_idx,:]\n",
    "                temp5=(theta_1_set-theta_not).T[play_idx,:]\n",
    "                temp6=temp4*temp5  \n",
    "                temp3=(2*X*((theta_1_set-theta_not).T[play_idx,:])-temp6)/(2*var)\n",
    "                \n",
    "#Recursive update of Q^{(1)}:\n",
    "\n",
    "                V=np.maximum(0,(V)+temp3)\n",
    "\n",
    "# Stopping Criteria. If criteria is met, then set flag variable to 1. Otherwise, continue and update theta_hat. \n",
    "# theta_hat is the estimate of post change paramaeter at time step t.\n",
    "                \n",
    "                if np.max(V)>=beta:\n",
    "                    flag=1\n",
    "                else:\n",
    "                    theta_hat_idx=np.argmax(V)\n",
    "                    theta_hat=theta_1_set[:,theta_hat_idx]\n",
    "                    \n",
    "                    \n",
    "  \n",
    " \n",
    "                    \n",
    "                \n",
    "                \n",
    "            \n",
    " \n",
    "            \n",
    "    \n",
    "\n",
    "        stop_EG1.append(change)\n",
    "        action_EG1.append(arm_played)\n",
    "        \n",
    "\n",
    "    print(\"EG done for file %d\"%j)\n",
    "\n",
    "end=time.time()\n",
    "\n",
    "print(\"Time taken by EG(full) = \",end-start)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c295ac85",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "np.save(\"stop_EGfull_1000itr_600s.npy\",stop_EG1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e467e5a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "np.save(\"action_EGfull_1000itr_600s.npy\",action_EG1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fdcec554",
   "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.9.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
