{
 "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 17 µs, sys: 2 µs, total: 19 µs\n",
      "Wall time: 21.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"
   ]
  },
  {
   "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 HierTS: 41.4 +/- 2.2 with number of tasks: 1\n",
      "Regrets of HierTS: 64.7 +/- 2.3 with number of tasks: 2\n",
      "Regrets of HierTS: 86.9 +/- 2.8 with number of tasks: 3\n",
      "Regrets of HierTS: 108.3 +/- 2.9 with number of tasks: 4\n",
      "Regrets of HierTS: 136.1 +/- 3.5 with number of tasks: 5\n",
      "Regrets of HierTS: 152.4 +/- 3.3 with number of tasks: 6\n",
      "Regrets of HierTS: 182.8 +/- 3.6 with number of tasks: 7\n",
      "Regrets of HierTS: 205.7 +/- 3.7 with number of tasks: 8\n",
      "Regrets of HierTS: 212.4 +/- 3.7 with number of tasks: 9\n",
      "Regrets of HierTS: 230.9 +/- 4.4 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 = (\"HierTS\", \"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",
    "\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 = HierLinTS(num_tasks, K, d, alg_params)\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_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
}
