{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 26,
   "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": 27,
   "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": 28,
   "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": 29,
   "metadata": {
    "id": "tF2d1ViMEAzW"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 25 µs, sys: 0 ns, total: 25 µs\n",
      "Wall time: 30 µ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\n",
    "\n",
    "class HierBayesUCB(object):\n",
    "  def __init__(self, num_tasks, K, d, params, delta):\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.delta = delta\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.bar_mu = np.copy(self.mu_q)\n",
    "      self.bar_Sigma = np.copy(self.Sigma_q)\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",
    "    self.bar_Sigma = np.linalg.pinv(Lambda_h)\n",
    "    # print(\"mu_h shape\", mu_h.shape)\n",
    "    self.bar_mu = self.bar_Sigma.dot(mu_h.reshape(-1,1))\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",
    "      M = np.linalg.pinv(np.linalg.inv(self.Sigma0) + Gram)\n",
    "      A = self.bar_mu\n",
    "      # print(\" M shape\", M.shape) #(4,4)\n",
    "      # print(\"self bar_mu shape\", A.shape)# (4,)\n",
    "      # print(\"B shape\", B.shape) #(4,)\n",
    "      hat_mu_s = M.dot(np.linalg.pinv(self.Sigma0).dot(self.bar_mu.reshape(-1,1)) +B.reshape(-1,1))\n",
    "      hat_Sigma_s = M + M.dot(np.linalg.pinv(self.Sigma0)).dot(self.bar_Sigma).dot(np.linalg.pinv(self.Sigma0)).dot(M)\n",
    "      # print(s,x,'yes')\n",
    "      # print(\"x shape\", x.shape) # (20, 4)\n",
    "      # print(\"hat_mu_s shape\", hat_mu_s.shape) # (4,4)\n",
    "      # print(\"hat_Sigma_s shape\", hat_Sigma_s.shape) # (4,4)\n",
    "      diag_x_Sigma_x = np.diagonal(x.dot(hat_Sigma_s).dot(x.T))\n",
    "      UCB_s = x.dot(hat_mu_s) + np.sqrt(2 * np.log(1./self.delta)) * np.sqrt(diag_x_Sigma_x.reshape(-1,1))\n",
    "      # print('yes_second')\n",
    "      arms.append(np.argmax(UCB_s))\n",
    "    return arms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "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 HierBayesUCB: 181.2 +/- 5.3 with variance_q: 5.00\n",
      "Regrets of HierBayesUCB: 145.4 +/- 5.0 with variance_q: 4.00\n",
      "Regrets of HierBayesUCB: 124.7 +/- 4.2 with variance_q: 3.00\n",
      "Regrets of HierBayesUCB: 103.1 +/- 3.6 with variance_q: 2.00\n",
      "Regrets of HierBayesUCB: 103.4 +/- 4.6 with variance_q: 1.00\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXsAAAEDCAYAAADUT6SnAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/NK7nSAAAACXBIWXMAAA9hAAAPYQGoP6dpAABXAElEQVR4nO29f3xU1Z3//7x3fmSSzEwSSAgkGX4qEQL1F1EJbrdWlGB/UHHX0NJPBQvy2e4WdivU3U8L7aK7j1XwW2FbXYkWatfWaKWrbSW0ULutBBREhQQEBCWTnyQkmTvJZH7e+/3jZkZCEjIJM5Nf5/l45DGZuXfuOffMzOu+7+u8zzmSpmkaAoFAIBjVyENdAYFAIBDEHyH2AoFAMAYQYi8QCARjACH2AoFAMAYQYi8QCARjACH2AoFAMAYQYi8QCARjACH2AoFAMAYQYi8QCARjACH2gjFDVVUViqIMdTUEo4iKioqhrkLUCLGPI1u2bKGwsPCK+yxdupRNmzYlqEbRUVpaSn5+PoWFheTn55Ofn8/ChQspLS2Ne9lVVVXk5+dHnseqfSoqKnj22Wex2+197hP+vEbSD1igU1VVxaZNm8jPz2ft2rWUlpYm5MLudDqH3e+3TzRB3HjiiSe0efPmXXGfPXv2aAcOHEhQjaJjx44dPepdWVmp3Xvvvdq9994b17IrKyu1mTNnRp7Hon2qq6u1efPmaS6Xq999Z86cGdV+sWTFihXdznk0sWfPHm3FihW9btuxY4d25513avPmzdM2btx41WWFvzux/Py+/e1vazNnzuz2d/n53HvvvdpLL70UszLjhYjsh5ji4mKKioqGuhr9UlBQwO7du6mqqkpo5Ht5+1RUVLBw4cIBHaO0tJT777//ilF9+NgOh6Pf/WJJeXk5LpcrYeUlivBd0tatW3E6nT22l5WVUVZWxs6dO9m9ezeVlZVs2bLlqsqM1+e3atUqTp06FfnbuXNnt+3r169PyF3v1SLEXhA1I9HvVhSFsrIy7rnnnn73PXDgQMIvvFu3buWhhx5KaJmJYMOGDRw+fJjVq1f3un3r1q1s3rwZh8OBw+Hg0Ucf5bnnnruqMisqKuLy+aWnp19xe7jM8vLymJcdS4TYDzErV67sFtGE/cZNmzZRWFjIwoULe0TSl24rKyuLvF5eXs7SpUsjHvulX761a9dGoqnejtkfFRUVLF26lFWrVnX7QV2pzGjOR1EUVq5cSX5+PkuXLu1Rr0vbZ+3ataxcuRKn0xnpS+jvAlRZWYndbqegoKDHNkVR2LJlC+Xl5ZSXl3Pw4MGEin1paSkOh6PXukVLWVkZW7Zswel0UlFRQXl5OVu2bIl8L8Lntnbt2mFzsXY6nSiK0q2tw21wNXeNlZWVQ3aXPHv27GHf1yPEfpjhdrvZunUrxcXF7N+/n9mzZ3frAFq7di1Op5P9+/ezc+dOtm7dSlVVFQAul4tHH32UU6dOsXnzZtatWxfZ5na7KSsro7S0lPXr1/f7o1AUJSKo+fn5rFy5kpKSEjZs2NBtvyuVGc35rFu3DpfLxb59+9i1axfHjx/vs07bt29n27ZtOByOyC11f7fsBw4cwOFw9Hp+K1asYM2aNRQXF+NwOKiqqkqYWCiKEoluB0u4vunp6axbtw6Hw0FxcTEbNmxg69atlJWVUVxcHDm/PXv2xPAMBk9vtg6Aw+Hoc1s0x7z8AhIrwtZhYWFhn52xCxYsGPZibxzqCgh6UlBQEPnSLlu2jJUrVwL6F3rv3r0cPnwYu92O3W5n/fr1vPHGGxQUFFBSUhI5RlFREQ6Hg4qKikjUFL5IRONp2u12Dh8+DOjCVFlZydatW6moqOjmWfZXZn/nU1FRwb59+yKCvGbNGvbu3TvwRusDp9NJXl5ej9e3bt3K4sWLI23hcrmi8nujzby4/PO4nO9///uUlJRclcC5XC4KCgpwOp3Mnz8/0oaKoqAoCosXL47s63Q6+7UjIHbnl2ii8esHe26VlZXs2rULh8PBihUrWLt2Ldu3b+/2nrS0tGHf9yLEfhgyZ86cyP9paWmR/8MR85133tnn/mVlZRw4cICampoeIjJ//vxBdV7Z7XaKiorYvXs3CxcuZMuWLd0i/CuV2d/52O32XiPvWOF2u3s9fllZGfv27etWl2iiwquJxMNUVFRw8OBB9u/ff1XHCde3oqKCbdu2RV4P2xmXftYnTpxgzZo1/R4zFud3NVz6/RgIBw4cYPbs2b1uczqdOByOQZ3bhg0bSEtLi7Tl+vXrWblyJYqidGtfu90+bGyyvhBiPwyx2Wx9bgtnxfTG0qVLSUtLY/Xq1RQVFbF06dJu22MhqosWLWLv3r0Rse+vTLjy+cSb3soOXzQvbY+KioqERaoHDhxAUZQeYzDy8/NZtGhRj6jxSiiKgtPp7HYndbnwKYoSuQsYDlx6B3KpYDqdzkFn0hw8eJD169f3eD1s7wyWy38z4eeXt/lIQIj9CKKgoCAyCvTyH4XT6aSqqopTp07FtQ4nTpyICMnVlulwOCJiFa/oPj09nba2tm6v9SZ84ei4v4yOWNgcGzZs6HZnVFVVxdKlSwfVjr11Sl4ufHv27IlYOok4v/4I2y0VFRUUFxcDn16AB+O5X8mvf+mllyJtPZhzu/y72VugEN4vkSm7g0GI/QjC4XBQUlLCunXrImlr5eXlOJ3OyJezrKyMkpISysvLqaqq6ubbXg3hDsWwxw6f3nIPtsyCggIKCgpYt24du3btwuVysXHjxiu+J+xxh/sRwql7V9q/srKy22tz5szpFu2VlZVF+kD6888TZXOEL6RhMeyL3uyLyy2pAwcOcM8990QV4Sbq/NavX8/WrVspKCggLS2NjRs3smrVqsj2aM8f+rYDL81Ug4Gfm9PpZOnSpTz66KMUFxfjdDrZunUrJSUlPYRdUZRuduVwRIh9nAlntVxOuJN1oGzevJktW7ZE7JI5c+awfv167HY7q1atYtOmTZHOx8t924HWO2wzhO8k5s+f360zNRZl7tq1i3Xr1lFYWEhRURElJSVXHKASvkDceeedzJkzp98fcFFREVu3bu32mt1uZ/Xq1ZHUx6KiIhYvXkx5efmw+cFWVFREspiuhNvt7hZhO51OFi1a1G2fZcuWRQbDRSOesaCsrKxbJJ2fn4/D4YgECiUlJZG0W9DtwUvvdqI5//AYirCoh783TqeTyspKqqqquvXLDBSHw8G2bdvYunVrJNuppKSk17EDFRUVffYZDBckTdO0oa6EQBBPCgsL2bZt24gYqRxGURTWrVvXY7TmWGGknX9+fj67d+8e1j6+yLMXjHoeeuihETGc/VISGYUPR0bS+ZeXl1NUVDSshR5EZC8YIyxcuJBt27YN+x+kYOQxUr5bQuwFY4LwiNldu3YN+6wJwchh7dq13HPPPSPiLkSIvUAgEIwBhGcvEAgEYwAh9gKBQDAGEGIvEAgEY4AxOaiqqck9oP2zsmwDfs9YQbRN74h26RvRNn0z2LbJyup//ikR2QsEAsEYQIi9QCAQjAGE2AsEAsEYQIi9QCAQjAGE2AsEAsEYYExm4wgEAsGVaJQlGmWJbFUjW03MJAONsoQTMHeVG2tEZC8QCIY9jbLEMaNMoywlpLxdFhMLM1LZZTElpLxwmTd3PcYDIfYCgWDYk2jxXe4NdHscDWUKsRcIRgGJjnwTXV6ixTdsoyTKwklEmULsBYIY0yhLHO16TBSJjnwTXd5QiO9oQ3TQCgQxZpfFxJPAwxYTj3j8CSlzuTfAk6lJCYt8E11ePPABASAgQbsk4ZEk/F3PPZJ+of6j2UCGqmHTwKppmDWQgHGaRqsk4ZbBi4RfgoyuC1FAAh8SF2WJgASSpkfVMvp7/RJ40LfVyRIyoEgS3jjHBkLsBaOeRGdWDIUQJjryTVR5PsArQWuX+FYaZT4xyFyUJBoMEhYNzBpUG3Rx9SERlHRRlTVdeO2ahkuSaJN0Ufaji3SzLCNpGkbAqkFKl5ib0TB2nda2FDMdkkR7159PAg3olCQsmka6qpEEmDSNi7KMAQ2TBiZgnKqRrGmoEqhIqOjvTdI0LIBRg4mqCoBNA3uc21KIvSDhYpjo8nZZTDyZmsTDHb6ERNqjxXLoRBfNpK7H/ggBbZJEEAhK0ClBhyRxUZJokiVaZAmfJGEG2lLMBCRdAMN/4QjbpOnvfddkwGmQMWoaqV1NucqezMygynhNY5Kq4pIkvDJMDmkUBkJY0EUedGGVAbcE6RqkaxpJmi7GqWjkhnSh7o0AkJtl47W2Ti43qrSutkmOsl2iJQD8h7WvGl09QuwFCRfDRJc3Ei0Hld471ELoUe7FLvEMolsAri7V+XGymVQ0vF0yFJDApEGWqtIiSzTJeqfqh0YZzyVS1SGBV5IwoiGji663K5o2dIltqqaRoumRck5XRPqAPRm3DPWyHLEkjIChy/ZI0WC8pjJe1Rivalg0SAM0SY9wZcDQda4GFWyaSqjrGP/Q6Sc/qGLsao/cLBuHWjp6iG+ikYCUIa7DYBBiLxj1fu9AI20V8KD7tu0yXJQkOiQJDWiVJVq67IAgEqmaRqcEjbJMi6xbBWHv9R9tFlok3Zsdr2p4JQggkdb1HhX9dl8Dzhpk2rqsgk4JgpLExJDK9JBKsgaHTQY0dFE2QUQ8TUByl6gC1BskAkiY0AXUpEFAhvdNRsapGhNUjVsCKsu9AdIuaY9k9GOEgCC6rZCuaUiApytC7+h6bJMkPjHI/C7JxFJfgAxNY2JII09VsWv9R7tZKWaaOgZ2kVcHtLegN4TYDzOGYuTeaPB7NT7NfqmX9ahU0iAFjZyQLhXvGWXeNRm4KEsRT1cCQhK0SBLvmAxUG2Q6uyLaFE3DqmqMv8RGyFA1MlUNE7rQKpJMiqaL6NSAGunE+4vZyGcCISZ0Ra9uScKs6e9pkyRSNA0DurBqwEMhvRyrpvu8Rg3qDDLnDBKtksTj7V6MgE3TsPYiqGHb4d/afTGPfG2aXu6nLQ3zAyHW2yws8QWHPNKOB+HvUqMskZfAEbTxLFOI/TAj0RZHLNDQxcYj6aKW1iV4lq7tfVkS4W1NksTbJgMnjTINskSnJBFC94rrZQmXJEUEM1nTSOnqQPNLEgF0G+CcQcbfpYCTQrpoBrsi0U+6fkTfsiczPxBiUkhFk3SRV7vePzWk8Tc+H9cEVVI0jeQr1Lk/wvcrD3oDVyWEM0MqM0NXcYBRRKLF98WulNIXE5hRFe8yhdgPMxJtcbQDJ4y6rP3CYsKERie6bSGjUS/LtMq6NRHkU4F1SxLVBokOwNe1So5R01PUFAlCksQ4VcXbldJmVzU0PhVYgNxMK3RlNcwLhCgIqlwXUknRwNBVj+KuyDqE7hV7kPBIegeguSvrwQ9MC6lMVrVeBdoP5GXZODgM/F7B4Ei0+K7wBljsDya0k32FN8DXUpMwx+m3L8R+mBFLiyMInDTIyECbrNsBZw0yHxsknAaZ8wa9U21Kl83xVpcvnIJuYQSRmKBqXBtQSdI0zBBJK0vVNCaHVKaNt+JudkeEV0IX8yBQI0t6xx5aJDqXNX379ZlWPrzYjqXrDiCeo/sSN7Rp6Eh05Jvo8hItvom0US8tMwtoilO5QuxHIMGuRy9QY5BxGiScskydQSKInqXxkUHmjEHG1tXJlq5qZGgaU0Iqs0Iqd/tDTA6pXBNSMaD7vU+7vQOOfDMB7bLvpgyYgemXfGmtl+wUjltsXReI0cZQ+L2JjnwTXd5QiO9oQ4j9MKO969Ep6/nIGarGKaOeqXHeILM3ychfTAYCkoSsaUxSNRwhlTxVI1dVsWgac4IqS3wBrgnq1kZ/ke3ISUgcHGPB70105DsUNofg6hBiPwQEgPMGiUZZ7kpjkzhuNHDcKHPOoBsaxRmpaICry2bJVlUmhTQW+4I8o3SS0jUEeyRGxqNdfOPtvfZGoiNfEWmPPITYxxAN6ADaZYmjRgPnDFKXvw3nDLq18pFBH+49TtWj8jRNj8xvCYT4Zqefa4MqM7NsVF1sx4R+YYi3oI8F8U203xtP71UgGAxC7AeJht4Bedxo4LdJRo4bZRplGZekdzjeEAgxravj0y1LzAiqfMEX5Noun9zehw5cHgsmInIfC+IrolDBWEeI/SB41yjzQ2sSVQYD14ZUvuQLsLLTT7aqkavqg2VGEkJ8BYLRjxD7AfK/JgOr7cls8Ph4tbMTc4yPPxSZHEJ8BYLRj1i8ZIAcMhko8QZY3RmIudBDd0tFIBAIYoWI7AdIlVHmHl+w/x0HiUhpEwgE8UBE9gPkhNFAQSh+c/BlqxqfCapC7AUCQUwRYj8AvEC1QebaoJhwVSAQjCyE2A8AV9fUtJb+dxUIBIJhhRD7AdAug03YKwKBYAQixH4AKJKE/fJZvwQCgWAEIMR+ALglCZvQeoFAMAIRYj8AFEm6ZHk2gUAgGDkMaZ69oijs2bOH8vJydu7c2W1bRUUFL730EgsWLMDhcHDgwAHmzp1LcXFxZJ/S0lLsdnvkWKtXr45rfd0SwsYRCAQjkiET+6qqKiorK1EUBZfL1WO7oigcPHiQvXv34nA4WL16dQ+hBygpKQH0i8OmTZvYvHlz3OrsliXRQSsQCEYkQyb2BQUFFBQUUF5e3uc++/fvj0Tul7Njxw72798feV5UVMTKlSvjKvaK8OwFAsEIZUR69k6nE0VRer0QVFRUxK1ct/DsBQJBnGhslDh6VH+MB8Na7MN+fllZGVu2bIm87nQ6e93fbrejKErc6iM8e4FgbNDYKHHsmBw34e2NXbtM3Hyz/hgPhu1EaAUFBQA4HA4AysrKWLt2Ldu3b+/zPWlpab36/5eTlWUbcH2ysmz4gRwgyyrG0F7KYNpzLCDapW+Ge9v853/Co4/Cxo0QR2e4G2vXwpNPwtq1SWRlJcX8+MNW7MMiH2bx4sVs2rTpipF7NEIP0NTkHlBdsrJsNDW5aUpLhs4ATf74zXo50gi3jaA7ol36ZjBt09go0dgokZ2tkZ0d/7vre++VePRRK/fe205TU2Lu5o1GABtGo5umpoG9N5qL57C1cS7vuA37806ns8eFIIyiKH1uiwVeIBlh4wgEiWbXLhMLF6bGzeK4nPAFJREXlkQxLCN7RVFYt24d+/bti4h3OKJ3OBzY7Xbsdnuvwl9UVBS3evklCfPo+ewFgkGT6Eh7+fIATz6ZxPLll6/SnDgCAf28L16U8HgkPB5ISYG2NonqagmvV6KzE3w+Cb8fOjpg/HgNi0WP2g0GMBq1yP+yDH4/XLggUVMj09oa3/6BIRf73qwXu93OqlWrugl5WVkZixYtikT4Dz30EBUVFZE8+/Ly8sj/8cIHxN5JEwhGHrt2mXjyySQeftjHI4/Ef5H6WEXaigLvvmugpUVClnURbmuTSErSSE4Gi0WjulqmuloX3gcftNDQIFNXJ9HcLGG36wKemqqRkqLR0SFhs2lMn65G3m+xaNjtMHmyRnOzRHu7RCgEwSAEg+H/9cekJI3MTI2iohATJqj86U/xk+QhE3un00l5eTl79uyhqqqKLVu2dBshu2bNmsjAKYC2trZunbOrV6+mtLQ0YvccP348rjn2AAEJTCIbR9APjY0STieYzVLCbIDRGmmHQtDUJEWyYjQNamslzpyRqa+XsFj06FhRJNxuCa8XzGZdxFtb9Sj84kVdqC9elGhqkpg9W2XCBA1VBa8XMjM1fD7wePTIPC9PIzdXX7Ni8eIgkydr5OSoTJyokZISv3MNxPmmRdK0sadeg+2gvWVcKs+7Opkbx5WqRhqiI7Injz9uTmjUOxRlBgKQm2ujttaNaYA2evg7o6rQ1ganTxtoapJoaJCordVFvLZWoq5OpqFBwmyGceM0ampkZFkX3JkzVXJyVHw+CbNZIy1Nw2bTI2ufTyIYhPR0jfHj9cg5/JiXp2K1xvf8BsvVtml/DLmNM5LwI2yckchojXqHukwAlwssXZnIViv4fHDqlIzHo0ff770n09go094O7e26peHxgNttxeORMBo1rr1WJTtbY9IkjUmTVG6/XRfynBw9orbbdQskN9fG8ePtZGaClLj094QRvntpbJTIy4v991SI/QDwCxsnJiRafEeqvxzvMn0+XYA7OqCuTqamRuKjj2SamiSSk3U/2evVo2xFkWhrk3C5JC5c0P1ugDlzrKiqhMGgMWGCRlubRFaWFvGxr79e5bOfDWC1alitkJqqMXlyKj5fB1ar7pMPRLjT00en0AO8+KIp8hiP76kQ+wHgRxKRfQxItPjGK+r1eqGlRSI9Pb5e7qWoKlRXS1RWGsjI0H1nTdP/AMrLjXz8sczZszI+HzidMoEATJyodyCGBbutTbdNQiFdgCdN0sjL05g6VWXGDDWSWTJunMbcuSHS0zXS0iAtTSMrS/e4b7rJitPZjtEIHg/U1MhkZOiifyWyskhY7vpgiXeU3RsrVgT42teSMJvjc3cmxH4A+CVGZerlaLc5Bhr1aho4nRJHjxr44AMDra16BGwyQWcnNDbKXLyot5nJBG43mEyfptOFhfe221IxmWDSJJWMDC1iYxiN+g65uRqKItHSImG3azQ06Cl4ACaTnqIX7mw0mfT31NTImEwwZ04Ij0fCYNDLCnaN89uxw0R+vsrs2SGSkuBv/zaA2Qz19XpqYHq6Fvm7ms/70s5ESYLUVMjPj19fVqLFN95Rdm9kZ2txvRAKsR8AfsA8CgdVjXabQ+3SIJ9Pj4rb2yUCgU895E8+kWhqkvnkEz3i/fBDA6EQXH99iJtvDjF7tobVqnf8paTokWtmph4NZ2ZqBAK64Kvqpyl2N95o5b/+qxNZ1rNJWlokbDY9ig4GIRTSy50wQe88dLkkJk7UcDhUJEkX02AQAgG9roGAfhFxOPQyL7cywp17u3d3JqxDMZEkWnxXrAiweHFQDKoaiwQBdZQOqhrKASuaptshPp8eHTc26lkYsgzJybqoffSRTF2dzNGjcmQwS2enPnCls1NCkqC5Wc/akCTdUrFaNXJyNNrbobJSD39nzLCSmalnbphMRPbLzdXT6hYvDpGRoTFzpsrUqT0FtS9MJhg3DugKBMJR7w03qDEQ3uH5hUt0pJ1o8U3UXW4iEWIfJb6uR3Ocy0m0pQLxibRbWuDYMQO1tTJer54bfeiQkawslWBQF4rPfCaVlhYJTdOtjWBQIitLJTdX96I7O/UI+JprVLKyVO6+O4jdDikpuqCbzbpoB4N6rrSq6paCxyPhdkNtrUxKikZ+vkphoZXTp9tJS4vZKY5pEh1pj0bxTTRC7KMkIIFR0+I+mVCiLZWBEgxCXZ3E+fMy587J1NbCyZPJVFbK2O0a48bpowpPn5bJz1eZPFnFYtHFeO1aH21tus/8hz8Y+Z//8ZCbq2dkGAz6sY0x+UbqolBYqPs34Ug7UZ2oQ9G5N9ojbcHVI8Q+SnxIcY/q4eosFU3TsyKOHjWgKHpnXEaG7jcfPWpg3z4jBoMeGV9KOHvit781kpKidwzKMrjdumUiSfDhhwbeestAVZWMzQZTpqhMmaJy441wzz0B/vmfVTo69OyU1FS47jqVrKzehSAQgL//e5g+Xetmc8RG6IeeoejcE5G2oD9Gyc8r/gQkSErAdzsaS6WuTuKddwwcOWKgtVWKZIi8/75MICAxe3aI8eP1TI/WVj3VrqAgxF13BTEa9YmawqiqnnkC8ItfmLo6D/W5O1JT9XzpUAhmzFD5l3/xcdNNIdLTP61LVpaJpqbhPeXzUES98Uyh66tMEWkLroQQ+yhJ9ICqUAgqKgy0temZHL/9rZGGBj17RFEkbr45RGFhiGnTVJKTNTIyoLAwFOl8HAiBAPz852Z+8QuRyREL4p1C11eZQugFV0KIfZT44jigSlWhqkqmqkrPOgFYtCiFQECfOS81VaOkJMDMmSpJSfq8IOH86pGI8JcFgp40djTgDJ7B7LORnTox5scXYh8lgRgMqNI0OHNG5sgRmepqGbNZ99ePHNH/v+mmUMQ/X7/ez+LFwVHjY1+K8JcFw53GjgYaPQ1kp0yMi/D2xq7K53ny3cd5+OZHeOTW78X8+KNQSuKDj+gGVLW1wfPPm/ngA5mLF/Uo3WjUSEqCykr9uW6/aDQ16Z2b//qvIaZP1yKDaXbtMlNcnDihF5G2QNCdeAtvbyyf9Q2efPdxls/6RlyOL8Q+Si5fpSocpe/fb+DgQQNNTfp8JF4v3H13kC99KRgZ6RgI6Lnfs2apXHutOuwmchKRtkDQnXgLb2+E7yDidSchxD5K9KkSdH7yExPPPGMmFIK//usQ99wTZMIEjWuuUUlL00bcwB0RaQuGO4m2VWIlvB2BDk61nOR06ymaOpuoaj7OeeUTAFKMKQTUAAbJgNlgRpbiO4pHiH2U6JOgaRw6ZODpp/XMlblz1chUr7FiKAbkiEhbMFASLb6JslUaPY2cbT1DQ0c9AL8+8wrvNLxDTXs1ncFODJIBWTJg6BLmoBrEr/oJqSEkSUKWZLxBL7XtNUhItPpamGybQv64WUxIyebm7Hksu245qqbiD/kxyUZCWghfyI8syeyv/kPczk2IfZT4uwZV/f73Bu6/P8j118dnhr+hGJAjEAyURHvaV2OraJqG4ndR215LfXstdR111LXXUt9eR5uvDYsxiVZvKy3eFs60nmaKfSoTuy5gL59+idtzP8tf5X2WFGMKIU0lpIVQtRAAJtmM2WBCQkZDQ9NUzIYkcm15oGlMTM0hxRTd0O1AKL7jMoTYR4lfgiRN4+BBI9/5jq//NwwSYakIBkOiI+1Ee9rhczrr+oifnfgpFzubSTVbOdf2ES3eFswGM+Ms4wmE/IS0EH41QJu3lTZfKxc7m/Grfial5pBjzSUnNYdJ1lxmjy/AnpSGP+Qnw5JBWlI612fdQFpSOoFQgNxnx/PLL7yKyTA6Bp8IsY8SP2DwwwcfyNxySyhu5QhLRTAYEh1px7IzMaSGMMj6wBFN07jgaeTt+oME1AAuv4vjTR/wXuNRAP7lz+u5a2oxM8fl4/a7+dKMrzApNQd/yMdF70Xd+0bGbEgiw5JBhmUcGZZxZCVnxd0Tv1oaOxoij3l2R8yPL8Q+SnySBK36dLYjrQNWMPq52kjbH/LTHnAjIXGq9RSBkB+jbCTP5iDNnIZRNqGh0eK9SLOniUaPLkzPHfsvlIBCVfNxmjubcdgcyJIBVQsR1EKMs4zHYZuM3WznoreZSak52GostLjcuP1u/tf5R96q/TPpSel0BjvpDHZilI3clrOAVGMKKaZU5mZez1euuY+//c0S/nfZoYRE2vEW3t548eQLkcdhk2ff3t6OtZcl2mtqagDIy8u7uloNQwISSG0adruIugVXJt4jIS/HF/KhovcheUOdVCvnOdN6ivZAO1aTlWRjCqmmVAJqgI5AB2daT1HtrkbVQrT723G2OznW9D7eYCchLcRk2xTsZjt+NUB9ey1tvja0rjEmGUkZjE/OZJxlPAAnW06QZknnnulfYkJKNjVuJ6qmYpSNGCQDjZ4G6tprOOlzYU+yU9l8nBRLEmpAIsWUyhdnLOHHdz6LN+QlxZhKstGCxZiMUe4uTfH2sy8n3sLbGyvmfJOv3Xw/Zp8tLscflNgXFhZy8uTJHq87nU6ee+45nn/++auu2HDDD6guSYi9oF9iYal4g15avS3YzDYkSeac6yzNniaaOi/Q5GmiubOJj13nqGw+RrX7PFaTLhBf+nUxIS3INekzsSfZcfvdeAIefCEvJtlEsjGZGenXMsU+NRK5L5yyiILMOUxLm95nfVRNRdO0iN0S9rSf/Nz2AUfaWVk2mprcA3pPoiPtFXO+yeLpXyA7JTGjZ0G3xLKyrh1w20TLoMRe62NCsDlz5lBZWXlVFRquBCQJtQ3s9qGuiWCgDOfOS1VTOXnxBB+1neb9C+9xqP4A1Uo1zZ1NWM022v36D3+yfQoTUrLJTM4iK3kCmSmZ3D21mIcLH2FmRj4yMrnPjufYA6fiYnPIkgxDOBgw0ZF2dmripklIFAMS+7vuugtJkpAkibvvvrvHdqfTyezZs2NWueFEAAgpiMh+BBLPzktVU/lE+Zhq5TyegIeQFiKo6lM+B7Ug51xn8Yf8+IJemjubeL/pva4skTbebTzMR21nmJCSTcH4OcwaX8D6ef/MtLTp5FjzsBgtqJpKQA2QZLjyNHyJtjnGQqQ92hiQ2G/evBlN03jwwQd5+OGHe2x3OByjVuyDCLEfqfQWaauaSpuvlfSkDDqDnbh8bRxtfLcrH7uGNm8rKirNnmbcAYVWbwsNHQ2kmFIYZxmPWTbjV/2cbTuDhMTUtGmkmmzIkkRI1bO1Fr7yWSxGC0mGJMwGM+lJ6dww4SYyk7Nw2CazfPYDzB43m1STNWKPXI4syf0KPSRefEWkPfIYkNjPnz8fgEWLFrFo0aK4VGi4EpAg2CZht8dnMNVYIl62iqqp1LidBLUgDe31NHjqueBppKFLCL/zv9+mubOZJs8FmjubkCQJTdMIaSGSjcncnF1IhmUcOdZcMrtS9fIzZjGuK30vO2UinmAHLd4WgmoAo2xism0KM9KvQbpkwqOwn31y5bmE5WgnWnxFpD3yGJRnv23bNk6ePMnvfvc7Tp48GemQff755ykqKmLWrFkxreRwIIBE0CU8+1hwNbaK26/gC/nJTM6koaOe3517nc6gl+NN7/NW7V/whrwkGZKYlJrDxNSJZKdMZFzyOAC+et3XmZSaw4SUCWSlTMBqstHqayE9KWPY52D3R6LFV0TaI49Bif3LL7/Mk08+ycMPP8wrr7wSeT0vL4+tW7eOymycgAQBF9gzRp+NM9w6MBWfi4P1FTR2NGCUjTR2NPCr02XIkky1+zyappGZnEWL9yJ3TSnGZrZROPFW/vHmDVw3bla3KBv0SHv70R/xxelLekTa4RTCWDIUOdpCfAX9MSixf+6553j11VfJy8vjySefjLy+aNEiNm3aFLPKDScCgN8lYZ8y+sR+KEdfNnc2c7rlQ061fki1cp5jTe/zTsMh5mZez2T7FEJqCJvZxmO3P47VbGVa2gxSTal87DrHxNSJcRHrq2UocrQFgv4YlNi3tbWRfumq0104nc4+0zJHOkEk/KO0g3agoy8DoQAftp6k1duC1Byg3e3HarJS215DUA2SZEhCkiTGWzJx+xVMBjMSEv6QD7ffzcmWEwDM3XUtnqCHazJmMjMjn6n2qSy99m/ZtfhFbOYr+2WzxxdEfX5DkTkSz8ExgtFJR2M7QWc7PjOkZvcctHq1DErsi4uLWbduHdu2bYu81t7ezg9+8APuv//+mFVuOBGQwO8Gmy2+Yp9IS0XTNFp9Lbj8LgAO1R+g9ZNWmroG73QE2kkyWLCZbbR6WyMZI39y/hGTbCI7JZtM23g8Xi+KX8Fhm4xJNkZSEFu8F0lPyiCoBgmoASxGC3ZzGlPsUwH4zdK9TE+7ps9MlFgxFJkj8RwcIxidVP34AEeerWLemgJueTT2CTCDEvvNmzezdu1a5s2bB8B9993HiRMnuP/++1m/fn1MKzhcCALBTgmLJb7lxNpS8Qa9tPlaCam6+P7u499w3vUJnygfc7r1FL6gl8zkLABePfMKE1KyyUqewHXjZmE12fAEO3D5XMzMuI6QFkLTVL455yFumHATkiQNajRkIBTgP9/7EVPt0+Mu9CAyRwQjgxvVIxwhmRvVI8AwEXuA7du343Q6OXFCvyWfPXs2DkdiOqOGAr8EQS9xF/uBWCrtgXYaO+pp9bbidFdzpOEdLnguYDKYaOio52PXOeo76vQ8bkkm1WTlzsl3UzjpVv5m5v1cN242OdZcgmqQ3GfH88Lil0bNdK6XIjovBSMB46qvQ+mr+mM8jj+YNx08eJD58+fjcDiuSuAVRWHPnj2Ul5ezc+fOHttLS0uxd+U6KorC6tWrB7Q9lgSRCHZCcnJ8bZzepo71hXwE1SDVynnebTxMRd1bHGt6H6e7Grs5jQxLBhNTJzEv+xZmjS/AF/IyKTWXaWnTmZo2jWRjclzrPFCGIltFIBgInZUf03mqhuT8PJLnTIt7eaFAiFZFX/hUzc4mHve7gxL7TZs28d3vfpe77rpr0AVXVVVRWVmJoii4XK4e20tLSwEoKSkBoKKigk2bNrF58+aotseagAQBLyT1P5hxwCg+F2869/Nm9X6aO5sAWLn367R6WzjV8iFuv0KyMYW0pDSuz7qBotzbWTlnFdPSZpCZnHnV5Y/20ZcCwUA5+f9e5tAhE7fdVsFNrz8Sm4M2NNL6fg3Nh6sxXGxiQlIbTUyg5mAd587KmFQvEL+BPIMS+1WrVrFlyxbmz5/f61TH0VBQUEBBQQHl5eW9bt+xYwf79++PPC8qKmLlypURMe9ve6wJAgGvhMUSm8i+3e/mSONhfnHyBco/foMbJtzEwimLKMpZwO/Pl/Ol6V/GnpTO3MzPYDPb6Ah0kGPNjUnZlyNGXwqGO3JjA3JjA2r2RNTs+H9vZm5cwqEvvMHMjUuie0N7O4bq80jeTvzNbtqPn8d3rgHXuVbqa1R87iC1nnFIMuSkugilpPJnl5Xs1HM4ZppZ8i/TSf9sAU8vHGZr0EqShNVq5c4774zYOZfS27w5A8HpdKIoSsSiuZSKigocDscVtxcVFV1V+b0RQI/skwfpiATVIL//pJz91b/n7fqDnGk9zazxBdwz7Yu8940nGZ+s54sHQgH+4Y//l6XX3t/NP09LSr/6k+gDMfpSMNyx7Hqe1Ccfp+PhR/A8EuOAJBjE8PE5pJYWZKUNqaUFc3UNYCT95NtYfvFj/K0ekEC7qOBNTuPC2Q46vAZSpE5a/FYURUazWDjrzcMTspBqCZJik0jLzmHiHRkk5Yzjpi/cTMaciT0G/YUJebzAMBP7AwcOkJ6eTnp6OoqiUFVVFdnW14kMBKfT2evrdrsdRVH63d4fWVkDz3/WzEb8nZCXZ6WXIQZ9UqPU8NzR5yg9WkpmSibLCpbxzcIVFDmKeu0MrW6rBiBoaScnffKA6zkYsrJszOHaq3q/oCeiXfpmoG3jLllB/ZMvYC1ZQVamFTQNTp+GmhqoqoKjR+HQIVAUMBr1JeUufZw0CaZOBa8XAgHUaTOodap0HDuL+cNjTE5rw5idCenpMG4cvvETgYlU/GcVjcyg1qmhamA0SpjMEpPy7dgmpeJpC5Cel0L2nDx8qpHlS/LJmp2FwTRw173toK5rlrom0ufHfkLJQc+NMxSkpaXhcrl6jegv3d4fA00VzMqy4fGEUEMG2tvdBPqYTVbTNJo6m6io/QtvOvfzJ+cfafO18uUZ9/LcXS9wc3Zh5GLY1uIFvD2Osf3tp/XHt54eEX72YFIvxwKjvV2uxlbpq22ktlYklwu5uQnDR2cwfHwW4+nTaBYLB0/ncJA1LPjyZj7v/g2S34dmtxPKm0xo6jSChbcQvLeE0KQcpFAIzeens9lDslXG1+rF0taAdO4TzrvN1FZD5at+ki0ytknX0zH5Vlw1HeCWMCQZkA0y/g4/EERb+iWumzWBxYumI5v1bf3R0uYZUHuEOfqdXwAm3v7OLwbcTxDNxXNQYn/ffffx6quv9rn9lVdeobKyksWLF3PbbbcNpohe6U/IoxH6weLzgSRpmM09t7UH2rngaWT9n9ZxpOEdrp9wI3c47uQXX/gV12bMxGzo5U19IPxswUhgQLaKpiG1tCApLuTGRvj4Q5LbOkACye3GePwYxspjyI0NaDYb6rjxhK6ZSWjadPwL7wafjznZLg4eg5n/8kVa7/ghmsmMlpEB8qfi2/JhM2fKTuH6pI0LRxvoaGxHNsiE/CHUgAqSzLjr0sldkMeSx+YyfnZW5L3tdW5k46f7GpoaeOGLb3DLsslI06bEqxm7Mevf7+czdc0Ecq4+6aI3BiX2YaukpqaG9PT0bp2069at49ChQyxatIh169bx2GOPDThrp690TkVRrpjuGd4eDwKdYLbApS5VRe1bvHrmZV48+QIWg4WHPvMtfvnFV6Oaf7wvhJ8tGAyJ7sD0Lv8GqU8+jnf5NyAYRHZWY/jkYyS/H0P1JxjOnUWTZWSXC9M7h5DratHsaaiZmVA4D4MqgaqipabiW/wFOjb8C6H86+g1mgJkjxd+8jTmJXehpuiDXfxuH+117Zx+9SS1f3HS9lELM/9mFrlFDuauvIGJt+Tg+qQN++Q0NE1DNsp92szWnO6RcdIvfwIkY/nlf+P7f4m5w06eM42sOz4zvJYlXLRoEXfffTfV1bq/XFBQwM9+9jOsVit79+7l17/+NbNmzaKoqIj/+q//GpTY2+12nE5nD/EOd772tz3WBLwSSV0DqlRN5bFDP+SXJ3/O8lkP8M7yD8izOUb8NLmCkUvMOzDDc1xJkh6Zt7WCLCNXV2Ooq6Wj3k09k7B8pYRxdR+ipaURmjoNLcmCmucgNH0GaBrBqdPwfv0BAoW3gkH3sbOybLRHIWjelk46Wzqp3v8JF946C8CLt+3EnJ5CR307/nY/lnHJTFk4jZv/8RZyFjgwW7tfLNKnZwzq9C8uXgZPvcbFxcuI/Sw1Q8OgB1WtWrUqMg9OaWkp69at44c//CGSJEXmsy8uLuYHP/jBFY/Vl/Xy0EMPUVFREcmjLy8vj/wfzfZY4++EpK60y80HN3Gg9i+8WVLBxNRJcStTIIiWbpF2H8h1tST9qgxj5THUCdloNjuoIYxVlcj19aBpGM99hGYw6gKvqgSvuRZDrRNJUUBVUXMdhPLyOOycwQHWcEthNoX/8QW0tPRey9Q0jdYzLVTveJ9gZwA0mDgzk85QiIsnmvC7fPhcPgwWA2jg+rgNLaThaeqgo6EdS0YyWZ+ZwHT/Wc5g5t6FrbQvW4I1x0rqRCuyMT4BVtXvGyKPt944Iy5lJJpBib3T6ew24dnq1at5+eWXe+zndvd99XY6nZSXl7Nnzx6qqqrYsmULc+fOpbi4OHLM0tLSSB7+8ePHu+XQ97c91gQ6IdUC59o+4ucndvHWsneE0At6RW5sAOcZZLMt/paKpiEpLqQO/bemyQYMH53RPXCnE7npAnLTBQzV5zGcqMJ/zxcJ3P7XyI0NSG43yDK+L9+LmucAVSV0zbWR46JpGD46QyjPgTplqh7ld3nk0z65yIFbfsbMR4ojQt/R0E7DkXoaDtfhd/voqGun7WwrfrePyZ+fRlJ6EkgSZ944g/tCB5mfmYAtz8742UkEvUECngDT77kG2WQgOTOZ9BnjMKXqGWtaTR1/vOklrN9ZgS0vJ75tCsxZcT3TF19DSnZq3MtKFIMS+/nz5/ODH/yAZcuWAfDss89SUFDAyZMnAX0GTKvVysGDByko6H0qWofDwerVq684xcGl28IXgYFsjyVBnx7Z7z7zK75yzVImWeP/hROMTCy7nocnH8dyNZaKpumC3NGO8Z23MdR0CXd9HXJrK0gS0oVGDI0N4POhZeircY2/5TNo9jSCc+YSmjINNSuLwLTp+O69j8BtRWj2tAFVQ83pPpBPUzXqDtbgP1cDgKGpiQttAQ5s+l+ajl1gwg3ZTCzMIX3GOPL+ajI2h50JN07slsUymEwlNTs78hj/qfP0KYbjMc3wUDIosX/sscfYsmULDzzwAKCLv9vtjoyqDQ+2OnjwIE899VQs6ztk6DNeauyv/gPfvvGfhro6ggGQqM5Lqd2NZrV9aqnc8yXkc2eRXW3g8yP5vMgN9ciNDZiOHNb3zxinR9GqqlsqR99FGzcO+UIj+PyQkkLg+hsITZuOmpNLYN4tqJmZSKqKOiGb0MQctMxMOuoUkm66Fd9bB0idMr57JsFVoKm6DdNwpI7GI/XUHawBDVJ9FwEDLy75HaaMVK5fcxNffuU+DEmDnlvxinQ0dkQe7XlibdDBMKhPxmazsXnz5j5tk/LyciorK9m5cyezZ8d+cMBQEPJCUoqX9y8c5bN5fz3U1REMgJh1XnZ2IgUDyDU1SMEAhnNnMR45jPnNfcgXGpHb2gg5JkeEdtziz6OmZ6DZ7WiWZEgyo2ZPQp04Ef+dd6FmT0RqbdGj9FAINI2O7/0QSXGhjR9PaNqMqEW76pmDHGEN80oPc8u/LR7QafkUH+7zLpqrmvC2dBLyhzDbkvjk92e5cLQB2Wxg4rxJTJyXQ37JbCbOy0FqaOCZm15i9TtfRUqArXLyxeORx1sfWRD38kYjg74Mnzx5kjfeeIMTJ070WHC8uLg47rZKogl2gmyrx262YzWLkZEjif46LyVXG1J7O1pKiu4/axrGw++QtP/3yDVOPRpvqMdw/hMIhVBz89BMJtScXILzbqH9iR/pvnb2RIynP4RAgIziz9P8cb0+gjMB3MhRfS50jgLdxV7TtG4phyF/iPpDtZzf/zGNR+ppfLcey/hkJs6bRHJWKrJBov5QLdcuncVfP7EQ22R7j5TFUIJtldHooSeaq15w/NKO2dG64LgGqD4J2dpIVsqEoa7OiOdqbRW5sYGkV1/BvG8vmtWG/3OfJ3jDjUgdHcjNTaBpaCYTkt8Pfj9SMAhAyn/+CLmhATVrAhhk5LpaTG8fRPL50JKSkDp0q0BLsqBlZuJbdA+BW+ejTpyEOmkSoWnT0ZJTrijgwbnXI9c4I/VU8xIzhbPx71ZC6UvUzL6LC/9xAEmSqDtYQ8AToOXDZpLSLGiqRsqEVALtfmSzgWmLZzB39Y18+Vf3YbAYBzTVSaJtldHooScaseB4FAQAOoHURrKShdhfLX3ZKlJzM4ZaJ2paOurESUjBgD45VdMFjO8fJen35UgXL2L86Ay+uxbhfeBB8HhI+n05yc/+BCwW3UYBpGAQzWxGMyehX65BtdsJFsztSiMMEbjxZjp++BjquPER71xytSG53bpID9L3trz4QuTxamwjNajS0dhBoMNPwO2n9UwLHY0d2HJtTLl7OoEOPw2H9eyXhsO1ALz1/x1jxpdnooU0Zn99LpbxyYyfnYnf7UdTNTrq2wn5Qky5a9pVzWMlbJWRh1hwPArCYq8lXyArJau/3QV+P3JtDeqEbOQLjUheLySZkdxuJEUhNFH3eKXWFmz/sEbPMKlxYqivI5STi+zSZx5EltFSrWhWK4Fbb8N779+gZk8keONNaOmfDpbxfbWflX0CASyv/w+e737vyraKJKGlZ3Q79mC4+IWvodx4N4GcTKKdJFXTNOoP1XLh/UY66two5100VV7A7/JhspkxW82kTUvHmmOj9s/V7Pv7PRiTTWTfPJGJhTnc/E+3sef/vEbJn/4PZlvPEdypuuvCuJnjr+rcwghbZeQhFhyPggAgd2oEk0ZnZB+1raJpyA31mN45hOHcWeSaGshMJyUpFbn6vJ7LXX0eubYGLS0NuaUFNT0dzWZH8npR7Xa9szK1q8/Dkkzg2pmoi7+ImpND8LrZn84h7fXqOd1hcb6KKFRubIg8JsJWOfa7Ot598m1ufvhWbr3CKkcdje2c++0Zzr3xEU0fXMCSbmHS/NxI5H7DPxQysXBSrxF4wBOITNoF+kpHAAZLfLJhLkfYKiMPseB4FPgB2Qd+c8Oo9Ox7s1WMh9/G/NafkTweCIWQ6+swfngSw8dnCdxcSDD/On1IvOpDqr9A6LpZ+O9eTGjyFNQpU/TRmX6/LtaXi1UgQFbueDq+94O+I+0YLvYbK1slWmYtn8u7T77NrOVze2y7eLKZugonZ39zhgvvN+D43FRmfW0On38qF2ueLWprxZTSvd1EaqKgP8SC41EQQBd7n/ECWclzhro6MSecreJbeDdJL71I0u9ex3T0XXxf/DKaPQ3NZCRwy234vnIf/jvv0ucH78KaZaOjrwEyfU1qleBI27vim/gXfyGuOfZqUKXlZDMXTzbTXq+3xzv/foCktCS8rZ2APhWA26mQU+Sg4Buf4Z7//kqPuVwGi/DQBf0Rtdi3t7ezZ88eAP72b/8WoMcMlCdPnhyV2TgRsU9uZlxybDzPIaejA8nvw3TkHUx/+TMA6X+zhODcz+BbshT3U0+jjY/PuSY60o71YCpN02g8Uk/NX6oxpZq5WNXE2d+cJindQubcCSRnpgCQef0EAm4/467LRJJhxpdmMuWu6RjMsU9WFB66oD+iEnu3283SpUsjk5aFs3HCUxvX1NSwdetWysvLWbBg9EUVAUDyQSi1k1TTyPoxSW4F49F3sfzy54AERiPG4x9gOHNan5XwhhsJ3DIfgJajlZFh9/EkEZH2pXQ0tuNp7CAlOzUqnzkU0PPQL7zXQNAToO1cG6nZqXRc6MBd7UI579JTFxfNIOBpw5ZrY9mfH8DmsEfef/K/jzP3wRsGtWLRYBAeuqA/ohL7Z599ltmzZ0c6ZNeuXcuOHTt46KGHeOKJJ3j55ZcpKipi9+7do2bE7KWEI/ug7CHZOMhFaBOI1NiI5ZWXSPrdaxiPvos6ZSqd33gQzWZD8nbiXf4NggVz0FJSwWhErnGS8sx/InV0JETsEzXnepjKXR982mF6icXhbe2kvcaNpmko5100vFNHbUUNzZUXSJ+eQe5fTcaUamJi4SQ6L3aSW+TAtqwA+5Q07FPS+ly1SPjnguFIVGL/+9//np/+9KeR5xs2bOCuu+6itLR0VIt8GD+AD4J0kmxMiWtZV8yM8fsxvX0Q44lK5Lo6JFcbpvfe1QcRmZP0uVcaG5D8fvyfvQPPP/yT7rH309mZaFsl0YQ7TNNnjOPEz4/RXuvG+b/VtHzYjGVcMpqmkT49gwk3TmT+xr9i4rxJmK7CSxf+uWA4EpXYO51O8vLyIs/DPv2rr746qkU+TACQvBpByUNKnCP7cGaMZ83f47/zLgwfn0Nua9VHjf7mNdTx4wnecJN+McjJxf21b4AsIQUCaLIBNScHdeKkAQ3TT7StEg80VcPn8tJ5sRPlvAvvxU6aq5pwOxWS0vS88w/+613SpqRhGZ/CTWsLybohG+uk2E99MWfF9dz8tevxxabvVSCICVGJvc3W8wchSdKYEHroGlTlkwjEIbKXG+oxfvA++LwYGurB7wPA8sJPMR0+RGjqdNTMTNSsCbh+/hLBm+bFtHxIvK0yUA+9N3yKD8+FDlo+vEjnhQ5OvXKCpmMXSEqzkD4jA7PdTObcCeTd7kCp1vua7tvz1YR46KnZ1lG/4Lhg5BGV2F/NsOrRQLiD1q95sBivIv/b78d4ohLje0dJ+t1vkNwuDJ98TDB/FqSkEJo4CQz6R9J64EjC5lVJNH156JdP2OW50MEnfzjHJ+VnuXiiGZPVRNATJOgN4nN5saRbsDnSsE22c+3S67j3t8t69dFDgRDvP/1uQs5NIBiuRCX2LpeLW2+9tdtrmqb1eC3M22+/ffU1G0YEAM0fIEQg6sherqvF9M4hJI8H2VlN0v+8iqH6vD4n+Y034fvKUkKOyfqc5LM+vUOSa5wk/3xnnM5keBD20LPmTODw1oO0nmmh8Ug97bVukrNSMKWYkIwybqeL3NsnM33xDG55ZAHBzgDGZBPGZCMpE1J6nRagN0SHqUAQpdiPxlGxA8EPEOhEQiLJ0I/AaBopTz5O8jM/JjivEC0llVBeHh3/+m8Ebp3f51qdYUZbZ6m7RqG91o0114Y110ZncyfHnjkKQMW//pnJd0wlt8jB9f/3ZtKnp+Np8hDsDBLsCJB1QzbGGAz/HysdpsXFd3R7Pm/eLTz22ONDVJuxy+HDb7Nly7+jKArXXTeLRx99vFcrPNFE9UtatWpVvOsxrAkAashDkpxyRUvLeOggqU9tQW5ooPUP/4s6feALFY/0zlJN1Qh6AniaPbz/9LucefUk1jw77bVuzHYz3oudTF6ozxfz1QMrenjoSWmxmyYhzFgacPTKK68PC2EZq7jdbl588We8/PJrADz99HbWrfs7fvrT/x7iml3FdAljiQCghTwkyX1k4qgqyc/8mJQf/wjPt7+D9xsr0KyD+8ElurP0agn5Q5zfd46zr5+h7mAN7hoFLaRhSjVx7dLr+OqBFaROtKKpGrVvOcmcm4XJaubZ32zr/+AxQgw4EiSKurpali9/IPL8W99ay+23z6O2tobc3LwrvDP+CLGPgrDYWwzd/Xrp4kWSX/gp5vLfQTBE22vlhGbmD00lE0jnxU5O/+okF95rwPnHT0jNtTHjyzOZ880bSJuahtmWhCR3vwOSZIm8z+pzzSs1ChBbDz0EqDE5Uv/IMKjVmV57bTdvvrmPDz88CUBOTi7z5t3Ct761NmZ127jxEerqasnJyR0a+yAU0tfTvRKBgP4XC2QZDAP7NOL5OeTnX9fteW2tvjD7UAs9CLGPigCgqR4shk8je/O+vdjW/T3+2/+KztV/h+8r93WbIGy0EfAEqNz5ASdeOEZ7vZspC6cz6bZcFv3HQtT0gSWUx8NDX5CRyjlj7yNaY830oMqh1o4Bvefpp7eTlpbGU089jdvt5pvf/Hq/t/Zut5uf99NZn5aWFokkc3JyWbJkKXfcsZDXXts9JPZBxoJ5GM+d7Xe/WK0KEZw+g9ZD70W9fyI+h0t58cWf8bWv9b4cZqIZveoUQ/yAqnkimTjm8jew/eO3cP/4WfwLFw1t5eKM3+3jw7ITHH3qHcYXZPK5H91F+oyMiC0yfhD55PHw0A+0diQ0sh8ItbU1HDnyTkRUoo22bTbbgKLNS0Xr85+/iy1b/h23253Q6L71wJF+I/uYjkGQo/80EvU5hHnxxZ9htQ7uvfFAiH0UBOgSe1My0oUL2P7xWyg7dhH47OeGumox4fJBTiFfkNOvfsiF9xs58+pJJtw0ibuf+wI5t8XmVjQeHrqBwVkrieDIkXe48867Is/dbjfWKPp0riaiVBTXwCsaCwyG/m0VkylhC7FfSiI/hyee+Dfy82f1+tkMFULsoyAAaFonKcYULK+8hP+znxs1Qg+fDnKa/Y25ZMwcT+Xz75OSnYrjr6fwlddLyCwQSzFeDXa7nfb2TyPZZ57ZziOPfL/bPqdOfciRI29jtdo4deok3/3u9wYUUR4+/DaHD78d2f+113Yzb94tIjPnEhLxOYAu9IWFt3LHHQtjVvdYIMQ+CgKAiodUczLmP/4B7/Lh4cFdDUFvkIbDddT8pZq6A3on0tnfnsHxWT+3ff92pn/x2jE/cjpW3HHHQp5+ejtvvrkPRVFYsuS+Hh15zzyznaeeepra2hoOHx74oMTCwls5ffpDHnzw67S3u5k58zoefVTk2F9KIj6Hw4ff5vXXf83rr/+62+vPP//fPcpKNELso8AbBIydpElmTG8fRCndNdRVGjAhX5AL7zfSeLRBF/gKJ/Yp6eR9djJzV91Aw+E6Vhxfk7D518caV4oMDx9+m5kzdSE4ffpDZs0a3JxTy5c/MKxsg+FIvD+HwsJbeeutI4OuXzwRYh8FnT7A5GGyounL9I0bvqtVfVx+lroDTowpJjRVo/FoPZ7GDlpPt5A6MRXH56ZyzZdn8vkf3R3pII1HKqRgYIRT8w4ffpslS+4b4tqMXUbz55CYXLURjtcHkqWD3BY/ocmTh7o6vaJpGqd/dZI/rtuLyWpGDaoEOwPM/vpcPrf1Lh46/22+8f5DfH77Iq5bVtAtE+bSVEhB4iksvJX2djdvvrmPI0feGfLb/bHKaP8cRGQfBbrYe5h40Ucob/iIvbe1k6PbD2MwG2g82oDycRv3vLCESbfmDug4Y2k6geFK2H75+c93DW1Fxjij+XMQYh8FXh9ISR4mXPSgOoZ2Dn9PYwcnf1lJ8/EL1FbUkPdXk7Hl2ZjxxWuZ+TezMKUOPKVNTCcwPHjxxZ8BJDw3XtCd0fo5CLGPAq9XF/vM5g5C1yd+jvmAJ8CF9xo4s/tDTr96kqmLZjD5zmnMe/g2xs8WaZGjBdHBOjwYrZ+DEPso8PlATvKQ3tyOGuc5Li4d4JRkT+JP6/fx0WunSZuWzuTPTeHrb39T2C0CgWDACLGPAp8PMHtIafeiZoyLa1nhAU5zHryelpPNmGxJrDj2EJZx8V37ViAQjG6GtdhXVFTw0ksvsWDBAhwOBwcOHGDu3LkUFxdH9iktLcVu19MFFUVh9erVMa+HzweSyUOyuxN/enrMjx+mo6Gd5uNNAJz8ZRVzV17Prd+7XeS+CwSCq2ZYi72iKBw8eJC9e/ficDhYvXp1D6EHKCkpAfSLw6ZNm9i8eXNM6+H3AaZOkto9+OIk9vVv17L3m7/lmiX6FMkPnvo7TJbEzx8iEAhGJ8Na7AH2798fidwvZ8eOHezfvz/yvKioiJUrV8ZF7I1SO6aOTtR+lhWMFuW8i0P//hbGJCPttW4ufNDI555cyISbJ3Gs9CidzZ2Y8oTYCwSC2DBiB1U5nU4URen1QlBRURHTsvw+SA91EEoyg+Xqls1TgypVLxzjlbtfxJZnZ3xBFvnLZvP1ww9yzZJ8McBJIBDEhWEf2e/Zs4e0tDRcLhfV1dVs2LAB0MW+N+x2O4qiXPGYWVkDy531+yA90IGanjbg94bRNI1Tr51i3z/vw5Jm4Wu/+SqTF/QcoPXXDy/g5q9dj3WSFdsgy0o0g22T0c5QtUtZWRmbNm1i3759OByJTxWOhtH+namqqmLr1q1s27atT2cCdB3btGkTTqeT2bNns3379ri1zbAW+4KCAoDIF7asrIy1a9eyffv2Pt8TvjBciYEunBDw2cjwewjY0mkbxKILbqfCsR1HOffGRxT98LORGSV7rYcRjA4rXsAbqwUe4khMF6IYRQxVu7jdbv77v3+B1Wrj4sV2LJbh99mM9u/ME0/8G4qiUFFRQXNzOz5f37PHfuMbD7Bhw/+jsPBWXnttNytXruSJJ/rWt76I5gIxrG0ch8PRLTJZvHgxe/fuvWLk3p/QD4aAF9J9nWjpGQN6n7fNy58e/gMvf/7nuGvdfOV/7mfGl2aKqYMFceOZZ7azZMnSK0aTgvjy3e9+j8ce63966VOnPgT0OXkAlixZSmVlJW53fC6EwzqyLy8v75Z9E/4CO53OPm9PFUWJ+a1rwKuR7vPCAMT+7G/P8Nb33iT3dgdL93yVjGvim58/1olmnetYMYg1roH4LzgeXnbvu9/9XmTIf6JRQyqaql1xn1AgRCgQikl5kiwhGwYWsyZi4fdoqKurISen+zxWDoeDurrauEzCNmzFXlEU1q1b1813DEf0DocDu92O3W7vVfiLiopiWpeAL0C6V0PK6H9q46A3yN5v/oYL7zVw508W4/jcFBHJJ4AFC1I5dy5BC45PVzl0aPgtOL5ly7/zd3/37QHVK9b8csEuXOfaElZe2vR0lh96MOr9E73g+JVQFKXXZRHjtaTksBV7u93OqlWrugl5WVkZixYtikT4Dz30EBUVFZE8+/Ly8sj/sSTg85DmlWBS+hX30zSNP33nD2gafP3IKkwpInUyURw40JHQyH4gJGKh6zff3Acw5EvhffXAin4j+1h69pIcfSCV6AXH++PyZRJBdy3s9rSYlwXDWOwB1qxZExk4BdDW1tatc3b16tWUlpZSXl4OwPHjx2OeYw8Q9Huw+01ofXw52uvdnHjhOBdPNuM618rS3y0TQp9golnneqhIxELXJ0+eoK6ulvvvXwJAXV0t//RPf8/y5Q+wZMnSqzuBASAb5H5XfjeYDEMyKnwoFn6/Ejk5edTV1XZ7TVGUHtZOrBjWYm+32/ud/uDS7Zf6+zHF7yHdZ0Cz9pwGuPqPH/P71b9j+hevZfx14/nc1oWYbUnxqYdgRJKIha6/9a213fa9/fZ5/OhHP4msvCRI3ILjVyLc+Wqz2SK+/Jtv7uOOOxby2mu7KSoqitu0ysM6G2e4cGfAQ5pfRkv9VOw1TePjPR/xhzVvsOj5L/H5bYu45Z8XkJyZMoQ1FQxH7rhjIS6Xizff3Mdrr+3uc6Hr5csfYN68W/odJyIYHIn6HJ5+ejsPPvh1AL75za/z/e8/Etn2+OOP8frruyPPf/Sjn/Daa7u5//4lHD78Ntu2bRtUmdEwrCP74UIg4MHuk7tF9sdK3+P9p49EOmEFgiuRiAXHL2W4Lno91CTic7hSGZenZObm5vHUU09Hntvt8RuDICL7KOgMdmLzg9bl7zUcruPI1oN86eX7mHr39CGunWA0cOlC1/Pm3TrEtRm7jObPQYh9FHQGPdj9RCL7ih/+mfkb/4pxM/tPxRQI+mO0L3Q9Uhjtn4MQ+yjwBj1Y/Sqa1crFE020nW0h//6hXYtWMLpYvvwB7rhjYVTZIYL4MZo/ByH2UeBVPVh9KprVRtULx8gvKcCQJLo7BLHl0oWuBUPHaP0chGJFgU/1kOIP4k1K5vSvPuRv9n5tqKskGIWM1oWuRxqj9XMQkX0U+NUOUn1BLtYHSUpLIn3GwCZEEwgEgqFGiH0UyEEFg6rRdLaDzDlZQ10dgUAgGDBC7KMgKdCG12yk+UQzmXMnDHV1BAKBYMAIsY+CpKALnyWJiyebySwQkb1AIBh5CLGPgs/e6CZoteD6uI20aelDXR2BQCAYMCIbJwrGm914k614Pu7ANjk+048KBFfLqVMf8swz26mrqyUnJ5cNG/6fmAhNEEFE9lEgu9tpM2aSnJUipi4WDEvcbjePP/4Yjz76OC+//BqFhbeyceM/D3W1BMMIIfZRILd30CZnkjY1fairIhD0is1mY9u2ZyLT4+bk5PZYGEMwthE2ThQYOzpxaxnYJotFnAWDJ95rn4aFvra2hp//fNeoHBgkGDxC7KPA1OGlQ7VhzRl982WMFkJqCFVLzLqEsiRjkAe20lKi1j594ol/4/XXf828ebckdIUqwfBHiH0UmDxeOkMppGenDnVVBH2w4JfzOOc6m5CypqfN4NDy96LeP5Frn373u9/ju9/9Hq+9tpsHH+z/giIYOwixj4I7MufxTpWNVCH2w5YDXz2S0Mh+IAzF2qdLlixly5Z/5/DhtyksHF3zsgsGhxD7KCiwTOZPnUZSsnuuQSsYHhhkA4b+VroeIhKx9unhw29TV1cbsW5ee203VquN664TU3ELdITYR4PbTXu7JCJ7waC4446FPP30dt58cx+KovS59ulTTz1NbW0Nhw+/PeAyCgtvjaxlCmC1ds/OEQiE2EdByNWO16ORIsReMEgSsfbpkiVLRaesoE9Enn0UtF/0kZQiY7SIa6MgPozmtU8FwwMh9lHgbguSOs481NUQjFJG+9qnguGBEPsoaA8mi85ZQVwZzWufCoYHQuyjwP2P3yd5auZQV0Mwyhmta58KhgfChI6C9gudpE4Ukb0gvozWtU8FwwMR2UeBu94t0i4FAsGIRoh9FLTXt4u0S4FAMKIRYh8FQuwFAsFIR4h9FNz8f28m6zPZQ10NgUAgGDRC7KPg5tU3Y7aKPHuBQDByEWIvEAgEYwAh9gKBQDAGGBV59qWlpdjt+pKBiqKwevXqIa6RQCAQDC9GfGRfWloKQElJCSUlJRQUFLBp06YhrpVAIBAML0a82O/YsYOSkpLI86KiIsrKyoawRgKBQDD8GNFi73Q6URQlYuFcSkVFxRDUSCAQCIYnI9qzdzqdvb5ut9tRFKXP92VlDXxmwcG8Z6wg2qZ3RLv0jWibvolX24zoyL4v0tLScLlcQ10NgUAgGDaMSrEXQi8QCATdGdFi73A4en1dUZQ+twkEAsFYZMSLvd1u79W7LyoqGoIaCQQCwfBkRIs9wEMPPdQt86a8vLxbKqZAIBAIQNI0TRvqSlwtpaWlEdvm+PHjbNiwIWbHHWsjcxVFYc+ePZSXl7Nz584e2/trk9HeZuFBfOG7yc2bN/fYPhbbJ/y9AaiursbpdPLYY491S4seq21zOStXruzx20pI22iCXtmxY4e2Y8eOyPMDBw5oGzduHMIaxZ/KykrtpZde0nbs2KHde++9Pbb31yajvc2eeOKJbs83btyorVixIvJ8LLfPxo0bterq6m7PRdv0ZM+ePdrMmTO7vZaothFi3wfz5s3TXC5Xt9cu/5BGK3v27OlV7Ptrk9HcZi6XS1uxYkW386usrNRmzpwZEbmx3D4rVqzoJkg7duzQ5s2bF3k+ltsmjMvl0l566aUe55Wothnxnn08ECNze9Jfm4yFNqusrOyWDBC2DhVFGfPts3Pnzm7WwvHjx5k/fz4gvjth9uzZw+LFi7u9lsi2GdEjaOPFYEfmjmb6a5PR3mZ2u53Dhw93ey38Y3M4HFRWVvb5vrHQPpdSXl6O2+1m27ZtgPjugP5d6S1DMJFtIyL7ASBG5vakvzYZzW22Y8cONm/e3GvUFWYstY+iKJSVleF0OikuLr5iu8DYa5uBjP2JR9sIsR8Ao+WLF0v6a5PR2mZbtmxh8eLF/ab5jqX2sdvtlJSUROycwsLCK0afY6VtysrKKC4uHtB74tE2Qux7QYzM7Ul/bTKW2qy8vJzJkyd386jHcvsoisKWLVu6CXtRURGKolBRUTGm26aqqoo5c+b0uT2RbSM8+164dGTu5Q06VkfmRtMmY6HNwj59OKJXFAWXyzWm28fpdPLcc8+xbNmybrngoJ/zWG4bl8tFVVVV5HsT9uDDY4PCdlci2kZE9n0wlkfm9nWL2F+bjPY2q6qqoqqqioKCApxOJ06nk7KyMtLS0oCx2z4FBQWsWrWqmxi98cYbFBQURARprLZNUVERq1evjvyFz2n16tURaydRbTMqRtDGi3iNzB2uOJ1OysvL2bNnD1VVVaxatYq5c+d28xv7a5PR2maKonDnnXf26kGfOnUq8v9Ybp9LV4hzOp2sX7++xwjasdg2YcrLy3njjTfYu3cvq1atYsGCBZGLYSLaRoi9QCAQjAGEjSMQCARjACH2AoFAMAYQYi8QCARjACH2AoFAMAYQYi8QCARjACH2AoFAMAYQYi8QCARjACH2AoFAMAYQYi8YM5SWlpKfn09hYSGFhYXk5+ezcOHCHpN4JZqqqiry8/MH9J6+5jkXCPpCiL1gTBFehOTw4cOcOnWKnTt34nQ6Wbp06YhaKGPhwoVDXQXBCEOIvWBM43A42L59Oy6Xiz179gx1daKiqqpqxE/9K0g8QuwFghFGX0vcCQRXQoi9YEzjdDpZu3YtaWlp3aaN3bJlCwsXLqSwsJBNmzZ1e8/KlSvZsmVL5PnlnvvatWspLS1l06ZNFBYWsnDhwm5T1CqKwsqVK8nPz2fp0qVRLxxdUVFBaWkpO3bsAPQ+COHdC6JFiL1gTKEoCvn5+ZG/hQsX4nA42L17d2SftWvXcuLECXbu3Mn+/ftpa2tj5cqVUZfhdrvZunUrxcXF7N+/n9mzZ3e7YKxbtw6Xy8W+ffvYtWsXx48fj+q44bnRFUVh/fr1rF69Wtg5gqgRYi8YU9jtdk6dOsWpU6ciAr9mzZrIvOtVVVXs3buXbdu2RVZY2r59O5WVlVFH4EBk4Q673c6yZcsiEbjT6aSioqLb8desWRP1ccN+fX+LeQsElyPEXjBmCQvy1q1bI69VVlb2KqZz5szhwIEDUR/70nVHwytZgS7W4aX6BkNlZSWzZ88e1HsFYxsh9oIxzfr16ykrK4tE3rFKv7TZbDE5zuUcOHCABQsWxOXYgtGNEHvBmCYc3Yc7XIuKinA6nT1Ev7Kykrlz5/Z6jL7W7O0Nh8OBoiiD7lg9ceJE5K6hvLx8UMcQjE2E2AvGPOvXr2fv3r2RxcSLiopYsWJFRPTXrl2Lw+GIrMXrcDg4ceIEoHvwl9pA/VFQUEBBQQHr1q2LiP7GjRujfr/T6aSgoGBgJygQIMReIOjh3e/cuZP58+ezdOlS7rzzTtLT07tl65SUlFBZWRlJyywpKRmQB79r1y7S0tIG9f7169dTXl5OeXl5t4XgBYL+EAuOCwQCwRhARPYCgUAwBhBiLxAIBGMAIfYCgUAwBhBiLxAIBGMAIfYCgUAwBhBiLxAIBGMAIfYCgUAwBhBiLxAIBGMAIfYCgUAwBhBiLxAIBGMAIfYCgUAwBvj/ARl2LCb4kVytAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 400x280 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# linear bandit\n",
    "alg_spec = (\"HierBayesUCB\", \"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",
    "delta = 1./(num_tasks * n)\n",
    "dim = 4\n",
    "sigma_q_scales = [5, 4, 3, 2, 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",
    "      # HierBayesUCB\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 = HierBayesUCB(num_tasks, K, d, alg_params, delta)\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=\"lower right\", ncol = 2, 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_for_HierBayesUCB_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
}
