{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "cellView": "form",
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 364,
     "status": "ok",
     "timestamp": 1652373079763,
     "user": {
      "displayName": "Imad Aouali",
      "userId": "10957666911602104909"
     },
     "user_tz": 240
    },
    "id": "dUrLMhAcZYDY",
    "outputId": "e7637e0a-2d44-4760-e4d6-bcbb43502924",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "python 3.6.8\n",
      "matplotlib 3.2.2\n",
      "30 joblib cores\n"
     ]
    }
   ],
   "source": [
    "# Imports and defaults\n",
    "import itertools\n",
    "import joblib\n",
    "from joblib import Parallel, delayed\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "from scipy.linalg import block_diag\n",
    "import time\n",
    "import os\n",
    "\n",
    "mpl.style.use(\"classic\")\n",
    "mpl.rcParams[\"figure.figsize\"] = [5, 3]\n",
    "\n",
    "mpl.rcParams[\"axes.linewidth\"] = 0.75\n",
    "mpl.rcParams[\"figure.facecolor\"] = \"w\"\n",
    "mpl.rcParams[\"grid.linewidth\"] = 0.75\n",
    "mpl.rcParams[\"lines.linewidth\"] = 0.75\n",
    "mpl.rcParams[\"patch.linewidth\"] = 0.75\n",
    "mpl.rcParams[\"xtick.major.size\"] = 3\n",
    "mpl.rcParams[\"ytick.major.size\"] = 3\n",
    "\n",
    "mpl.rcParams[\"pdf.fonttype\"] = 42\n",
    "mpl.rcParams[\"ps.fonttype\"] = 42\n",
    "mpl.rcParams[\"font.size\"] = 9\n",
    "mpl.rcParams[\"axes.titlesize\"] = \"medium\"\n",
    "mpl.rcParams[\"legend.fontsize\"] = \"medium\"\n",
    "\n",
    "import platform\n",
    "print(\"python %s\" % platform.python_version())\n",
    "print(\"matplotlib %s\" % mpl.__version__)\n",
    "print(\"%d joblib cores\" % joblib.cpu_count())\n",
    "\n",
    "def linestyle2dashes(style):\n",
    "  if style == \"--\":\n",
    "    return (3, 3)\n",
    "  elif style == \":\":\n",
    "    return (0.5, 2.5)\n",
    "  else:\n",
    "    return (None, None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "cellView": "form",
    "id": "f7uZWkUjRmUs",
    "tags": []
   },
   "outputs": [],
   "source": [
    "# Bandit environments\n",
    "class CoBandit(object):\n",
    "  \"\"\"Contextual bandit with K arms.\"\"\"\n",
    "\n",
    "  def __init__(self, K, contexts, Theta, sigma=1.0):\n",
    "    self.K = K  # number of arms\n",
    "    self.contexts = np.copy(contexts)  # [number of contexts] x d feature matrix\n",
    "    self.num_contexts = self.contexts.shape[0]  # number of contexts\n",
    "    self.d = self.contexts.shape[1]  # number of features\n",
    "    self.Theta = np.copy(Theta)  # [number of arms] x d arm parameters\n",
    "    self.sigma = sigma  # reward noise\n",
    "\n",
    "    self.randomize()\n",
    "    \n",
    "  def sigmoid(self, x):\n",
    "    y = 1 / (1 + np.exp(- x))\n",
    "    return y\n",
    "\n",
    "\n",
    "  def randomize(self):\n",
    "    # randomly choose one context per arm (does not have to be the same)\n",
    "    self.ndx = np.random.randint(self.num_contexts, size=self.K)\n",
    "    self.X = self.contexts[self.ndx, :]\n",
    "\n",
    "    # mean and stochastic rewards\n",
    "    self.mut = self.sigmoid((self.X * self.Theta).sum(axis=-1))\n",
    "    self.rt = (np.random.rand(self.K) < self.mut).astype(float)\n",
    "    self.best_arm = np.argmax(self.mut)\n",
    "\n",
    "  def reward(self, arm):\n",
    "    # instantaneous reward of the arm\n",
    "    return self.rt[arm]\n",
    "\n",
    "  def regret(self, arm):\n",
    "    # instantaneous regret of the arm\n",
    "    return self.rt[self.best_arm] - self.rt[arm]\n",
    "\n",
    "  def pregret(self, arm):\n",
    "    # expected regret of the arm\n",
    "    return self.mut[self.best_arm] - self.mut[arm]\n",
    "\n",
    "  def print(self):\n",
    "    return \"Contextual bandit: %d dimensions, %d arms\" % (self.d, self.K)\n",
    "\n",
    "\n",
    "def evaluate_one(Alg, params, env, n, period_size=1):\n",
    "  \"\"\"One run of a bandit algorithm.\"\"\"\n",
    "  alg = Alg(env, n, params)\n",
    "\n",
    "  regret = np.zeros(n // period_size)\n",
    "  for t in range(n):\n",
    "    # generate state\n",
    "    env.randomize()\n",
    "\n",
    "    # take action and update agent\n",
    "    arm = alg.get_arm(t)\n",
    "    alg.update(t, arm, env.reward(arm))\n",
    "\n",
    "    # track performance\n",
    "    regret_at_t = env.regret(arm)\n",
    "    regret[t // period_size] += regret_at_t\n",
    "\n",
    "  return regret, alg\n",
    "\n",
    "\n",
    "def evaluate(Alg, params, env, n=1000, period_size=1, printout=True):\n",
    "  \"\"\"Multiple runs of a bandit algorithm.\"\"\"\n",
    "  if printout:\n",
    "    print(\"Evaluating %s\" % Alg.print(), end=\"\")\n",
    "  start = time.time()\n",
    "\n",
    "  num_exps = len(env)\n",
    "  regret = np.zeros((n // period_size, num_exps))\n",
    "  alg = num_exps * [None]\n",
    "\n",
    "  output = Parallel(n_jobs=-1)(delayed(evaluate_one)(Alg, params, env[ex], n, period_size)\n",
    "    for ex in range(num_exps))\n",
    "  for ex in range(num_exps):\n",
    "    regret[:, ex] = output[ex][0]\n",
    "    alg[ex] = output[ex][1]\n",
    "  if printout:\n",
    "    print(\" %.1f seconds\" % (time.time() - start))\n",
    "\n",
    "  if printout:\n",
    "    total_regret = regret.sum(axis=0)\n",
    "    print(\"Regret: %.2f +/- %.2f (median: %.2f, max: %.2f, min: %.2f)\" %\n",
    "      (total_regret.mean(), total_regret.std() / np.sqrt(num_exps),\n",
    "      np.median(total_regret), total_regret.max(), total_regret.min()))\n",
    "\n",
    "  return regret, alg"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class dTS:\n",
    "  def __init__(self, env, n, params):\n",
    "    self.env = env  # bandit environment that the agent interacts with\n",
    "    self.K = self.env.K  # number of arms\n",
    "    self.d = self.env.d  # number of features\n",
    "    self.n = n  # horizon\n",
    "    self.W = self.env.W\n",
    "    self.L = self.env.L\n",
    "    self.Sigma = np.copy(self.env.Sigma)\n",
    "    self.sigma = self.env.sigma  # reward noise\n",
    "    \n",
    "    # override default values\n",
    "    for attr, val in params.items():\n",
    "      if isinstance(val, np.ndarray):\n",
    "        setattr(self, attr, np.copy(val))\n",
    "      elif isinstance(val, list):\n",
    "        setattr(self, attr, val.copy())\n",
    "      else:\n",
    "        setattr(self, attr, val)\n",
    "     \n",
    "    self.Lambda = 1 / self.Sigma\n",
    "        \n",
    "    # sufficient statistics\n",
    "    self.G_hat = np.zeros((self.K, self.d, self.d))\n",
    "    self.B_hat = np.zeros((self.K, self.d))\n",
    "    \n",
    "  def update(self, t, arm, r):\n",
    "    # update sufficient statistics\n",
    "    x = self.env.X[arm, :]\n",
    "    self.G_hat[arm, :, :] += np.outer(x, x) / np.square(self.sigma)\n",
    "    self.B_hat[arm, :] += x * r / np.square(self.sigma)\n",
    "\n",
    "  def get_arm(self, t):\n",
    "        \n",
    "    #Computing Gaussian parameters\n",
    "    Sigma_hat = []\n",
    "    for i in range(self.K):\n",
    "        Sigma_hat_helper = np.linalg.inv(self.Lambda[0] * np.eye(self.d) + self.G_hat[i, :, :])\n",
    "        Sigma_hat.append(Sigma_hat_helper)\n",
    "    \n",
    "    G_bar = np.zeros((self.L, self.d, self.d))\n",
    "    B_bar = np.zeros((self.L, self.d))\n",
    "    Sigma_bar = []\n",
    "    \n",
    "    for l in range(self.L):\n",
    "        W = self.W[l]\n",
    "        W_T = self.W[l].T\n",
    "        if l==0:\n",
    "            G_helper = np.zeros((self.d, self.d))\n",
    "            B_helper = np.zeros(self.d)\n",
    "            for i in range(self.K):\n",
    "                G_helper += self.Lambda[l] * np.eye(self.d) - (self.Lambda[l]**2) * Sigma_hat[i]\n",
    "                B_helper += Sigma_hat[i].dot(self.B_hat[i, :])\n",
    "            G_bar[l, :, :] = W_T.dot(G_helper.dot(W))\n",
    "            B_bar[l, :] = self.Lambda[l] * W_T.dot(B_helper)\n",
    "            Sigma_bar.append(np.linalg.inv(self.Lambda[l+1] * np.eye(self.d) + G_bar[l, :, :])) \n",
    "        else:\n",
    "            G_bar[l, :, :] = self.Lambda[l] * np.eye(self.d) - (self.Lambda[l]**2) * Sigma_bar[l-1]\n",
    "            B_bar[l, :] = self.Lambda[l] * W_T.dot(Sigma_bar[l-1].dot(B_bar[l-1, :]))\n",
    "            Sigma_bar.append(np.linalg.inv(self.Lambda[l+1] * np.eye(self.d) + G_bar[l, :, :]))\n",
    "    \n",
    "    #Hierarchical sampling\n",
    "    #when l=self.L\n",
    "    mu_bar = Sigma_bar[self.L-1].dot(B_bar[self.L-1, :])\n",
    "    psi =  np.random.multivariate_normal(mu_bar, Sigma_bar[self.L-1])\n",
    "    \n",
    "    for l in range(self.L-2, -1, -1):\n",
    "        mu_bar = self.Lambda[l+1] * self.W[l+1].dot(psi) + B_bar[l, :]\n",
    "        mu_bar = Sigma_bar[l].dot(mu_bar)\n",
    "        psi = np.random.multivariate_normal(mu_bar, Sigma_bar[l])\n",
    "        \n",
    "    self.mu = np.zeros(self.K)\n",
    "    for i in range(self.K):\n",
    "      mu_ti = self.Lambda[0] * self.W[0].dot(psi) + self.B_hat[i, :]\n",
    "      mu_ti = Sigma_hat[i].dot(mu_ti)\n",
    "      # posterior sampling\n",
    "      theta_tilde = np.random.multivariate_normal(mu_ti, Sigma_hat[i])\n",
    "      self.mu[i] = self.env.X[i, :].dot(theta_tilde)\n",
    "    \n",
    "    arm = np.argmax(self.mu)\n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"dTS\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class GLBdTS:\n",
    "  def __init__(self, env, n, params):\n",
    "    self.env = env  # bandit environment that the agent interacts with\n",
    "    self.K = self.env.K  # number of arms\n",
    "    self.d = self.env.d  # number of features\n",
    "    self.n = n  # horizon\n",
    "    self.W = self.env.W\n",
    "    self.L = self.env.L\n",
    "    self.Sigma = np.copy(self.env.Sigma)\n",
    "    self.sigma = self.env.sigma  # reward noise\n",
    "    self.Theta0 = np.zeros((self.K, self.d))\n",
    "\n",
    "    \n",
    "    self.irls_Theta = np.copy(self.Theta0)\n",
    "    self.irls_error = 1e-3\n",
    "    self.irls_num_iter = 1000\n",
    "    self.batch_size = 30 #self.n\n",
    "\n",
    "    \n",
    "    # override default values\n",
    "    for attr, val in params.items():\n",
    "      if isinstance(val, np.ndarray):\n",
    "        setattr(self, attr, np.copy(val))\n",
    "      elif isinstance(val, list):\n",
    "        setattr(self, attr, val.copy())\n",
    "      else:\n",
    "        setattr(self, attr, val)\n",
    "     \n",
    "    self.Lambda = 1 / self.Sigma\n",
    "        \n",
    "\n",
    "    # sufficient statistics\n",
    "    self.G_hat = np.zeros((self.K, self.d, self.d))\n",
    "    self.B_hat = np.zeros((self.K, self.d))\n",
    "    \n",
    "    self.pulls = np.zeros(self.K, dtype=int) # number of observations for each arm    \n",
    "    self.context_ndx = [[] for i in range(self.K)]\n",
    "    self.r = [[] for i in range(self.K)]\n",
    "  \n",
    "  def sigmoid(self, x):\n",
    "    y = 1 / (1 + np.exp(- x))\n",
    "    return y\n",
    "\n",
    "  def solve(self, arm):\n",
    "    # iterative reweighted least squares for Bayesian logistic regression\n",
    "    # Sections 4.3.3 and 4.5.1 in Bishop (2006)\n",
    "    # Pattern Recognition and Machine Learning\n",
    "    small_eye = 1e-5 * np.eye(self.d)\n",
    "    theta = np.copy(self.irls_Theta[arm,:])\n",
    "    nbr_obs = self.pulls[arm]\n",
    "    t = self.r[arm]\n",
    "    context_ndx = self.context_ndx[arm]\n",
    "    num_iter = 0\n",
    "    \n",
    "    while num_iter < self.irls_num_iter:\n",
    "        theta_old = np.copy(theta)\n",
    "        Gram =  small_eye\n",
    "        Phiyt = np.zeros(self.d)\n",
    "\n",
    "        if nbr_obs <= self.batch_size:\n",
    "            batch = np.arange(nbr_obs)\n",
    "        else:\n",
    "            batch = np.random.choice(nbr_obs, size=self.batch_size)\n",
    "\n",
    "        for i in batch:\n",
    "            x = self.env.contexts[context_ndx[i],:]\n",
    "            y = self.sigmoid((x * theta).sum())\n",
    "            Gram += y * (1-y) * np.outer(x, x) \n",
    "            Phiyt += (y-t[i]) * x\n",
    "\n",
    "        PhiRz = Gram.dot(theta) - Phiyt\n",
    "        theta = np.linalg.solve(Gram, PhiRz)\n",
    "        \n",
    "        if np.linalg.norm(theta - theta_old) < self.irls_error:\n",
    "            break;\n",
    "        num_iter += 1\n",
    "    \n",
    "    if num_iter == self.irls_num_iter:\n",
    "        self.irls_Theta[arm,:] = self.Theta0[arm,:]\n",
    "    else:\n",
    "        self.irls_Theta[arm,:] = np.copy(theta)\n",
    "\n",
    "    return theta, Gram\n",
    "\n",
    "  def update(self, t, arm, r):    \n",
    "    self.r[arm].append(r)\n",
    "    self.pulls[arm] += 1\n",
    "    self.context_ndx[arm].append(self.env.ndx[arm])\n",
    "    \n",
    "    x = self.env.X[arm, :]\n",
    "    theta, self.G_hat[arm, :, :] = self.solve(arm)\n",
    "    self.B_hat[arm,:] = self.G_hat[arm, :, :].dot(theta)\n",
    "\n",
    "  def get_arm(self, t):\n",
    "        \n",
    "    #Computing Gaussian parameters\n",
    "    Sigma_hat = []\n",
    "    for i in range(self.K):\n",
    "        Sigma_hat_helper = np.linalg.inv(self.Lambda[0] * np.eye(self.d) + self.G_hat[i, :, :])\n",
    "        Sigma_hat.append(Sigma_hat_helper)\n",
    "    \n",
    "    G_bar = np.zeros((self.L, self.d, self.d))\n",
    "    B_bar = np.zeros((self.L, self.d))\n",
    "    Sigma_bar = []\n",
    "    \n",
    "    for l in range(self.L):\n",
    "        W = self.W[l]\n",
    "        W_T = self.W[l].T\n",
    "        if l==0:\n",
    "            G_helper = np.zeros((self.d, self.d))\n",
    "            B_helper = np.zeros(self.d)\n",
    "            for i in range(self.K):\n",
    "                G_helper += self.Lambda[l] * np.eye(self.d) - (self.Lambda[l]**2) * Sigma_hat[i]\n",
    "                B_helper += Sigma_hat[i].dot(self.B_hat[i, :])\n",
    "            G_bar[l, :, :] = W_T.dot(G_helper.dot(W))\n",
    "            B_bar[l, :] = self.Lambda[l] * W_T.dot(B_helper)\n",
    "            Sigma_bar.append(np.linalg.inv(self.Lambda[l+1] * np.eye(self.d) + G_bar[l, :, :])) \n",
    "        else:\n",
    "            G_bar[l, :, :] = self.Lambda[l] * np.eye(self.d) - (self.Lambda[l]**2) * Sigma_bar[l-1]\n",
    "            B_bar[l, :] = self.Lambda[l] * W_T.dot(Sigma_bar[l-1].dot(B_bar[l-1, :]))\n",
    "            Sigma_bar.append(np.linalg.inv(self.Lambda[l+1] * np.eye(self.d) + G_bar[l, :, :]))\n",
    "    \n",
    "    #Hierarchical sampling\n",
    "    #when l=self.L\n",
    "    mu_bar = Sigma_bar[self.L-1].dot(B_bar[self.L-1, :])\n",
    "    psi =  np.random.multivariate_normal(mu_bar, Sigma_bar[self.L-1])\n",
    "    \n",
    "    for l in range(self.L-2, -1, -1):\n",
    "        mu_bar = self.Lambda[l+1] * self.W[l+1].dot(psi) + B_bar[l, :]\n",
    "        mu_bar = Sigma_bar[l].dot(mu_bar)\n",
    "        psi = np.random.multivariate_normal(mu_bar, Sigma_bar[l])\n",
    "        \n",
    "    self.mu = np.zeros(self.K)\n",
    "    for i in range(self.K):\n",
    "      mu_ti = self.Lambda[0] * self.W[0].dot(psi) + self.B_hat[i, :]\n",
    "      mu_ti = Sigma_hat[i].dot(mu_ti)\n",
    "      # posterior sampling\n",
    "      theta_tilde = np.random.multivariate_normal(mu_ti, Sigma_hat[i])\n",
    "      self.mu[i] = self.env.X[i, :].dot(theta_tilde)\n",
    "    \n",
    "    arm = np.argmax(self.mu)\n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"GLBdTS\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class HierTS:\n",
    "  def __init__(self, env, n, params):\n",
    "    self.env = env  # bandit environment that the agent interacts with\n",
    "    self.K = self.env.K  # number of arms\n",
    "    self.d = self.env.d  # number of features\n",
    "    self.n = n  # horizon\n",
    "    self.B_L = self.env.B_L\n",
    "    self.mar_Sigma_1 = np.copy(self.env.mar_Sigma_1)\n",
    "    self.Sigma_L = np.copy(self.env.Sigma[L]) * self.B_L.dot(self.B_L.T) + 10**(-3) * np.eye(self.d)\n",
    "    self.sigma = self.env.sigma  # reward noise\n",
    "    \n",
    "    # override default values\n",
    "    for attr, val in params.items():\n",
    "      if isinstance(val, np.ndarray):\n",
    "        setattr(self, attr, np.copy(val))\n",
    "      elif isinstance(val, list):\n",
    "        setattr(self, attr, val.copy())\n",
    "      else:\n",
    "        setattr(self, attr, val)\n",
    "     \n",
    "    self.mar_Lambda_1 = np.linalg.inv(self.mar_Sigma_1)\n",
    "    self.Lambda_L = np.linalg.inv(self.Sigma_L)\n",
    "        \n",
    "    # sufficient statistics\n",
    "    self.G_hat = np.zeros((self.K, self.d, self.d))\n",
    "    self.B_hat = np.zeros((self.K, self.d))\n",
    "    \n",
    "  def update(self, t, arm, r):\n",
    "    # update sufficient statistics\n",
    "    x = self.env.X[arm, :]\n",
    "    self.G_hat[arm, :, :] += np.outer(x, x) / np.square(self.sigma)\n",
    "    self.B_hat[arm, :] += x * r / np.square(self.sigma)\n",
    "\n",
    "  def get_arm(self, t):\n",
    "        \n",
    "    #Computing Gaussian parameters\n",
    "    Sigma_hat = []\n",
    "    for i in range(self.K):\n",
    "        Sigma_hat_helper = np.linalg.inv(self.mar_Lambda_1 + self.G_hat[i, :, :])\n",
    "        Sigma_hat.append(Sigma_hat_helper)\n",
    "    \n",
    "    G_bar = np.zeros((self.d, self.d))\n",
    "    B_bar = np.zeros((self.d))\n",
    "    \n",
    "    \n",
    "    G_helper = np.zeros((self.d, self.d))\n",
    "    B_helper = np.zeros(self.d)\n",
    "    for i in range(self.K):\n",
    "        G_helper += self.mar_Lambda_1 - self.mar_Lambda_1.dot(Sigma_hat[i].dot(self.mar_Lambda_1))\n",
    "        B_helper += Sigma_hat[i].dot(self.B_hat[i, :])\n",
    "        \n",
    "    G_bar = G_helper\n",
    "    B_bar = self.mar_Lambda_1.dot(B_helper)\n",
    "    Sigma_bar = np.linalg.inv(self.Lambda_L + G_bar) \n",
    "    \n",
    "    #Hierarchical sampling\n",
    "    mu_bar = Sigma_bar.dot(B_bar)\n",
    "    psi =  np.random.multivariate_normal(mu_bar, Sigma_bar)\n",
    "            \n",
    "    self.mu = np.zeros(self.K)\n",
    "    for i in range(self.K):\n",
    "      mu_ti = self.mar_Lambda_1.dot(psi) + self.B_hat[i, :]\n",
    "      mu_ti = Sigma_hat[i].dot(mu_ti)\n",
    "      # posterior sampling\n",
    "      theta_tilde = np.random.multivariate_normal(mu_ti, Sigma_hat[i])\n",
    "      self.mu[i] = self.env.X[i, :].dot(theta_tilde)\n",
    "    \n",
    "    arm = np.argmax(self.mu)\n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"HierTS\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class LogBanditAlg:\n",
    "  def __init__(self, env, n, params):\n",
    "    self.env = env  # bandit environment that the agent interacts with\n",
    "    self.K = self.env.K  # number of arms\n",
    "    self.d = self.env.d  # number of features\n",
    "    self.n = n  # horizon\n",
    "    self.Theta0 = np.zeros((self.K, self.d))  # prior means of arm parameters\n",
    "    self.mar_Sigma = np.copy(self.env.mar_Sigma)\n",
    "    self.sigma = self.env.sigma  # reward noise\n",
    "\n",
    "\n",
    "    self.irls_Theta = np.zeros((self.K, self.d))\n",
    "    self.irls_error = 1e-3\n",
    "    self.irls_num_iter = 1000\n",
    "    self.batch_size = 30 #self.n\n",
    "    \n",
    "    for attr, val in params.items():\n",
    "      setattr(self, attr, val)\n",
    "\n",
    "    # sufficient statistics\n",
    "    self.pulls = np.zeros(self.K, dtype=int) # number of observations for each arm\n",
    "    \n",
    "    self.marLambda = np.linalg.inv(self.mar_Sigma)\n",
    "\n",
    "    self.G = np.zeros((self.K, self.d, self.d))\n",
    "    \n",
    "    self.context_ndx = [[] for i in range(self.K)]\n",
    "    self.r = [[] for i in range(self.K)]\n",
    "\n",
    "  def update(self, t, arm, r):    \n",
    "    self.r[arm].append(r)\n",
    "    self.pulls[arm] += 1\n",
    "    self.context_ndx[arm].append(self.env.ndx[arm])\n",
    "    \n",
    "    x = self.env.X[arm, :]\n",
    "    self.G[arm, :, :] += np.outer(x, x) / np.square(self.sigma)      \n",
    "    \n",
    "    \n",
    "  def sigmoid(self, x):\n",
    "    y = 1 / (1 + np.exp(- x))\n",
    "    return y\n",
    "\n",
    "  def solve(self, arm):\n",
    "    # iterative reweighted least squares for Bayesian logistic regression\n",
    "    # Sections 4.3.3 and 4.5.1 in Bishop (2006)\n",
    "    # Pattern Recognition and Machine Learning\n",
    "    \n",
    "    small_eye = 1e-5 * np.eye(self.d)\n",
    "    theta = np.copy(self.irls_Theta[arm,:])\n",
    "    nbr_obs = self.pulls[arm]\n",
    "    t = self.r[arm]\n",
    "    context_ndx = self.context_ndx[arm]\n",
    "\n",
    "    num_iter = 0\n",
    "    \n",
    "    while num_iter < self.irls_num_iter:\n",
    "        theta_old = np.copy(theta)\n",
    "        Gram = small_eye\n",
    "        Phiyt = np.zeros(self.d)\n",
    "\n",
    "        if nbr_obs <= self.batch_size:\n",
    "            batch = np.arange(nbr_obs)\n",
    "        else:\n",
    "            batch = np.random.choice(nbr_obs, size=self.batch_size)\n",
    "\n",
    "        for i in batch:\n",
    "            x = self.env.contexts[context_ndx[i],:]\n",
    "            y = self.sigmoid((x * theta).sum())\n",
    "            Gram += y * (1-y) * np.outer(x, x) \n",
    "            Phiyt += (y-t[i]) * x\n",
    "\n",
    "        PhiRz = Gram.dot(theta) - Phiyt\n",
    "        theta = np.linalg.solve(Gram, PhiRz)\n",
    "        \n",
    "        if np.linalg.norm(theta - theta_old) < self.irls_error:\n",
    "            break;\n",
    "        num_iter += 1\n",
    "    \n",
    "    if num_iter == self.irls_num_iter:\n",
    "        self.irls_Theta[arm,:] = self.Theta0[arm,:]\n",
    "    else:\n",
    "        self.irls_Theta[arm,:] = np.copy(theta)\n",
    "\n",
    "    return theta, Gram\n",
    "\n",
    "class LogUCB(LogBanditAlg):\n",
    "  def __init__(self, env, n, params):\n",
    "    LogBanditAlg.__init__(self, env, n, params)\n",
    "\n",
    "    self.cew = self.confidence_ellipsoid_width(n)\n",
    "    self.inv_Gt = np.zeros((self.K, self.d, self.d))\n",
    "    \n",
    "  def confidence_ellipsoid_width(self, t):\n",
    "    # Section 4.1 in Filippi (2010)\n",
    "    # Parametric Bandits: The Generalized Linear Case\n",
    "    delta = 1 / self.n\n",
    "    c_m = np.amax(np.linalg.norm(self.env.contexts, axis=1))\n",
    "    c_mu = 0.25 # minimum derivative of the mean function\n",
    "    k_mu = 0.25\n",
    "    kappa = np.sqrt(3 + 2 * np.log(1 + 2 * np.square(c_m / self.sigma)))\n",
    "    R_max = 1.0\n",
    "    width = (2 * k_mu * kappa * R_max / c_mu) * \\\n",
    "      np.sqrt(2 * self.d * np.log(t) * np.log(2 * self.d * self.n / delta))\n",
    "    return width\n",
    "\n",
    "  def get_arm(self, t):\n",
    "    self.mu = np.zeros(self.K)\n",
    "    \n",
    "    if t==0:\n",
    "        for i in range(self.K):\n",
    "            Gt = self.marLambda + self.G[i, :, :]\n",
    "            self.inv_Gt[i, :, :] = np.linalg.inv(Gt)\n",
    "            theta, _ = self.solve(i)\n",
    "    else:\n",
    "        Gt = self.marLambda + self.G[self.At, :, :]\n",
    "        self.inv_Gt[self.At, :, :] = np.linalg.inv(Gt)\n",
    "        theta, _ = self.solve(self.At)\n",
    "                \n",
    "    for i in range(self.K):\n",
    "        # UCBs\n",
    "        theta_hat = self.irls_Theta[i,:]\n",
    "        inv_Gt = self.inv_Gt[i, :, :]\n",
    "        self.mu[i] = self.sigmoid(self.env.X[i, :].dot(theta_hat)) + self.cew * \\\n",
    "          np.sqrt((self.env.X[i, :].dot(inv_Gt).dot(self.env.X[i, :])))\n",
    "\n",
    "    arm = np.argmax(self.mu)\n",
    "    self.At = arm\n",
    "    \n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"GLM-UCB\"\n",
    "\n",
    "\n",
    "class UCBLog(LogBanditAlg):\n",
    "  def __init__(self, env, n, params):\n",
    "    LogBanditAlg.__init__(self, env, n, params)\n",
    "\n",
    "    self.cew = self.confidence_ellipsoid_width(n)\n",
    "    self.inv_Gt = np.zeros((self.K, self.d, self.d))\n",
    "    \n",
    "  def confidence_ellipsoid_width(self, t):\n",
    "    # Theorem 2 in Li (2017)\n",
    "    # Provably Optimal Algorithms for Generalized Linear Contextual Bandits\n",
    "    delta = 1 / self.n\n",
    "    sigma = 0.5\n",
    "    kappa = 0.25 # minimum derivative of a constrained mean function\n",
    "    width = (sigma / kappa) * \\\n",
    "      np.sqrt((self.d / 2) * np.log(1 + 2 * self.n / self.d) + \\\n",
    "      np.log(1 / delta))\n",
    "    return width\n",
    "\n",
    "  def get_arm(self, t):\n",
    "    self.mu = np.zeros(self.K)\n",
    "    \n",
    "    if t==0:\n",
    "        for i in range(self.K):\n",
    "            Gt = self.marLambda + self.G[i, :, :]\n",
    "            self.inv_Gt[i, :, :] = np.linalg.inv(Gt)\n",
    "            theta, _ = self.solve(i)\n",
    "    else:\n",
    "        Gt = self.marLambda + self.G[self.At, :, :]\n",
    "        self.inv_Gt[self.At, :, :] = np.linalg.inv(Gt)\n",
    "        theta, _ = self.solve(self.At)\n",
    "                \n",
    "    for i in range(self.K):\n",
    "        # UCBs\n",
    "        theta_hat = self.irls_Theta[i,:]\n",
    "        inv_Gt = self.inv_Gt[i, :, :]\n",
    "        self.mu[i] = self.sigmoid(self.env.X[i, :].dot(theta_hat)) + self.cew * \\\n",
    "          np.sqrt((self.env.X[i, :].dot(inv_Gt).dot(self.env.X[i, :])))\n",
    "\n",
    "    arm = np.argmax(self.mu)\n",
    "    self.At = arm\n",
    "    \n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"UCB-GLM\"\n",
    "\n",
    "\n",
    "class LogTS(LogBanditAlg):\n",
    "  def __init__(self, env, n, params):\n",
    "    LogBanditAlg.__init__(self, env, n, params)\n",
    "    self.Grams = np.zeros((self.K, self.d, self.d))\n",
    "\n",
    "  def get_arm(self, t):\n",
    "    self.mu = np.zeros(self.K)\n",
    "    \n",
    "    if t==0:\n",
    "        for i in range(self.K):\n",
    "            thetabar, Gram = self.solve(i)\n",
    "            self.Grams[i,:,:] = Gram\n",
    "    else:\n",
    "        thetabar, Gram = self.solve(self.At)\n",
    "        self.Grams[self.At,:,:] = Gram\n",
    "        \n",
    "    for i in range(self.K):\n",
    "        # posterior sampling\n",
    "        Sigma_ti = np.linalg.inv(self.marLambda + self.Grams[i, :, :])\n",
    "        mu_ti = self.marLambda.dot(self.Theta0[i, :]) + self.Grams[i, :, :].dot(self.irls_Theta[i,:])\n",
    "        mu_ti = Sigma_ti.dot(mu_ti)\n",
    "        theta_tilde = np.random.multivariate_normal(mu_ti, Sigma_ti)\n",
    "        self.mu[i] = self.env.X[i, :].dot(theta_tilde)\n",
    "    arm = np.argmax(self.mu)\n",
    "    self.At = arm\n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"GLM-TSL\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class LinTS:\n",
    "  def __init__(self, env, n, params):\n",
    "    self.env = env  # bandit environment that the agent interacts with\n",
    "    self.K = self.env.K  # number of arms\n",
    "    self.d = self.env.d  # number of features\n",
    "    self.n = n  # horizon\n",
    "    self.mar_Sigma = np.copy(self.env.mar_Sigma)\n",
    "    self.sigma = self.env.sigma  # reward noise\n",
    "    \n",
    "    # override default values\n",
    "    for attr, val in params.items():\n",
    "      if isinstance(val, np.ndarray):\n",
    "        setattr(self, attr, np.copy(val))\n",
    "      elif isinstance(val, list):\n",
    "        setattr(self, attr, val.copy())\n",
    "      else:\n",
    "        setattr(self, attr, val)\n",
    "     \n",
    "    self.marLambda = np.linalg.inv(self.mar_Sigma)\n",
    "        \n",
    "    # sufficient statistics\n",
    "    self.G_hat = np.zeros((self.K, self.d, self.d))\n",
    "    self.B_hat = np.zeros((self.K, self.d))\n",
    "    \n",
    "  def update(self, t, arm, r):\n",
    "    # update sufficient statistics\n",
    "    x = self.env.X[arm, :]\n",
    "    self.G_hat[arm, :, :] += np.outer(x, x) / np.square(self.sigma)\n",
    "    self.B_hat[arm, :] += x * r / np.square(self.sigma)\n",
    "\n",
    "  def get_arm(self, t):\n",
    "    self.mu = np.zeros(self.K)\n",
    "    for i in range(self.K):\n",
    "      # linear model posterior\n",
    "      Sigma_hat = np.linalg.inv(self.marLambda + self.G_hat[i, :, :])\n",
    "      mu_ti = Sigma_hat.dot(self.B_hat[i, :]) #Because prior mean is 0\n",
    "      # posterior sampling\n",
    "      theta_tilde = np.random.multivariate_normal(mu_ti, Sigma_hat)\n",
    "      self.mu[i] = self.env.X[i, :].dot(theta_tilde)\n",
    "      \n",
    "    arm = np.argmax(self.mu)\n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"LinTS\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class LinUCB:\n",
    "  def __init__(self, env, n, params):\n",
    "    self.env = env  # bandit environment that the agent interacts with\n",
    "    self.K = self.env.K  # number of arms\n",
    "    self.d = self.env.d  # number of features\n",
    "    self.n = n  # horizon\n",
    "    self.Lambda = 1 / np.sum(self.env.Sigma)\n",
    "    self.S = self.env.S\n",
    "    self.sigma = self.env.sigma  # reward noise\n",
    "    \n",
    "    # override default values\n",
    "    for attr, val in params.items():\n",
    "      if isinstance(val, np.ndarray):\n",
    "        setattr(self, attr, np.copy(val))\n",
    "      elif isinstance(val, list):\n",
    "        setattr(self, attr, val.copy())\n",
    "      else:\n",
    "        setattr(self, attr, val)\n",
    "             \n",
    "    # sufficient statistics\n",
    "    self.G_hat = np.zeros((self.K, self.d, self.d))\n",
    "    self.B_hat = np.zeros((self.K, self.d))\n",
    "    \n",
    "    self.cew = self.confidence_ellipsoid_width(n)\n",
    "    \n",
    "  def update(self, t, arm, r):\n",
    "    # update sufficient statistics\n",
    "    x = self.env.X[arm, :]\n",
    "    self.G_hat[arm, :, :] += np.outer(x, x)\n",
    "    self.B_hat[arm, :] += x * r\n",
    "\n",
    "  def confidence_ellipsoid_width(self, t):\n",
    "    # Theorem 2 in Abassi-Yadkori (2011)\n",
    "    # Improved Algorithms for Linear Stochastic Bandits\n",
    "    delta = 1 / self.n\n",
    "    L = np.amax(np.linalg.norm(self.env.contexts, axis=1))\n",
    "    Lambda = self.Lambda\n",
    "    R = self.sigma\n",
    "    S = self.S\n",
    "    width = np.sqrt(Lambda) * S + R * np.sqrt(self.d * np.log((1 + t * np.square(L) / Lambda) / delta))\n",
    "    return width\n",
    "\n",
    "  def get_arm(self, t):\n",
    "    self.mu = np.zeros(self.K)\n",
    "    for i in range(self.K):\n",
    "      # linear model\n",
    "      Gt = self.Lambda * np.eye(self.d) + self.G_hat[i, :, :]\n",
    "      Sigma_hat = np.linalg.inv(Gt)\n",
    "      theta_hat = np.linalg.solve(Gt, self.B_hat[i, :])\n",
    "\n",
    "      # UCBs\n",
    "      self.mu[i] = self.env.X[i, :].dot(theta_hat) + self.cew * \\\n",
    "        np.sqrt(self.env.X[i, :].dot(Sigma_hat).dot(self.env.X[i, :]))\n",
    "\n",
    "    arm = np.argmax(self.mu)\n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"LinUCB\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluating GLBdTS 2730.0 seconds\n",
      "Regret: 146.12 +/- 11.78 (median: 126.50, max: 408.00, min: 29.00)\n",
      "Evaluating dTS 3093.1 seconds\n",
      "Regret: 2398.98 +/- 57.35 (median: 2446.00, max: 3372.00, min: 1385.00)\n",
      "Evaluating LinTS 3062.6 seconds\n",
      "Regret: 2445.26 +/- 55.64 (median: 2425.50, max: 3393.00, min: 1311.00)\n",
      "Evaluating GLM-TSL 3312.2 seconds\n",
      "Regret: 1457.82 +/- 43.76 (median: 1457.00, max: 1957.00, min: 717.00)\n",
      "Evaluating UCB-GLM 554.6 seconds\n",
      "Regret: 2465.72 +/- 135.86 (median: 2237.50, max: 4595.00, min: 179.00)\n",
      "Evaluating LinUCB 661.2 seconds\n",
      "Regret: 2452.96 +/- 138.68 (median: 2233.50, max: 4731.00, min: 225.00)\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 400x240 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluating GLBdTS"
     ]
    }
   ],
   "source": [
    "sigma = 1\n",
    "n = 5000\n",
    "num_runs = 50\n",
    "\n",
    "algs = [\n",
    " (\"GLBdTS\", {}, \"orange\", \"-\", \"GLM-dTS\"),\n",
    " (\"dTS\", {}, \"red\", \"-\", \"LindTS\"),\n",
    " (\"HierTS\", {}, \"green\", \"-\", \"HierTS\"),\n",
    " (\"LinTS\", {}, \"blue\", \"-\", \"LinTS\"),\n",
    " (\"LogTS\", {}, \"grey\", \"-\", \"GLM-TS\"),\n",
    " (\"UCBLog\", {}, \"purple\", \"-\", \"UCB-GLM\"),\n",
    " (\"LinUCB\", {}, \"cyan\", \"-\", \"LinUCB\"),\n",
    "]\n",
    "\n",
    "exps = [\n",
    "    {\"K\": 1000, \"d_l\": [20, 10, 5, 2], \"d\": 20}, \n",
    "    #{\"K\": 10000, \"d_l\": [20, 10, 5, 2], \"d\": 20}, \n",
    "]\n",
    "\n",
    "for exp in exps:\n",
    "  # set parameters of the experiment\n",
    "  for attr, val in exp.items():\n",
    "    globals()[attr] = val\n",
    "    \n",
    "  L = len(d_l) # depth of the diffusion\n",
    "\n",
    "  # bandit environments\n",
    "  envs = []\n",
    "  for run in range(num_runs):\n",
    "        \n",
    "    # possible contexts\n",
    "    contexts = 2 * np.random.rand(100, d) - 1\n",
    "        \n",
    "    W = []\n",
    "    for l in range(L):\n",
    "        W_helper = 2 * np.random.rand(d, d) - 1\n",
    "        W_helper[d_l[l]:] = 0\n",
    "        W.append(W_helper)\n",
    "    \n",
    "    # set the covariances\n",
    "    Sigma = np.zeros(L+1) \n",
    "    for l in range(L+1):\n",
    "        Sigma[l] = 1 #constant covariances\n",
    "    \n",
    "    mar_Sigma = Sigma[0] * np.eye(d)\n",
    "    B_l = np.eye(d)\n",
    "    mar_Sigma_1 = np.copy(mar_Sigma) #if L=1, then mar_Sigma_1 = Sigma[0] * np.eye(d)\n",
    "    for l in range(L):\n",
    "        B_l = B_l.dot(W[l])\n",
    "        B_l_transpose = B_l.T\n",
    "        mar_Sigma += Sigma[l+1] * B_l.dot(B_l_transpose)\n",
    "        if l==L-2:\n",
    "            mar_Sigma_1 = np.copy(mar_Sigma) #if L>1, this condition is triggered and mar_Sigma_1 = mar_Sigma wihtout the last covariance\n",
    "        \n",
    "    # generate latent parameters and action parameters\n",
    "    psi = np.random.multivariate_normal(np.zeros(d), Sigma[L] * np.eye(d))\n",
    "    for l in range(L-1, 0, -1):\n",
    "        psi = np.random.multivariate_normal(W[l].dot(psi), Sigma[l] * np.eye(d))\n",
    "    \n",
    "    Theta = np.zeros((K, d))\n",
    "    for i in range(K):\n",
    "        Theta[i, :] = np.random.multivariate_normal(W[0].dot(psi),  Sigma[0] * np.eye(d))\n",
    "        \n",
    "    # initialize bandit environment\n",
    "    env = CoBandit(K, contexts, Theta, sigma=sigma)\n",
    "\n",
    "    # pass parameters for algorithm initialization\n",
    "    env.L = L\n",
    "    env.W = W\n",
    "    env.Sigma = Sigma\n",
    "    env.mar_Sigma = mar_Sigma\n",
    "    env.mar_Sigma_1 = mar_Sigma_1\n",
    "    env.B_L = B_l\n",
    "    env.S = np.linalg.norm(Theta, axis=1).max()\n",
    "    envs.append(env)\n",
    "\n",
    "  # simulation\n",
    "  for alg in algs:\n",
    "    # all runs for a single algorithm\n",
    "    alg_class = globals()[alg[0]]\n",
    "    regret, logs = evaluate(alg_class, alg[1], envs, n)\n",
    "    #regret, logs = evaluate_one(alg_class, alg[1], envs[0], n)\n",
    "\n",
    "    # # save results    \n",
    "    fname = \"results_glb/K={}_d_l={}_d={}\".format(K, d_l, d)\n",
    "    isExist = os.path.exists(fname)\n",
    "    if not isExist:\n",
    "        os.makedirs(fname)\n",
    "    np.save(fname + \"/{}.npy\".format(alg[4]), regret)\n",
    "\n",
    "    # plot\n",
    "    cum_regret = regret.cumsum(axis=0)\n",
    "    step = np.arange(1, n + 1)\n",
    "    sube = (step.size // 10) * np.arange(1, 11) - 1\n",
    "    plt.plot(step, cum_regret.mean(axis=1),\n",
    "      alg[2], dashes=linestyle2dashes(alg[3]), label=alg[4])\n",
    "    plt.errorbar(step[sube], cum_regret[sube, :].mean(axis=1),\n",
    "      cum_regret[sube, :].std(axis=1) / np.sqrt(cum_regret.shape[1]),\n",
    "      fmt=\"none\", ecolor=alg[2])\n",
    "\n",
    "  plt.legend(loc=\"upper left\", frameon=False)\n",
    "  plt.title(\"Linear difusion, K = %d, L=%d, d_l =\" % (K, L) + str(d_l) +  \", d = %d\" % (d))\n",
    "  plt.xlabel(\"Round n\")\n",
    "  plt.ylabel(\"Regret\")\n",
    "\n",
    "  plt.tight_layout()\n",
    "  plt.show()"
   ]
  }
 ],
 "metadata": {
  "colab": {
   "collapsed_sections": [],
   "name": "GenHierTS.ipynb",
   "provenance": []
  },
  "kernelspec": {
   "display_name": "Python Kernel (MOAB #56460)",
   "language": "python",
   "name": "python-kernel-56460"
  },
  "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.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
