{
 "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 20 µs, sys: 0 ns, total: 20 µ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": 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: 1230.1 +/- 67.2 with variance: 10.00\n",
      "Regrets of HierTS: 846.7 +/- 46.5 with variance: 7.00\n",
      "Regrets of HierTS: 524.6 +/- 27.2 with variance: 4.00\n",
      "Regrets of HierTS: 282.0 +/- 9.2 with variance: 1.00\n",
      "Regrets of HierTS: 137.4 +/- 4.7 with variance: 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_scales = [10, 7, 4, 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 in sigma_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$=%d'%(sigma))\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$=%.1f'%(sigma))\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: %.2f\" % (alg_spec[0],\n",
    "      cum_regret[-1, :].mean(),\n",
    "      cum_regret[-1, :].std() / np.sqrt(cum_regret.shape[1]), sigma))\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.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
}
