{
 "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 25 µs, sys: 0 ns, total: 25 µs\n",
      "Wall time: 27.2 µ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: 43.8 +/- 1.7 with number of tasks: 1\n",
      "Regrets of HierBayesUCB: 63.4 +/- 1.9 with number of tasks: 2\n",
      "Regrets of HierBayesUCB: 83.7 +/- 2.0 with number of tasks: 3\n",
      "Regrets of HierBayesUCB: 103.8 +/- 2.2 with number of tasks: 4\n",
      "Regrets of HierBayesUCB: 119.7 +/- 2.2 with number of tasks: 5\n",
      "Regrets of HierBayesUCB: 127.8 +/- 2.4 with number of tasks: 6\n",
      "Regrets of HierBayesUCB: 141.4 +/- 2.5 with number of tasks: 7\n",
      "Regrets of HierBayesUCB: 158.8 +/- 2.7 with number of tasks: 8\n",
      "Regrets of HierBayesUCB: 165.5 +/- 3.0 with number of tasks: 9\n",
      "Regrets of HierBayesUCB: 169.4 +/- 2.8 with number of tasks: 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",
    "len_num_tasks_scales = 11\n",
    "num_tasks_scales = np.arange(0,len_num_tasks_scales)\n",
    "num_tasks_per_round = 1\n",
    "\n",
    "n = 400  # total number of iterations\n",
    "# sube = (400 // 10) * np.arange(1, 11) - 1\n",
    "sube = np.arange(0,len_num_tasks_scales) # num_tasks = 11\n",
    "# sube = np.arange(0, 11)*2 # num_tasks = 21\n",
    "\n",
    "all_regret = np.zeros((len_num_tasks_scales, n, num_runs))\n",
    "\n",
    "# n = 200 * num_tasks // num_tasks_per_round # total number of iterations\n",
    "dim = 4\n",
    "\n",
    "# meta-prior parameters\n",
    "sigma_q_scale = 1\n",
    "# prior parameters\n",
    "sigma_0 = 1\n",
    "# reward noise\n",
    "sigma = 1\n",
    "\n",
    "\n",
    "flag = 0\n",
    "for d in [dim]:\n",
    "  K = 5 * d\n",
    "  plt.figure(figsize=(4, 2.8))\n",
    "  for num_tasks in num_tasks_scales:\n",
    "    # meta-prior parameters\n",
    "    if num_tasks == 0:\n",
    "      continue\n",
    "    else:\n",
    "      mu_q = np.zeros(d)\n",
    "      # prior parameters\n",
    "      Sigma_q = np.square(sigma_q_scale) * np.eye(d)\n",
    "      Sigma_0 = np.square(sigma_0) * np.eye(d)\n",
    "      delta = 1./(num_tasks * 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",
    "      all_regret[num_tasks, :, :] = cum_regret\n",
    "\n",
    "      print(\"Regrets of %s: %.1f +/- %.1f with number of tasks: %d\" % (alg_spec[0],\n",
    "        cum_regret[-1, :].mean(),\n",
    "        cum_regret[-1, :].std() / np.sqrt(cum_regret.shape[1]), num_tasks))\n",
    "      flag +=1\n",
    "\n",
    "  # plt.plot(step, cum_regret.mean(axis=1),\n",
    "  #     dashes=linestyle2dashes(alg_spec[2]), color=color_list[flag],\n",
    "  #     label=r'$m$=%d'%(num_tasks))\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",
    "  num_legend = n//100\n",
    "  \n",
    "  for i in range(num_legend):\n",
    "    num_idx = (i+1) * 100-1\n",
    "    plt.plot(num_tasks_scales, all_regret[:, num_idx, :].mean(axis=1),\n",
    "        dashes=linestyle2dashes(alg_spec[2]), color=color_list[i],\n",
    "        label=r'$n$=%d'%(num_idx+1))\n",
    "    plt.errorbar(num_tasks_scales[sube], all_regret[sube, num_idx, :].mean(axis=1),\n",
    "      all_regret[sube, num_idx, :].std(axis=1) / np.sqrt(all_regret[:, num_idx, :].shape[1]),\n",
    "      fmt=\"none\", ecolor=color_list[i], capsize=1)\n",
    "  \n",
    "  plt.title(r\"Linear Bandit ($d$ = %d, $\\sigma_{q}$ = %.1f, $L$=%d)\" % (d, sigma_q_scale, num_tasks_per_round))\n",
    "  plt.xlabel(\"Number of Tasks $m$\")\n",
    "  plt.xticks(np.arange(len(num_tasks_scales), step=2))\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_num_tasks.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
}
