{
 "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",
    "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 35 µs, sys: 0 ns, total: 35 µs\n",
      "Wall time: 37.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: 1043.3 +/- 68.8 with variance: 10.00\n",
      "Regrets of HierBayesUCB: 798.0 +/- 52.6 with variance: 7.00\n",
      "Regrets of HierBayesUCB: 432.8 +/- 25.2 with variance: 4.00\n",
      "Regrets of HierBayesUCB: 92.7 +/- 4.3 with variance: 1.00\n",
      "Regrets of HierBayesUCB: 38.1 +/- 0.9 with variance: 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_scales = [10, 7, 4, 1, 0.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 in sigma_scales:\n",
    "    # meta-prior parameters\n",
    "    mu_q = np.zeros(d)\n",
    "    # prior parameters\n",
    "    sigma_0 = 0.1\n",
    "    # reward noise\n",
    "    # sigma = 1\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",
    "      # HierTS\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$=%d'%(sigma))\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$=%.1f'%(sigma))\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: %.2f\" % (alg_spec[0],\n",
    "      cum_regret[-1, :].mean(),\n",
    "      cum_regret[-1, :].std() / np.sqrt(cum_regret.shape[1]), sigma))\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.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
}
