{
 "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 20 µs, sys: 2 µs, total: 22 µs\n",
      "Wall time: 24.1 µ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: 922.2 +/- 35.4 with variance_q: 4.00\n",
      "Regrets of HierTS: 698.9 +/- 29.9 with variance_q: 3.00\n",
      "Regrets of HierTS: 470.7 +/- 21.0 with variance_q: 2.00\n",
      "Regrets of HierTS: 248.8 +/- 8.7 with variance_q: 1.00\n",
      "Regrets of HierTS: 219.6 +/- 8.5 with variance_q: 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 = [4, 3, 2, 1, 0.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_q_scale in sigma_q_scales:\n",
    "    # meta-prior parameters\n",
    "    mu_q = np.zeros(d)\n",
    "    # prior parameters\n",
    "    sigma_0 = 0.1\n",
    "    # reward noise\n",
    "    sigma = 1\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_{q}$=%d'%(sigma_q_scale))\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_{q}$=%.1f'%(sigma_q_scale))\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_q: %.2f\" % (alg_spec[0],\n",
    "      cum_regret[-1, :].mean(),\n",
    "      cum_regret[-1, :].std() / np.sqrt(cum_regret.shape[1]), sigma_q_scale))\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_q.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
}
