{
 "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 22 µs, sys: 3 µs, total: 25 µs\n",
      "Wall time: 28.4 µ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": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 509
    },
    "id": "cuyuQsXLZ8mb",
    "outputId": "66d35493-c129-467d-a0c6-59dd771db5ab"
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "colab": {
     "background_save": true
    },
    "id": "-n5kZrFvKPFh"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "L = 1: 127.5 +/- 2.1\n",
      "L = 2: 131.5 +/- 2.1\n",
      "L = 3: 130.8 +/- 2.2\n",
      "L = 4: 129.9 +/- 2.3\n",
      "L = 5: 139.4 +/- 1.9\n",
      "L = 6: 137.8 +/- 2.0\n",
      "L = 7: 145.4 +/- 2.1\n",
      "L = 8: 146.7 +/- 2.2\n",
      "L = 9: 150.1 +/- 2.2\n",
      "L = 10: 156.6 +/- 2.5\n",
      "L = 1: 61.1 +/- 1.7\n",
      "L = 2: 66.1 +/- 1.8\n",
      "L = 3: 63.0 +/- 1.8\n",
      "L = 4: 66.4 +/- 1.5\n",
      "L = 5: 65.0 +/- 1.5\n",
      "L = 6: 71.3 +/- 1.3\n",
      "L = 7: 73.2 +/- 2.2\n",
      "L = 8: 73.8 +/- 1.7\n",
      "L = 9: 79.0 +/- 1.6\n",
      "L = 10: 83.7 +/- 1.5\n",
      "L = 1: 32.9 +/- 2.0\n",
      "L = 2: 28.1 +/- 2.0\n",
      "L = 3: 27.3 +/- 1.7\n",
      "L = 4: 30.3 +/- 2.0\n",
      "L = 5: 29.3 +/- 1.5\n",
      "L = 6: 37.3 +/- 2.0\n",
      "L = 7: 33.0 +/- 1.5\n",
      "L = 8: 39.0 +/- 1.7\n",
      "L = 9: 40.6 +/- 1.8\n",
      "L = 10: 39.1 +/- 1.6\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x280 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "#@title\n",
    "# linear bandit\n",
    "alg_specs = [\n",
    "  (\"HierBayesUCB\", \"green\", \"-\")]\n",
    "\n",
    "color_list = [\"cyan\", \"blue\", \"green\", \"red\", \"purple\"]\n",
    "\n",
    "d_scales = [8,4,2]\n",
    "flag = 0\n",
    "plt.figure(figsize=(4, 2.8))\n",
    "for d in d_scales:\n",
    "  K = 5 * d\n",
    "\n",
    "  num_runs = 100\n",
    "  num_tasks = 10\n",
    "  sigma_q_scale = 1\n",
    "\n",
    "  # meta-prior parameters\n",
    "  mu_q = np.zeros(d)\n",
    "  sigma_q = sigma_q_scale * np.ones(d)\n",
    "  # prior parameters\n",
    "  sigma_0 = 0.1 * np.ones(d)\n",
    "  # reward noise\n",
    "  sigma = 0.5\n",
    "\n",
    "  Sigma_q = np.diag(np.square(sigma_q))\n",
    "  Sigma_0 = np.diag(np.square(sigma_0))\n",
    "\n",
    "  all_num_tasks_per_round = np.arange(1, num_tasks + 1)\n",
    "  total_regret = np.zeros((len(all_num_tasks_per_round), num_runs)) \n",
    "\n",
    "  for i, num_tasks_per_round in enumerate(all_num_tasks_per_round):\n",
    "    K = 5 * d\n",
    "    n = num_tasks * 200 // num_tasks_per_round\n",
    "    delta = 1./(num_tasks * n)\n",
    "\n",
    "    for alg_spec in alg_specs:\n",
    "      regret = np.zeros((n, num_runs))\n",
    "\n",
    "      for run in range(num_runs):\n",
    "        # true hyper-prior\n",
    "        mu_star = mu_q + sigma_q * np.random.randn(d)\n",
    "\n",
    "        envs = []\n",
    "        for task 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",
    "        if alg_spec[0] == 'OracleTS':\n",
    "          # OracleTS\n",
    "          alg_params = {\n",
    "            \"theta0\": np.copy(mu_star),\n",
    "            \"Sigma0\": np.copy(Sigma_0),\n",
    "            \"sigma\": sigma}\n",
    "          alg = IndLinTS(num_tasks, K, d, alg_params)\n",
    "        elif alg_spec[0] == \"TS\":\n",
    "          # TS\n",
    "          alg_params = {\n",
    "            \"theta0\": np.copy(mu_q),\n",
    "            \"Sigma0\": Sigma_q + Sigma_0,\n",
    "            \"sigma\": sigma}\n",
    "          alg = IndLinTS(num_tasks, K, d, alg_params)\n",
    "\n",
    "        elif alg_spec[0] == 'HierTS':\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 = HierLinTS(num_tasks, K, d, alg_params)\n",
    "\n",
    "        else:\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",
    "        for t in range(n):\n",
    "          tasks = np.random.choice(\n",
    "              np.arange(num_tasks), size=num_tasks_per_round, replace=False)\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",
    "\n",
    "          regret[t, run] = np.sum(\n",
    "              [envs[s].regret(arm) for s, arm in zip(tasks, arms)])\n",
    "\n",
    "      total_regret[i] = regret.cumsum(axis=0)[-1, :]\n",
    "      print(\"L = %d: %.1f +/- %.1f\" % (num_tasks_per_round,\n",
    "        total_regret[i].mean(),\n",
    "        total_regret[i].std() / np.sqrt(total_regret.shape[1])))\n",
    "\n",
    "  \n",
    "  plt.plot(all_num_tasks_per_round, total_regret.mean(axis=1),\n",
    "    dashes=linestyle2dashes(alg_spec[2]), color=color_list[flag],\n",
    "    label=r'$d$=%d'%(d), )\n",
    "  plt.errorbar(all_num_tasks_per_round, total_regret.mean(axis=1),\n",
    "    total_regret.std(axis=1) / np.sqrt(total_regret.shape[1]),\n",
    "    fmt=\"none\", ecolor=color_list[flag],capsize=2)\n",
    "  flag += 1\n",
    "\n",
    "plt.title(r\"Linear Bandit ($m$ = %d, $\\sigma_q$ = %.1f)\" % (num_tasks, sigma_q_scale))\n",
    "plt.xlabel(\"Number of Concurrent Tasks $L$\")\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_number_of_concurrent_tasks.pdf', transparent = False)\n",
    "plt.show()\n"
   ]
  }
 ],
 "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
}
