{
 "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 18 µs, sys: 3 µs, total: 21 µs\n",
      "Wall time: 22.6 µ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"
   ]
  },
  {
   "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: 634.6 +/- 16.0\n",
      "L = 2: 670.7 +/- 17.9\n",
      "L = 3: 652.4 +/- 14.6\n",
      "L = 4: 671.8 +/- 15.2\n",
      "L = 5: 658.2 +/- 14.2\n",
      "L = 6: 628.0 +/- 16.3\n",
      "L = 7: 641.8 +/- 17.9\n",
      "L = 8: 665.4 +/- 18.4\n",
      "L = 9: 660.6 +/- 16.6\n",
      "L = 10: 640.0 +/- 16.4\n",
      "L = 1: 313.3 +/- 10.7\n",
      "L = 2: 317.7 +/- 12.5\n",
      "L = 3: 328.1 +/- 11.4\n",
      "L = 4: 321.7 +/- 11.3\n",
      "L = 5: 322.6 +/- 11.0\n",
      "L = 6: 341.6 +/- 11.2\n",
      "L = 7: 341.0 +/- 11.7\n",
      "L = 8: 313.3 +/- 10.3\n",
      "L = 9: 306.3 +/- 10.7\n",
      "L = 10: 331.7 +/- 12.6\n",
      "L = 1: 93.6 +/- 5.6\n",
      "L = 2: 90.8 +/- 4.7\n",
      "L = 3: 92.0 +/- 5.1\n",
      "L = 4: 93.6 +/- 4.5\n",
      "L = 5: 99.4 +/- 5.2\n",
      "L = 6: 92.2 +/- 5.0\n",
      "L = 7: 93.4 +/- 4.7\n",
      "L = 8: 103.7 +/- 5.0\n",
      "L = 9: 109.8 +/- 5.3\n",
      "L = 10: 100.8 +/- 5.1\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",
    "  (\"HierTS\", \"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",
    "\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",
    "        else:\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",
    "        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 (m = %d, $\\sigma_q$ = %.3f)\" % (num_tasks, sigma_q_scale))\n",
    "# plt.xlabel(\"Number of Concurrent Tasks L\")\n",
    "# plt.ylabel(\"Regret\")\n",
    "# plt.legend(loc=\"upper left\", frameon=False)\n",
    "# plt.yticks(np.arange(50, 101, step=10))\n",
    "# plt.tight_layout()\n",
    "# plt.tick_params(axis='both', which='both', length=0)\n",
    "# plt.savefig('regret_in_concurrent_setting_%d_dimension.pdf'%(d), transparent = False)\n",
    "# plt.show()\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_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
}
