{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "GiOSzG6ch663",
    "outputId": "c6ab9c0e-181f-4e5d-8aba-73aee33014f7"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "python 3.7.16\n",
      "matplotlib 3.5.3\n"
     ]
    }
   ],
   "source": [
    "\n",
    "import collections\n",
    "import copy\n",
    "import itertools\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "from scipy.special import loggamma\n",
    "from scipy.stats import beta\n",
    "import time\n",
    "\n",
    "\n",
    "# mpl.style.use(\"classic\")\n",
    "mpl.style.use('seaborn-darkgrid')\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\"] = 12\n",
    "mpl.rcParams[\"axes.titlesize\"] = \"medium\"\n",
    "mpl.rcParams[\"legend.fontsize\"] = \"medium\"\n",
    "mpl.rcParams['text.usetex'] = True\n",
    "# mpl.rcParams['text.latex.unicode'] = True\n",
    "plt.rc('text', usetex=True) #Use latex\n",
    "\n",
    "\n",
    "import platform\n",
    "print(\"python %s\" % platform.python_version())\n",
    "print(\"matplotlib %s\" % mpl.__version__)\n",
    "\n",
    "import tensorflow.compat.v2 as tf\n",
    "# import tensorflow_probability as tfp\n",
    "tf.enable_v2_behavior()\n",
    "from tensorflow import keras\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": 2,
   "metadata": {
    "cellView": "form",
    "id": "aQKjSr2UzUYq"
   },
   "outputs": [],
   "source": [
    "#@title Bandit simulator and environments\n",
    "class GaussBandit(object):\n",
    "  \"\"\"Gaussian bandit.\"\"\"\n",
    "\n",
    "  def __init__(self, mu, sigma=0.5):\n",
    "    self.mu = np.copy(mu)\n",
    "    self.K = self.mu.size\n",
    "    self.sigma = sigma\n",
    "\n",
    "    self.best_arm = np.argmax(self.mu)\n",
    "\n",
    "    self.randomize()\n",
    "\n",
    "  def randomize(self):\n",
    "    # generate random rewards\n",
    "    self.rt = self.mu + self.sigma * np.random.randn(self.K)\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.mu[self.best_arm] - self.mu[arm]\n",
    "\n",
    "  def print(self):\n",
    "    return \"Gaussian bandit with arms (%s)\" % \\\n",
    "      \", \".join(\"%.3f\" % s for s in self.mu)\n",
    "\n",
    "\n",
    "class LinBandit(object):\n",
    "  \"\"\"Linear bandit.\"\"\"\n",
    "\n",
    "  def __init__(self, X, theta, noise=\"normal\", sigma=0.5):\n",
    "    self.X = np.copy(X)\n",
    "    self.K = self.X.shape[0]\n",
    "    self.d = self.X.shape[1]\n",
    "    self.theta = np.copy(theta)\n",
    "    self.noise = noise\n",
    "    if self.noise == \"normal\":\n",
    "      self.sigma = sigma\n",
    "\n",
    "    self.mu = self.X.dot(self.theta)\n",
    "    self.best_arm = np.argmax(self.mu)\n",
    "\n",
    "    self.randomize()\n",
    "\n",
    "  def randomize(self):\n",
    "    # generate random rewards\n",
    "    if self.noise == \"normal\":\n",
    "      self.rt = self.mu + self.sigma * np.random.randn(self.K)\n",
    "    elif self.noise == \"bernoulli\":\n",
    "      self.rt = (np.random.rand(self.K) < self.mu).astype(float)\n",
    "    elif self.noise == \"beta\":\n",
    "      self.rt = np.random.beta(4 * self.mu, 4 * (1 - self.mu))\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.mu[self.best_arm] - self.mu[arm]\n",
    "\n",
    "  def print(self):\n",
    "    if self.noise == \"normal\":\n",
    "      return \"Linear bandit: %d dimensions, %d arms\" % \\\n",
    "        (self.d, self.K)\n",
    "    elif self.noise == \"bernoulli\":\n",
    "      return \"Bernoulli linear bandit: %d dimensions, %d arms\" % \\\n",
    "        (self.d, self.K)\n",
    "    elif self.noise == \"beta\":\n",
    "      return \"Beta linear bandit: %d dimensions, %d arms\" % \\\n",
    "        (self.d, self.K)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "cellView": "form",
    "id": "B-9Qk7f1zZDU"
   },
   "outputs": [],
   "source": [
    "#@title Baseline algorithms that handle each task independently\n",
    "\n",
    "class GaussTS(object):\n",
    "  def __init__(self, K, params):\n",
    "    self.K = K\n",
    "    self.sigma = 0.5\n",
    "\n",
    "    self.mu0 = np.zeros(self.K)\n",
    "    self.sigma0 = 0.5 * np.ones(self.K)\n",
    "\n",
    "    for attr, val in params.items():\n",
    "      setattr(self, attr, val)\n",
    "\n",
    "    self.pulls = np.zeros(self.K)  # number of pulls\n",
    "    self.reward = np.zeros(self.K)  # cumulative reward\n",
    "\n",
    "  def update(self, t, arm, r):\n",
    "    self.pulls[arm] += 1\n",
    "    self.reward[arm] += r\n",
    "\n",
    "  def get_arm(self, t):\n",
    "    if t < self.K:\n",
    "      # each arm is initially pulled once\n",
    "      self.mu = np.zeros(self.K)\n",
    "      self.mu[t] = 1\n",
    "    else:\n",
    "      # posterior distribution\n",
    "      sigma2 = np.square(self.sigma)\n",
    "      sigma02 = np.square(self.sigma0)\n",
    "      post_var = 1.0 / (1.0 / sigma02 + self.pulls / sigma2)\n",
    "      post_mean = post_var * (self.mu0 / sigma02 + self.reward / sigma2)\n",
    "\n",
    "      # posterior sampling\n",
    "      self.mu = post_mean + np.sqrt(post_var) * np.random.randn(self.K)\n",
    "\n",
    "    arm = np.argmax(self.mu)\n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"Gaussian TS\"\n",
    "\n",
    "\n",
    "class LinBanditAlg(object):\n",
    "  def __init__(self, K, d, params):\n",
    "    self.K = K\n",
    "    self.d = d\n",
    "    self.theta0 = np.zeros(self.d)\n",
    "    self.sigma0 = 1.0\n",
    "    self.sigma = 0.5\n",
    "    self.crs = 1.0 # confidence region scaling\n",
    "\n",
    "    for attr, val in params.items():\n",
    "      setattr(self, attr, val)\n",
    "\n",
    "    if not hasattr(self, \"Sigma0\"):\n",
    "      self.Sigma0 = np.square(self.sigma0) * np.eye(self.d)\n",
    "\n",
    "    # sufficient statistics\n",
    "    self.Gram = np.linalg.inv(self.Sigma0)\n",
    "    self.B = self.Gram.dot(self.theta0)\n",
    "\n",
    "  def update(self, t, x, arm, r):\n",
    "    x_a = x[arm]\n",
    "    self.Gram += np.outer(x_a, x_a) / np.square(self.sigma)\n",
    "    self.B += x_a * r / np.square(self.sigma)\n",
    "\n",
    "\n",
    "class LinTS(LinBanditAlg):\n",
    "  def get_arm(self, t, x):\n",
    "    Gram_inv = np.linalg.inv(self.Gram)\n",
    "    thetabar = Gram_inv.dot(self.B)\n",
    "\n",
    "    # posterior sampling\n",
    "    thetatilde = np.random.multivariate_normal(thetabar,\n",
    "      np.square(self.crs) * Gram_inv, tol=1e-6)\n",
    "    self.mu = x.dot(thetatilde)\n",
    "\n",
    "    arm = np.argmax(self.mu)\n",
    "    return arm\n",
    "\n",
    "  @staticmethod\n",
    "  def print():\n",
    "    return \"LinTS\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "id": "tF2d1ViMEAzW"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 25 µs, sys: 0 ns, total: 25 µs\n",
      "Wall time: 27.7 µs\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "class IndLinTS(object):\n",
    "  def __init__(self, num_tasks, K, d, params):\n",
    "      self.num_tasks = num_tasks\n",
    "      self.K = K\n",
    "      self.d = d\n",
    "\n",
    "      # set up independent bandit algorithms\n",
    "      self.algs = []\n",
    "      for _ in range(num_tasks):\n",
    "        self.algs.append(LinTS(K, d, copy.deepcopy(params)))\n",
    "\n",
    "  def update(self, t, tasks, xs, arms, rs):\n",
    "    for s, x, arm, r in zip(tasks, xs, arms, rs):\n",
    "      self.algs[s].update(t, x, arm, r)\n",
    "\n",
    "  def get_arm(self, t, tasks, xs):\n",
    "    return [self.algs[s].get_arm(t, x) for s, x in zip(tasks, xs)]\n",
    "\n",
    "\n",
    "class HierLinTS(object):\n",
    "  def __init__(self, num_tasks, K, d, params):\n",
    "      self.num_tasks = num_tasks\n",
    "      self.K = K\n",
    "      self.d = d\n",
    "      self.mu_q = np.zeros(self.d)\n",
    "      self.Sigma_q = np.eye(self.d)\n",
    "      self.sigma0 = 1.0\n",
    "      self.sigma = 0.5\n",
    "      self.crs = 1.0  # confidence region scaling\n",
    "\n",
    "      for attr, val in params.items():\n",
    "        setattr(self, attr, val)\n",
    "\n",
    "      if not hasattr(self, \"Sigma0\"):\n",
    "        self.Sigma0 = np.square(self.sigma0) * np.eye(self.d)\n",
    "\n",
    "      # hyper-posterior\n",
    "      self.mu_tildes = np.tile(self.mu_q, (self.num_tasks, 1))\n",
    "      self.Sigma_tildes = np.tile(self.Sigma_q, (self.num_tasks, 1, 1))\n",
    "  \n",
    "      # sufficient statistics\n",
    "      self.Grams= (np.zeros((self.num_tasks, self.d, self.d)) +\n",
    "                   1e-6 * np.eye(self.d)[np.newaxis, ...])\n",
    "      self.Bs = np.zeros((self.num_tasks, self.d))\n",
    "      self.counts = np.zeros(self.num_tasks)\n",
    "\n",
    "\n",
    "  def update(self, t, tasks, xs, arms, rs):\n",
    "    for s, x, arm, r in zip(tasks, xs, arms, rs):\n",
    "      x_a = x[arm]\n",
    "      self.Grams[s] += np.outer(x[arm], x[arm]) / np.square(self.sigma)\n",
    "      self.Bs[s] += x[arm] * r / np.square(self.sigma)\n",
    "      self.counts[s] += 1\n",
    "\n",
    "    # hyper-posterior update\n",
    "    mu_h = np.linalg.solve(self.Sigma_q, self.mu_q)\n",
    "    Lambda_h = np.linalg.inv(self.Sigma_q)\n",
    "\n",
    "    # compute hyper-posterior parameters\n",
    "    for s in range(self.num_tasks):\n",
    "      if self.counts[s] >= self.d:\n",
    "        Gram = self.Grams[s]\n",
    "        B = self.Bs[s]\n",
    "        M = np.linalg.pinv(np.linalg.inv(self.Sigma0) + Gram)\n",
    "        Lambda_h += Gram - Gram.dot(M).dot(Gram)\n",
    "        mu_h += B - Gram.dot(M).dot(B)\n",
    "\n",
    "    for s in range(self.num_tasks):\n",
    "      mu_h_s = np.copy(mu_h)\n",
    "      Lambda_h_s = np.copy(Lambda_h)\n",
    "      if self.counts[s] >= self.d:\n",
    "        Gram = self.Grams[s]\n",
    "        B = self.Bs[s]\n",
    "        M = np.linalg.pinv(np.linalg.inv(self.Sigma0) + Gram)\n",
    "        # subtract observations from task to keep independence\n",
    "        mu_h_s -= (B - Gram.dot(M).dot(B))\n",
    "        Lambda_h_s -= (Gram - Gram.dot(M).dot(Gram))\n",
    "\n",
    "      self.mu_tildes[s] = np.linalg.solve(Lambda_h_s, mu_h_s)\n",
    "      self.Sigma_tildes[s] = np.linalg.pinv(Lambda_h_s)\n",
    "\n",
    "  def get_arm(self, t, tasks, xs):\n",
    "    arms = []\n",
    "    for s, x in zip(tasks, xs):\n",
    "      Gram = self.Grams[s]\n",
    "      B = self.Bs[s]\n",
    "      Sigma_tilde = self.Sigma_tildes[s]\n",
    "      mu_tilde = self.mu_tildes[s]\n",
    "\n",
    "      thetatilde_s = np.linalg.solve(Sigma_tilde, mu_tilde)\n",
    "      Lambda_hat_s = np.linalg.pinv(self.Sigma0 + Sigma_tilde) + Gram\n",
    "      thetabar_hat_s = np.linalg.solve(Lambda_hat_s, thetatilde_s + B)\n",
    "      Sigma_hat_s = np.linalg.pinv(Lambda_hat_s)\n",
    "\n",
    "      # posterior sampling\n",
    "      thetasample_s = np.random.multivariate_normal(\n",
    "          thetabar_hat_s, np.square(self.crs) * Sigma_hat_s)\n",
    "      mu = x.dot(thetasample_s)\n",
    "\n",
    "      arms.append(np.argmax(mu))\n",
    "    return arms\n",
    "\n",
    "class HierBayesUCB(object):\n",
    "  def __init__(self, num_tasks, K, d, params, delta):\n",
    "      self.num_tasks = num_tasks\n",
    "      self.K = K\n",
    "      self.d = d\n",
    "      self.mu_q = np.zeros(self.d)\n",
    "      self.Sigma_q = np.eye(self.d)\n",
    "      self.sigma0 = 1.0\n",
    "      self.sigma = 0.5\n",
    "      self.delta = delta\n",
    "      self.crs = 1.0  # confidence region scaling\n",
    "\n",
    "      for attr, val in params.items():\n",
    "        setattr(self, attr, val)\n",
    "\n",
    "      if not hasattr(self, \"Sigma0\"):\n",
    "        self.Sigma0 = np.square(self.sigma0) * np.eye(self.d)\n",
    "\n",
    "      # hyper-posterior\n",
    "      self.bar_mu = np.copy(self.mu_q)\n",
    "      self.bar_Sigma = np.copy(self.Sigma_q)\n",
    "  \n",
    "      # sufficient statistics\n",
    "      self.Grams= (np.zeros((self.num_tasks, self.d, self.d)) +\n",
    "                   1e-6 * np.eye(self.d)[np.newaxis, ...])\n",
    "      self.Bs = np.zeros((self.num_tasks, self.d))\n",
    "      self.counts = np.zeros(self.num_tasks)\n",
    "\n",
    "\n",
    "  def update(self, t, tasks, xs, arms, rs):\n",
    "    for s, x, arm, r in zip(tasks, xs, arms, rs):\n",
    "      x_a = x[arm]\n",
    "      self.Grams[s] += np.outer(x[arm], x[arm]) / np.square(self.sigma)\n",
    "      self.Bs[s] += x[arm] * r / np.square(self.sigma)\n",
    "      self.counts[s] += 1\n",
    "\n",
    "    # hyper-posterior update\n",
    "    mu_h = np.linalg.solve(self.Sigma_q, self.mu_q)\n",
    "    Lambda_h = np.linalg.inv(self.Sigma_q)\n",
    "\n",
    "    # compute hyper-posterior parameters\n",
    "    for s in range(self.num_tasks):\n",
    "      if self.counts[s] >= self.d:\n",
    "        Gram = self.Grams[s]\n",
    "        B = self.Bs[s]\n",
    "        M = np.linalg.pinv(np.linalg.inv(self.Sigma0) + Gram)\n",
    "        Lambda_h += Gram - Gram.dot(M).dot(Gram)\n",
    "        mu_h += B - Gram.dot(M).dot(B)\n",
    "\n",
    "    self.bar_Sigma = np.linalg.pinv(Lambda_h)\n",
    "    # print(\"mu_h shape\", mu_h.shape)\n",
    "    self.bar_mu = self.bar_Sigma.dot(mu_h.reshape(-1,1))\n",
    "\n",
    "  def get_arm(self, t, tasks, xs):\n",
    "    arms = []\n",
    "    for s, x in zip(tasks, xs):\n",
    "      Gram = self.Grams[s]\n",
    "      B = self.Bs[s]\n",
    "      M = np.linalg.pinv(np.linalg.inv(self.Sigma0) + Gram)\n",
    "      A = self.bar_mu\n",
    "      # print(\" M shape\", M.shape) #(4,4)\n",
    "      # print(\"self bar_mu shape\", A.shape)# (4,)\n",
    "      # print(\"B shape\", B.shape) #(4,)\n",
    "      hat_mu_s = M.dot(np.linalg.pinv(self.Sigma0).dot(self.bar_mu.reshape(-1,1)) +B.reshape(-1,1))\n",
    "      hat_Sigma_s = M + M.dot(np.linalg.pinv(self.Sigma0)).dot(self.bar_Sigma).dot(np.linalg.pinv(self.Sigma0)).dot(M)\n",
    "      # print(s,x,'yes')\n",
    "      # print(\"x shape\", x.shape) # (20, 4)\n",
    "      # print(\"hat_mu_s shape\", hat_mu_s.shape) # (4,4)\n",
    "      # print(\"hat_Sigma_s shape\", hat_Sigma_s.shape) # (4,4)\n",
    "      diag_x_Sigma_x = np.diagonal(x.dot(hat_Sigma_s).dot(x.T))\n",
    "      UCB_s = x.dot(hat_mu_s) + np.sqrt(2 * np.log(1./self.delta)) * np.sqrt(diag_x_Sigma_x.reshape(-1,1))\n",
    "      # print('yes_second')\n",
    "      arms.append(np.argmax(UCB_s))\n",
    "    return arms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 509
    },
    "id": "cuyuQsXLZ8mb",
    "outputId": "66d35493-c129-467d-a0c6-59dd771db5ab"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Regrets of HierBayesUCB: 304.7 +/- 6.2 with variance_0: 0.50\n",
      "Regrets of HierBayesUCB: 268.9 +/- 5.8 with variance_0: 0.40\n",
      "Regrets of HierBayesUCB: 225.1 +/- 6.3 with variance_0: 0.30\n",
      "Regrets of HierBayesUCB: 151.0 +/- 5.5 with variance_0: 0.20\n",
      "Regrets of HierBayesUCB: 103.4 +/- 4.3 with variance_0: 0.10\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x280 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# linear bandit\n",
    "alg_spec = (\"HierBayesUCB\", \"green\", \"-\")\n",
    "color_list = [\"cyan\", \"blue\", \"green\", \"red\", \"purple\"]\n",
    "num_runs = 100 # calculate the average\n",
    "num_tasks = 10\n",
    "num_tasks_per_round = 5\n",
    "n = 200 * num_tasks // num_tasks_per_round # total number of iterations\n",
    "delta = 1./(num_tasks * n)\n",
    "dim = 4\n",
    "# sigma_q_scales = [0.5, 1, 5, 10]\n",
    "sigma_q_scale = 1\n",
    "# sigma_0_scales = [1.5, 1.4, 1.3, 1.2, 1.1]\n",
    "sigma_0_scales = [0.5, 0.4, 0.3, 0.2, 0.1]\n",
    "sigma = 1\n",
    "\n",
    "step = np.arange(1, n + 1)\n",
    "sube = (step.size // 10) * np.arange(1, 11) - 1\n",
    "\n",
    "flag = 0\n",
    "for d in [dim]:\n",
    "  K = 5 * d\n",
    "  plt.figure(figsize=(4, 2.8))\n",
    "  for sigma_0 in sigma_0_scales:\n",
    "    # meta-prior parameters\n",
    "    mu_q = np.zeros(d)\n",
    "    # prior parameters\n",
    "    # reward noise\n",
    "    Sigma_q = np.square(sigma_q_scale) * np.eye(d)\n",
    "    Sigma_0 = np.square(sigma_0) * np.eye(d)\n",
    "\n",
    "    regret = np.zeros((n, num_runs))\n",
    "    for run in range(num_runs):\n",
    "      # true hyper-prior\n",
    "      mu_star = mu_q + sigma_q_scale * np.random.randn(d)\n",
    "      envs = []\n",
    "      for _ in range(num_tasks):\n",
    "        # sample problem instance from N(\\mu_*, \\sigma_0^2 I_d)\n",
    "        theta = mu_star + sigma_0 * np.random.randn(d)\n",
    "        # sample arms from a unit ball\n",
    "        X = np.random.randn(K, d)\n",
    "        X /= np.linalg.norm(X, axis=-1)[:, np.newaxis]\n",
    "        envs.append(LinBandit(X, theta, sigma=sigma))\n",
    "\n",
    "      # initialize algorithms\n",
    "      # HierBayesUCB\n",
    "      alg_params = {\n",
    "          \"mu_q\": np.copy(mu_q),\n",
    "          \"Sigma_q\": np.copy(Sigma_q),\n",
    "          \"Sigma0\": np.copy(Sigma_0),\n",
    "          \"sigma\": sigma,\n",
    "      }\n",
    "      alg = HierBayesUCB(num_tasks, K, d, alg_params, delta)\n",
    "        \n",
    "\n",
    "      for t in range(n):\n",
    "        tasks = np.random.randint(0, num_tasks, size=num_tasks_per_round)\n",
    "\n",
    "        for s in tasks:\n",
    "          envs[s].randomize()\n",
    "\n",
    "        Xs = [envs[s].X for s in tasks]\n",
    "        arms = alg.get_arm(t, tasks, Xs)\n",
    "        rs = [envs[s].reward(arm) for s, arm in zip(tasks, arms)]\n",
    "        alg.update(t, tasks, Xs, arms, rs)\n",
    "        regret[t, run] = np.sum(\n",
    "            [envs[s].regret(arm) for s, arm in zip(tasks, arms)])\n",
    "\n",
    "    cum_regret = regret.cumsum(axis=0)\n",
    "    if flag <= 3:\n",
    "      plt.plot(step, cum_regret.mean(axis=1),\n",
    "        dashes=linestyle2dashes(alg_spec[2]), color=color_list[flag],\n",
    "        label=r'$\\sigma_{0}$=%.1f'%(sigma_0))\n",
    "    else:\n",
    "      plt.plot(step, cum_regret.mean(axis=1),\n",
    "        dashes=linestyle2dashes(alg_spec[2]), color=color_list[flag],\n",
    "        label=r'$\\sigma_{0}$=%.1f'%(sigma_0))\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=color_list[flag], capsize=1)\n",
    "\n",
    "    print(\"Regrets of %s: %.1f +/- %.1f with variance_0: %.2f\" % (alg_spec[0],\n",
    "      cum_regret[-1, :].mean(),\n",
    "      cum_regret[-1, :].std() / np.sqrt(cum_regret.shape[1]), sigma_0))\n",
    "    flag +=1\n",
    "\n",
    "  plt.title(r\"Linear Bandit ($d$ = %d, $m$ = %d, $L$=%d)\" % (d, num_tasks, num_tasks_per_round))\n",
    "  plt.xlabel(\"Round $t$\")\n",
    "  plt.xticks(np.arange(n + 1, step=100))\n",
    "  plt.ylabel(\"Regret\")\n",
    "  plt.ylim(bottom=0)\n",
    "  plt.legend(loc=\"upper left\", frameon=False, prop={'size': 10})\n",
    "  # plt.gcf().set_facecolor('white')\n",
    "  plt.tight_layout()\n",
    "  plt.tick_params(axis='both', which='both', length=0)\n",
    "  plt.savefig('regrets_for_HierBayesUCB_of_different_variances_sigma_0.pdf', transparent = False)\n",
    "  plt.show()"
   ]
  }
 ],
 "metadata": {
  "colab": {
   "collapsed_sections": [],
   "name": "HierTS_Public.ipynb",
   "provenance": []
  },
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.16"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
