{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import numpy.random as ra\n",
    "import numpy.linalg as la\n",
    "from tqdm import tqdm\n",
    "import matplotlib.pyplot as plt\n",
    "folder = \"xps/LinearBandits/Stationary/\"\n",
    "plt.style.use(\"ggplot\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def calc_sqrt_beta(d,sigma,lamda,delta,S_hat,logdetG):\n",
    "    '''Returns confidence interval width for ridge regression'''\n",
    "    return sigma * np.sqrt( logdetG - d * np.log(lamda) + np.log(1/(delta**2)) ) + np.sqrt(lamda) * S_hat \n",
    "\n",
    "def runit(nb_exp,X,theta,S_hat,sigma,lamda,delta,seed,algorithm='ridge'):\n",
    "    '''function used to run an instance of linear bandits using OFUL with ridge or forward regression\n",
    "    :param nb_exp: integer, number of runs used to average noise\n",
    "    :param X: tuple, dimensions (T,n_a,d) where T: #observations, n_a: #actions, d: #covariates\n",
    "    :param theta: vector of R^d, reward function parameter\n",
    "    :param S_hat: float, such that ||theta||_2 < S_hat\n",
    "    :param sigma: float, variance of the noise\n",
    "    :param lamda: float, regularization\n",
    "    :param delta: float, power of the confidence intervals\n",
    "    :param seed: float, fix seed for reproducibility\n",
    "    :param algorithm: string, ridge or forward\n",
    "    '''\n",
    "    np.random.seed(seed)\n",
    "    T, nb_actions, d = X.shape\n",
    "    regret = np.zeros((nb_exp,T))\n",
    "    pred_norms = np.zeros((nb_exp,T))\n",
    "    x_inv_norms = np.zeros((nb_exp,T))\n",
    "        \n",
    "    for i in tqdm(range(nb_exp)):\n",
    "        G, invG = lamda*np.eye(d), np.eye(d)/lamda\n",
    "        logdetG = d*np.log(lamda)\n",
    "        bT = np.zeros(d) \n",
    "        sqrt_beta = calc_sqrt_beta(d,sigma,lamda,delta,S_hat,logdetG)\n",
    "        theta_hat = np.zeros(d)\n",
    "        for t in range(T):\n",
    "            x = X[t]\n",
    "            #\tX_invVt_norm_sq = np.sum(x * x, axis=1) / ridge;\n",
    "            X_invVt_norm_sq = np.sum(np.dot(x, invG) * x, 1)\n",
    "            x_inv_norms[i,t] = max(X_invVt_norm_sq)\n",
    "            if algorithm == 'ridge':\n",
    "                preds = np.dot(x, theta_hat)\n",
    "                pred_norms[i,t] = np.linalg.norm(preds)\n",
    "                obj = preds + sqrt_beta * np.sqrt(X_invVt_norm_sq) \n",
    "            if algorithm == 'forward':\n",
    "                sqrt_beta += S_hat*np.sqrt(X_invVt_norm_sq/(1+X_invVt_norm_sq))*np.linalg.norm(x,axis=1)\n",
    "                #preds = np.dot(np.dot(1/(1+np.dot(np.dot(x,invG),x.T)),x),theta_hat)\n",
    "                preds = np.dot(x,theta_hat)/(1+X_invVt_norm_sq)\n",
    "                pred_norms[i,t] = np.linalg.norm(preds)\n",
    "                obj = preds + sqrt_beta * np.sqrt(X_invVt_norm_sq/(1+X_invVt_norm_sq)) \n",
    "            pulled_idx = np.argmax(obj)  #Pull the arm with the highest estimated value\n",
    "            xt = x[pulled_idx, :]\n",
    "            bT += (np.dot(xt, theta_star) + sigma * np.random.normal(0,d))*xt\n",
    "            regret[i,t] = np.max(np.dot(x,theta_star))-np.dot(xt, theta_star)\n",
    "            G += np.outer(xt, xt)\n",
    "            tempval1 = np.dot(invG, xt)    # d by 1, O(d^2)\n",
    "            tempval2 = np.dot(tempval1, xt)      # scalar, O(d)\n",
    "            logdetG += np.log(1 + tempval2)\n",
    "            if (t % 20 == 0):\n",
    "                invG = np.linalg.inv(G)\n",
    "            else:\n",
    "                invG -= np.outer(tempval1, tempval1) / (1 + tempval2) \n",
    "            theta_hat = np.dot(invG, bT)\n",
    "            #print(theta_hat)\n",
    "            sqrt_beta = calc_sqrt_beta(d, sigma, lamda, delta, S_hat, logdetG);\n",
    "    return regret, pred_norms, x_inv_norms\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Experiments"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "T, d, n_a = int(1e5), 100, 10  # , number of covariates, number of actions\n",
    "\n",
    "#generate X\n",
    "X = np.random.randn(T, n_a, d)\n",
    "X_max = 200\n",
    "X /= np.max(np.linalg.norm(X,axis=2))/X_max\n",
    "\n",
    "S, sigma = 1, 0.1\n",
    "# generate theta_star\n",
    "theta_star = np.random.normal(0,1,d); theta_star *= S/np.linalg.norm(theta_star)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/100 [00:00<?, ?it/s]/home/reda/anaconda3/lib/python3.7/site-packages/ipykernel_launcher.py:37: RuntimeWarning: invalid value encountered in sqrt\n",
      "/home/reda/anaconda3/lib/python3.7/site-packages/ipykernel_launcher.py:51: RuntimeWarning: invalid value encountered in log\n",
      "100%|██████████| 100/100 [32:42<00:00, 19.63s/it]\n",
      "100%|██████████| 100/100 [36:31<00:00, 21.91s/it]\n"
     ]
    }
   ],
   "source": [
    "lamda = 1/T #0.1;\n",
    "delta = 0.001\n",
    "seed = 1996\n",
    "S_hat = S\n",
    "nb_exp = 100\n",
    "algorithms = ['ridge','forward']\n",
    "labels = ['OFUL$^{r}$','OFUL$^{f}$']\n",
    "# colors = ['blue','darkviolet','darkorange']#,'darkred']\n",
    "colors = ['#377eb8', '#ff7f00', '#4daf4a','#f781bf', '#a65628', '#984ea3','#999999', '#e41a1c', '#dede00']\n",
    "# s_colors = ['cyan','violet','orange']#,'red']\n",
    "regrets = np.zeros((nb_exp,len(algorithms),T))\n",
    "pred_norms = np.zeros((nb_exp,len(algorithms),T))\n",
    "x_inv_norms = np.zeros((nb_exp,len(algorithms),T))\n",
    "\n",
    "for i,a in enumerate(algorithms):\n",
    "    regrets[:,i,:], pred_norms[:,i,:], x_inv_norms[:,i,:] = runit(nb_exp,X,theta_star,S_hat,sigma,lamda,delta,seed,a)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Regret"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/reda/anaconda3/lib/python3.7/site-packages/ipykernel_launcher.py:28: UserWarning: Creating legend with loc=\"best\" can be slow with large amounts of data.\n",
      "/home/reda/anaconda3/lib/python3.7/site-packages/ipykernel_launcher.py:28: UserWarning: Creating legend with loc=\"best\" can be slow with large amounts of data.\n",
      "/home/reda/anaconda3/lib/python3.7/site-packages/ipykernel_launcher.py:28: UserWarning: Creating legend with loc=\"best\" can be slow with large amounts of data.\n",
      "/home/reda/anaconda3/lib/python3.7/site-packages/IPython/core/pylabtools.py:128: UserWarning: Creating legend with loc=\"best\" can be slow with large amounts of data.\n",
      "  fig.canvas.print_figure(bytes_io, **kw)\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "mean_regret = np.cumsum(regrets.mean(axis=0),axis=1)\n",
    "fst_quantile_regret = np.cumsum(np.quantile(regrets,0.25,axis=0),axis=1)\n",
    "thd_quantile_regret = np.cumsum(np.quantile(regrets,0.75,axis=0),axis=1)\n",
    "std_regret = np.cumsum(regrets,axis=2).std(axis=0)\n",
    "_ = np.log(lamda+(X_max**2)*np.arange(1,T+1)/d)\n",
    "\n",
    "labels = ['OFUL','OFUL$^{f}$']\n",
    "\n",
    "\n",
    "for i,a in enumerate(algorithms):\n",
    "#     plt.plot(np.divide(mean_regret[i],np.sqrt(T*d*_*np.arange(1,T+1))),label=labels[i],color=colors[i])\n",
    "    plt.plot(mean_regret[i],label=labels[i],color=colors[i])\n",
    "#     plt.fill_between(np.arange(T),np.divide(fst_quantile_regret[i],np.sqrt(T*d*_*np.arange(1,T+1))),np.divide(thd_quantile_regret[i],np.sqrt(T*d*_*np.arange(1,T+1))),color=colors[i],alpha=0.2)\n",
    "    plt.fill_between(np.arange(T),fst_quantile_regret[i],thd_quantile_regret[i],color=colors[i],alpha=0.2)\n",
    "    plt.plot(fst_quantile_regret[i],color=colors[i],linestyle=\"dashed\")\n",
    "    plt.plot(thd_quantile_regret[i],color=colors[i],linestyle=\"dashed\")\n",
    "#     plt.fill_between(np.arange(T),mean_regret[i]-std_regret[i],mean_regret[i]+std_regret[i],alpha=0.1)\n",
    "log_ = np.log(lamda+np.arange(1,T+1)/d)\n",
    "# plt.plot(4*np.sqrt(T*d*log_)*(np.sqrt(lamda)*S_hat+sigma*np.sqrt(2*np.log(1/delta)+d*(log_-np.log(lamda)))),label=\"bound\",linestyle=\"dashed\")\n",
    "# plt.xscale('log')\n",
    "plt.yscale('log')\n",
    "plt.xlabel(\"round t\")\n",
    "plt.ylabel('Cumulative Bandit regret')\n",
    "# plt.ylim([0,500])\n",
    "# plt.plot(np.cumsum(regret[:,1,:].mean(axis=0)),label='forward')\n",
    "# plt.plot(np.cumsum(regret,axis=1).std(axis=0))\n",
    "plt.legend()\n",
    "plt.savefig(folder+\"lamda_\"+str(lamda)+'_sigma_'+str(sigma)+\"_X_\"+str(X_max)+\"_S_\"+str(S)+\"_delta_\"+str(delta)+\".pdf\",bbox_inches='tight')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "In the following cells, we visualize how the parameter estimate's norm evolves over time, this is not discussed in the paper, but it allows us to see how much the ``bounded predictions'' assumption is violated "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Prediction norms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "mean_norms = pred_norms.mean(axis=0)\n",
    "plt.plot(mean_norms[0]-mean_norms[1])\n",
    "# plt.plot(np.cumsum(regret[:,1,:].mean(axis=0)),label='forward')\n",
    "# plt.plot(np.cumsum(regret,axis=1).std(axis=0))\n",
    "plt.legend()\n",
    "# plt.yscale(\"log\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### distance of new points to old"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "# Here we visualize the sum of features' norms $x_t^T G_{t}^{-1} x_t$ for both algorithms\n",
    "mean_x_inv_norms = x_inv_norms.mean(axis=0)\n",
    "plt.plot(mean_x_inv_norms[0])\n",
    "# plt.plot(np.cumsum(regret[:,1,:].mean(axis=0)),label='forward')\n",
    "# plt.plot(np.cumsum(regret,axis=1).std(axis=0))\n",
    "plt.legend()\n",
    "plt.yscale(\"log\")"
   ]
  }
 ],
 "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
}
