{
 "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 19 µs, sys: 2 µs, total: 21 µs\n",
      "Wall time: 25.5 µ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: 517.8 +/- 9.3 with variance_0: 0.50\n",
      "Regrets of HierTS: 480.5 +/- 10.0 with variance_0: 0.40\n",
      "Regrets of HierTS: 436.9 +/- 10.5 with variance_0: 0.30\n",
      "Regrets of HierTS: 383.5 +/- 12.2 with variance_0: 0.20\n",
      "Regrets of HierTS: 259.0 +/- 8.4 with variance_0: 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 = (\"HierTS\", \"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",
    "dim = 4\n",
    "# sigma_q_scales = [0.5, 1, 5, 10]\n",
    "sigma_q_scale = 1\n",
    "# sigma_0_scales = [1.5, 1.4, 1.3, 1.2, 1.1]\n",
    "sigma_0_scales = [0.5, 0.4, 0.3, 0.2, 0.1]\n",
    "sigma = 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_0 in sigma_0_scales:\n",
    "    # meta-prior parameters\n",
    "    mu_q = np.zeros(d)\n",
    "    # prior parameters\n",
    "    # reward noise\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",
    "    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_{0}$=%.1f'%(sigma_0))\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_{0}$=%.1f'%(sigma_0))\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_0: %.2f\" % (alg_spec[0],\n",
    "      cum_regret[-1, :].mean(),\n",
    "      cum_regret[-1, :].std() / np.sqrt(cum_regret.shape[1]), sigma_0))\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_of_different_variances_sigma_0.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
}
