{
  "cells": [
    {
      "cell_type": "code",
      "execution_count": 38,
      "metadata": {
        "id": "EA5dBSsGzKmM"
      },
      "outputs": [],
      "source": [
        "import math\n",
        "import torch\n",
        "import torch.nn.functional as torchF\n",
        "import matplotlib.pyplot as plt\n",
        "import tqdm"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 39,
      "metadata": {
        "id": "E0T0pLce0aWx"
      },
      "outputs": [],
      "source": [
        "class BanditInstance:\n",
        "    def __init__(self, T, d, A, eps):\n",
        "        self.T = T      # horizon length \n",
        "        self.d = d      # feature dimension \n",
        "        self.A = A      # number of actions \n",
        "        self.eps = eps  # noise level \n",
        "        self.rewards = torch.zeros(T, A)\n",
        "        self.arms = torch.zeros(T, A, d)\n",
        "\n",
        "    def get_arms(self):\n",
        "        return self.arms[self.t]\n",
        "\n",
        "    def choose(self, a):\n",
        "        max_reward = torch.max(self.rewards[self.t])\n",
        "        obv_reward = self.rewards[self.t, a] + torch.randn(1) * self.eps\n",
        "\n",
        "        self.regret[self.t] = max(0, max_reward - self.rewards[self.t, a]) \n",
        "        self.tot_regret += self.regret[self.t]\n",
        "\n",
        "        # print(\"Round %d, Regret %.2f, Total %.2f\" % (self.t, self.regret[self.t], self.tot_regret))\n",
        "\n",
        "        self.range.set_description(\"Total Regret %.2f\" % self.tot_regret)\n",
        "\n",
        "        self.t += 1\n",
        "\n",
        "        return obv_reward \n",
        "\n",
        "    def horizon(self):\n",
        "        self.regret = torch.zeros(T)\n",
        "        self.tot_regret = 0\n",
        "        self.t = 0\n",
        "        self.range = tqdm.tqdm(range(T))\n",
        "        return self.range\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 40,
      "metadata": {
        "id": "RdNeg0QE0csV"
      },
      "outputs": [],
      "source": [
        "class LinBandit(BanditInstance):\n",
        "    def __init__(self, T, d, A, eps, zeta):\n",
        "        self.zeta = zeta \n",
        "\n",
        "        super().__init__(T, d, A, eps)\n",
        "        theta = torchF.normalize(torch.randn(d), dim=0)\n",
        "        self.arms = torchF.normalize(torch.randn(T, A, d), dim=2)\n",
        "        self.rewards = torch.inner(theta, self.arms) + torch.FloatTensor(T, A).uniform_(-zeta, zeta)\n",
        "\n",
        "T = 3000             # horizon length\n",
        "d = 100              # feature dimensions\n",
        "A = 20               # number of actions\n",
        "eps = 0.2            # noise level\n",
        "zeta = 0.00          # misspecified level \n",
        "\n",
        "LB = LinBandit(T, d, A, eps, zeta)\n",
        "PASS = 5"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 41,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "tqNJHnJS0eUZ",
        "outputId": "881a9d5a-4af0-4e87-fb28-6f31352878d0"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "Total Regret 95.64: 100%|██████████| 3000/3000 [00:12<00:00, 232.02it/s]\n",
            "Total Regret 95.03: 100%|██████████| 3000/3000 [00:12<00:00, 245.69it/s]\n",
            "Total Regret 104.00: 100%|██████████| 3000/3000 [00:12<00:00, 230.81it/s]\n",
            "Total Regret 104.92: 100%|██████████| 3000/3000 [00:13<00:00, 223.86it/s]\n",
            "Total Regret 111.38: 100%|██████████| 3000/3000 [00:15<00:00, 189.75it/s]\n"
          ]
        }
      ],
      "source": [
        "# Linear UCB\n",
        "logUCB = torch.zeros(PASS, T)\n",
        "\n",
        "for N in range(PASS):\n",
        "    chosen = torch.zeros(T, d)\n",
        "    obv_reward = torch.zeros(T)\n",
        "\n",
        "    A_hat = torch.eye(d)\n",
        "    b_hat = torch.zeros(d)\n",
        "    for t in LB.horizon():\n",
        "        bonus = (eps ** 2) * math.sqrt(d) + (zeta ** 2) * math.sqrt(t)\n",
        "\n",
        "        cur_arms = LB.get_arms()\n",
        "        invA_hat = torch.inverse(A_hat)\n",
        "        theta_hat = invA_hat @ b_hat \n",
        "\n",
        "        reward_hat = torch.inner(theta_hat, cur_arms) + bonus * torch.diag(cur_arms @ invA_hat @ cur_arms.T)\n",
        "        a = max(range(A), key = lambda a: reward_hat[a])\n",
        "        chosen[t] = cur_arms[a]\n",
        "        obv_reward[t] = LB.choose(a)\n",
        "\n",
        "        A_hat += torch.outer(chosen[t], chosen[t])\n",
        "        b_hat += chosen[t] * obv_reward[t]\n",
        "\n",
        "    logUCB[N] = LB.regret\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 42,
      "metadata": {
        "id": "W-rJJ3L-gqs4",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "7a33a87c-5fe4-494c-c369-174f8eb15fb3"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "Total Regret 121.52: 100%|██████████| 3000/3000 [00:19<00:00, 155.09it/s]\n",
            "Total Regret 122.28: 100%|██████████| 3000/3000 [00:18<00:00, 166.03it/s]\n",
            "Total Regret 113.32: 100%|██████████| 3000/3000 [00:17<00:00, 174.05it/s]\n",
            "Total Regret 124.02: 100%|██████████| 3000/3000 [00:17<00:00, 170.50it/s]\n",
            "Total Regret 107.67: 100%|██████████| 3000/3000 [00:17<00:00, 171.36it/s]\n"
          ]
        }
      ],
      "source": [
        "# Linear Thompson Sampling \n",
        "logTS = torch.zeros(PASS, T)\n",
        "\n",
        "inv_beta = 0.01\n",
        "\n",
        "for N in range(PASS):\n",
        "    chosen = torch.zeros(T, d)\n",
        "    obv_reward = torch.zeros(T)\n",
        "\n",
        "    A_hat = torch.eye(d)\n",
        "    b_hat = torch.zeros(d)\n",
        "    for t in LB.horizon():\n",
        "        cur_arms = LB.get_arms()\n",
        "        invA_hat = torch.inverse(A_hat)\n",
        "        theta_hat = invA_hat @ b_hat \n",
        "        \n",
        "        # thompson sampling\n",
        "        theta = torch.distributions.multivariate_normal.MultivariateNormal(theta_hat, inv_beta * invA_hat).sample()\n",
        "\n",
        "        reward_hat = torch.inner(theta, cur_arms)\n",
        "        a = max(range(A), key = lambda a: reward_hat[a])\n",
        "        chosen[t] = cur_arms[a]\n",
        "        obv_reward[t] = LB.choose(a)\n",
        "\n",
        "        A_hat += torch.outer(chosen[t], chosen[t])\n",
        "        b_hat += chosen[t] * obv_reward[t]\n",
        "\n",
        "    logTS[N] = LB.regret\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 43,
      "metadata": {
        "id": "j4VBYgmJ2_cc",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "9fff2bf3-65d7-4cc5-f999-3a04571daacc"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "Total Regret 114.10: 100%|██████████| 3000/3000 [12:03<00:00,  4.15it/s]\n",
            "Total Regret 114.11: 100%|██████████| 3000/3000 [12:26<00:00,  4.02it/s]\n",
            "Total Regret 108.46: 100%|██████████| 3000/3000 [12:11<00:00,  4.10it/s]\n",
            "Total Regret 123.11: 100%|██████████| 3000/3000 [12:34<00:00,  3.98it/s]\n",
            "Total Regret 99.22: 100%|██████████| 3000/3000 [12:07<00:00,  4.13it/s]\n"
          ]
        }
      ],
      "source": [
        "# FGTS.LP\n",
        "logFGTS = torch.zeros(PASS, T)\n",
        "\n",
        "inv_beta = 0.01\n",
        "lam = math.sqrt(1 / T + zeta ** 2 / d)\n",
        "\n",
        "Kt = 200\n",
        "Ki = 14\n",
        "\n",
        "for N in range(PASS):\n",
        "    theta_hat = torch.randn(d) / math.sqrt(d)\n",
        "\n",
        "    chosen = torch.zeros(T, d)\n",
        "    obv_arms = torch.zeros(T, A, d)\n",
        "    obv_reward = torch.zeros(T)\n",
        "    opt_arms = torch.zeros(T, d)\n",
        "\n",
        "    for t in LB.horizon():\n",
        "        perm = torch.randperm(t)\n",
        "        obv_arms[t] = LB.get_arms()\n",
        "\n",
        "        for step in range(Kt):\n",
        "            lr = 1 / (d * (step + 1))\n",
        "            theta_hat.requires_grad = True\n",
        "\n",
        "            if step < int(math.sqrt(Kt)) or step % int(math.sqrt(Kt)) == 0:\n",
        "                opt_arms = obv_arms[torch.arange(T), torch.argmax(torch.inner(theta_hat, obv_arms), dim=1)]\n",
        "\n",
        "            L = torch.pow(torch.inner(theta_hat, chosen[:t]) - obv_reward[:t], 2)\n",
        "            F = torch.inner(theta_hat, opt_arms[:t])\n",
        "\n",
        "            loss = torch.sum(L) - lam * torch.sum(F)\n",
        "            loss.backward()\n",
        "\n",
        "            with torch.no_grad():\n",
        "                theta_hat = theta_hat - theta_hat.grad * lr - theta_hat * lr * inv_beta * d + torch.randn(d) * math.sqrt(2 * lr * inv_beta)\n",
        "\n",
        "        a = torch.argmax(torch.inner(theta_hat, obv_arms[t]))\n",
        "        chosen[t] = obv_arms[t, a]\n",
        "        obv_reward[t] = LB.choose(a)\n",
        "\n",
        "    logFGTS[N] = LB.regret\n",
        "\n",
        "torch.save(logFGTS, \"miss-zeta=%.2f.pt\" % zeta)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 44,
      "metadata": {
        "id": "163gmArECJHR",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 279
        },
        "outputId": "bf488c5a-0df9-42a2-ef8b-0d86743b91a3"
      },
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 432x288 with 1 Axes>"
            ],
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYUAAAEGCAYAAACKB4k+AAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAgAElEQVR4nOy9d3Rc133v+9mnTB8UorGTYBHFKpJisTpldctykR3F5Vpyv3bsm9wkfldJ7nqxvdJv/N5dztNKnGLHzk3c4rhFkSVZzY5kkRIpUWLvJAii1+mn7vfHHgAECwiAGLBof9aaNZgzp+wBwfObX/v+hJQSjUaj0WgAjEu9AI1Go9FcPmijoNFoNJphtFHQaDQazTDaKGg0Go1mGG0UNBqNRjOMdakXcDHU19fLhQsXXuplaDQazRXFjh07eqSUDed674o2CgsXLmT79u2XehkajUZzRSGEOHG+93T4SKPRaDTDaKOg0Wg0mmG0UdBoNBrNMFd0TuFceJ5Ha2srpVLpUi/lsicWizF37lxs277US9FoNJcJV51RaG1tJZ1Os3DhQoQQl3o5ly1SSnp7e2ltbaW5uflSL0ej0VwmXHXho1KpRF1dnTYIF0AIQV1dnfaoNBrNKK46owBogzBO9O9Jo9GcyVVpFDQajUYzObRRqACmabJ27drhx/HjxwF45ZVX2LJlC0uXLmX9+vXcf//97Nq1i82bN7N27Vrmz59PQ0PDqOO+8Y1vsHr1atasWcOqVav4yU9+ctb1vvnNb/L5z3/+rO0LFy4cPvbuu++mo6Oj0h9do9FUCCklXuCRcTP0FHsqdp2rLtF8ORCPx9m5c+eobZ2dnTz00EN8+9vf5sYbbwTgxRdf5MiRI2zbtg1QN/ft27fz2GOPASpp/id/8ie89tprVFdXk8vl6O7untBann/+eerr6/mDP/gD/vRP/5S/+qu/moJPqNFoKkkQBjiBQ97LUwpKeKGH4zuEhADErTj18fqKXFsbhWniscce45FHHhk2CAA333zzmMd0dXWRTqdJpVIApFKp4Z8nyq233qoNgkZzGeKHPoEMKPklMk6GrJdFcukmYl7VRuHL/76HvW2ZKT3nitlVfPGBlWPuUywWWbt2LQDNzc386Ec/Ys+ePTzyyCMTutZ1111HU1MTzc3N3HHHHTz44IM88MADk1r3448/zurVqyd1rEajmVr80CfrZukv9VMMipd6OaO4qo3CpeJc4aMz2bx5M5lMhrvvvpuvfvWr59zHNE2efPJJXn31VZ599ll++7d/mx07dvClL31p3Gu5/fbbMU2TNWvW8Md//McT+RgajWYKkFLihR4Fr0DRL6rHZWYITqdiRkEI8Q3gnUCXlHJVedtfAg8ALnAE+JiUcqD83u8DnwAC4DellE9d7Bou9I1+Olm5ciWvvfYa7373uwHYtm0bP/jBD3j88cfHPE4IwaZNm9i0aRN33XUXH/vYxyZkFIZyChqNZvrIe3mybpacm8MN3UsaDpoolaw++iZw7xnbfg6sklKuAQ4Cvw8ghFgBfABYWT7mr4UQZgXXNu187nOf45vf/Ca/+tWvhrcVCoUxj2lra+O1114bfr1z504WLFhQsTVqNJqJE4QBGTdDR76DowNH2du7l+OZ4/SWenFCpzIGoYI2pmKegpTyl0KIhWdse/q0l1uB95d/fjfwXSmlAxwTQhwGNgEvV2p9083MmTP53ve+x6OPPsqpU6dobGykvr6eP/zDPzzvMZ7n8YUvfIG2tjZisRgNDQ187WtfAxh+/sxnPgOoyqUf//jHw8du3bq1gp9Go3lrIqXED30KvgoFTZsnIMHxQ4peQMkNsAzJoprKXEpIWbkPUzYKjw+Fj85479+B70kp/1kI8RiwVUr5z+X3vg78TEr5g3Mc92ng0wDz58+//sSJ0bMi9u3bx/Lly6f6o1y16N+XRnN+Sn5pOCmccTMEMpiWUJDvS4pegBMEOF5IyQs4/VZdHUuyZdFZt9VxI4TYIaXccK73LkmiWQjxPwEf+JeJHiul/Dvg7wA2bNhw5QTqNBrNZc1QQjjjZoa9gKG+gEri+iGuH+L4IUEYUnAD/OAtVJIqhPgoKgF9hxxxU04B807bbW55m0aj0VSMIAzIeTmKfpFBZxBf+hW9nh9IciUfX4a4vsT1J2YAvNCjq3iK3f0dVKc81jWum/I1TqtREELcC/wP4DYp5elZ1p8C3xZC/L/AbGAp8Mp0rk2j0bw1CGVIyS/RW+ol61a4UUxC3vXLuQAVBhr3oVLSVWzjVOE4J7KHOZU/TlepjVCqc1h26coyCkKI7wBbgHohRCvwRVS1URT4eVmhc6uU8jNSyj1CiO8De1Fhpc9JKcf/29NoNJrzIKXEDV0yToa8l6fgFypqCHxfknN9HC8g7/qE44xABaHP8dwhTmQP0+d0cTSzn4zXD0DMjDMvuYhlNauZnVjANTOu5X3Lb6/I+itZffTBc2z++hj7/wnwJ5Vaj0ajeWtR9IsMlAbIuJmKhoU8PyTvBuQdHy8Ixx0O8kOfE7lDHMseoCV7mPZCC8WggECQtmuYm1zIspp3My+1iLpYE+ZpVfrVsWTFpO91R7NGo7kq8EMfJ3AoeAWybraiXcMlLyBb8ik4AV4wPlfACUocyx7gWOYA7cWTnMofxwmKCAQzE/NYXruOa2vWsii9jJiVqNjaL4Q2ChUglUqRy+VGbfva175GIpHg4YcfPu9xZ6qkAmzZsoWvfOUrbNiwgVwux+/+7u/yzDPPUFNTQzqd5i/+4i/YvHkzpmmyevVqpJSYpsljjz02SnxPo7kacQKHrJsl62Yp+GM3g14UUhmCwaJHwQ0Iwgt7A0U/z7HsAU5kD3Eos4euYhsApjCZmZjHqtoNXFuzhsVVK4iY0cqtfYJoozBNDDWZXQyf/OQnaW5u5tChQxiGwbFjx9i7dy8wWm/pqaee4vd///f5xS9+cdHX1GguN0p+iZ5iD1k3W5mSUQluEJJ3fEpeiB+EuEHIhVq6Sn6BI5l9HMns40TuEF3FNiQSU1jMTy3hzjmbmJtsZmH6GizDnvp1TxHaKEwTX/rSl0ilUnzhC19gy5YtbN68meeff56BgQG+/vWvc8stt4x5/NDchX/5l3/BMJQ6SXNzM83NzWftm8lkqK2trcjn0Gimm6Gy0byXJ+fl8EJvyq8xVCqad/2zGsXORyhDOgonOTi4m0ODuzmZO0JISNSIMT+1mJW1G1hUdS1zkguxL2MjcCZXt1H42e9Bx66pPefM1XDfn1/0aXzf55VXXuGJJ57gy1/+Ms8888yY++/Zs4e1a9dimueWhBqS6y6VSrS3t/Pcc89d9Bo1mkuFEzj0l/opeAWcwJlyjyAIVMdwwfUpuuG48gJFv0BvqZOOYisncoc4mtnPoNsHwOzEfG6ZdS9Lq1czL9mMaVy5t9Yrd+VXOA8++CAA119//fC4zvNVE4ynyuD08NHLL7/Mww8/zO7duytWoaDRTCVe4JH1smScDKWgRDDVFekSco7qFxhvcrjkFziRO0xPqYP9A29wIntoZPKZmWRBegl3zHk311SvImVXT+16L0AoJSUvIGZPvW7o1W0UpuAbfaWIRlViyTRNfF+Vy9XV1dHf3z9qv76+Purr66mpqeGNN94gCILzegtD3HDDDfT09NDd3U1jY2NlPoBGc5GEMqS/1K8qhfzilHoDMgTHD8i7ASUvwPXDCyaHs94gBwd2cTS7n55SB+35luE11UWbuHnWvcxNNtMYn82MaAOGmL4R96GUdA0GHO3yOdzpsfdUL5+9LcZv3rF0yq91dRuFK4yNGzfy+c9/no6ODmbOnMn27dtxHId58+ZhGAYbNmzgi1/8In/0R3+EEILjx4+zZ88e7r///lHn2b9/P0EQUFdXd4k+iUZzboZkpvtKfZSC0pSdNwzB9QOKboDjh5TGKR+R8zIcHtzD7v7tHBh4E4kkaaVpjM/mlln3sbhqOY3x2SSt9LR53aGUDBZCOgYCDnV4dA4GHO3yKLjq86RjgvULE6yfX5m8oTYKFaBQKDB37tzh17/zO78zruOampr46le/yjve8Q7CMCSVSvGd73xnOLH8D//wD/zu7/4uS5YsIR6PU19fz1/+5V8Co0eASin51re+dUGPQqOZDoIwoN/pn3KROccLKThDyeHxn7Ov1M2hzG7e6N3GydyRYUNwy8x7WVO3icb4nGn1AhxPcqrfZ0+rS1u/T0uvT66kDIBpQEPaZOXcCMtmRrl2VhXzapPUxJNcN6cyw7MqKp1daTZs2CC3b98+apuWgp4Y+velqRRBGNBb6qWv1DdlOYIhQ1DwgnFXCTlBiYMDb3Ios4eW3BF6Sh0ANMZns6p2I8tqVjMrMX/aDEHnoM+pvoBj3R6HOjzaB9TvxhAwq8ZkZo3FokaLxiqLlbNqqIqlSFhJbCMyfI54xGRJY2rSa7jspLM1Gs3Vx5DGkBu4ZN0sg87gRXsFQSBx/IBM0R93SGioVPTQ4B4OZVSpaCADElaKucmFbG7cwuKqFTTEZk1LSChbDDnV73O0y2PnCZfOQWUEbBMWNdqsXRBlZo3Jslk2dYkqomaMqBknZsan1WMZQhsFjUZzUeS9PP2lfjJu5qKF5kpuQM71KTrBuBrGhhjKDRzK7OHw4B7yfhaAmfF53Nh0F0uqV9KcXlbxm2wYStoHA453exzv9jnc4dGXV4ZRAIuaLG5ZlqS50WZ2jU3MsolZCVJWFREzijHWFOLQx/ByWKUBrIgJjZMfsjMW2ihoNJpJkXWz9BR7Ji8vIaHonZYc9sYnHwHghS6t+eMcGtjFocwe2gstACSsFEuqVrK0eiVLqleSrnCpaMEJOd7jc7zb41i3z4keH8dTnyEZFSxqtLl1uc3cGRbN9SnqEyliVoKIEcEU1pieiggcDDeH6WUxvCLIkaY9YaUr9pm0UdBoNOPCD336Sn0UvMKkewlyJaUkWvJCCq4/bk8AIOcNcix7kGOZ/ezu30HBz2EIk/mpxdw5570srV5Z8dxA0Q050umx95TH4U5vOBQkBMyuMdnQHKW5waK50aYxbRO3ksTMOCm76oINbSJwMLw8hpfHdDNwiaYHaKOg0WjGpOAV6C52U/AKE8oRyFANmBkaNVn0xj9bACDvZWkvnOTQ4G4ODr5JdzlBHDGiLKpazvX1N7EwvYx4hRRFS15ITyakMxPQPuBzqMOjpccnlBCxYEmTzYbmKAsbLBbU28QjFgkrScxMEDPjY4vchT4icDHdjDIEQemSGYEz0UZBo9GcRShDBp1BMm6GnJe78AGMzBouuP6k5gyHMqQld4Qjmb2czB3lcGYPAIYwWVy1nPX1N9OcvoZZyQWjZgtMFb3ZgAPtHoc6VDL4VP/ITdoQML/O4s5VcZbNijCvziJqCyxhkbBTJKw0cTNxflUCv4ThFxGBg+kMIkJnytc/VWijUAEmI5391FNP8eijjwJw+PBh5syZQzweZ82aNXzta1/jU5/6FG+++SZSSmpqanjyySdJpSZfkqbRnImUkr5S3/Dg+gsljcMQCq4/LCQ3mep2KSUdxVbe7N3Gm32vMOj2IRA0xmdzY9OdXFO9mjnJ5op4AwU35HCHx/42j32n3OGEcHXCYFa1yb1rIsyssWiqNmmsMrFMdcO3DJuoESNmJUjb1ecOV8kQ08lgOv0YXh4qoeZaIbRRmCYuJJ19zz33cM899wCjZygA/Nmf/RlNTU3s2qXE/Q4cOIBtXzmqi5rLFykl3cVu8l4eN3DHnlBWnjdc8kKKro/jj7866MxrthdOcmDwTXb1vkJXqQ1DmCypWsFdcx/k2uo1FRky4wWSo10ee0+5HGr3aOsPkKhQ0LJZEW5fabNslk1jlTnqG78hTKJmlLRdQ8yMn1P2Wvgl5QW4g8MewZWKNgrTxMVIZ7e3t7NgwYLh18uWLZuOJWuuUqSUFP0iGTfDoDN4XkMgQ/DCkIITkHM83CCcUE7gdNzA4Wh2HwcHdnFgcNewuuiC1FLeteC/sLL2epL21FbUhFJyolv1Bxzv8TnQ5uH4EtNQ/QH3XhdlSZNNc6OFaSgjYAqTmJnAECZxK0HcTJw3QWy4OQw/j+lmEZUc8DPNXNVG4S9e+Qv29+2f0nNeO+NaHt306EWfZyLS2R//+Me5++67+cEPfsAdd9zBI488wtKlUy+Epbl6CcKArJul3+mn5JfGTBgHgSRT8ugvuJM2AqByBEcy+9jR/Z/sH3gDX3pEjChLqlZw++wHWFa9mnSkZvIXOAe9uYCTvT4nun12nnCGQ0J1KYN1CyOsma8MQdQeHfuPmjGqIzPG1jgKPVUZ5OUxnYHLJjE81VzVRuFy5lzS2edj7dq1HD16lKeffppnnnmGjRs38vLLL2t5Cs2YDI2q7C/144bueffz/JCCqxRFXV/1CkxW/cYPfdoLJ3ijdxt7+18n4/UTN5NsaLiF5bXrWJBaijWFswYcT3K40+NAu8vB9tGSEctm29y/LsE1syJUxUfH/Q1hEDXjJKwUSSs1OiQkJaJsAIZCQYZfhLFCa9OIAOJ25cpur2qjMBXf6CvFuaSzxyKVSvHggw/y4IMPYhgGTzzxhDYKmnOSdbN0F7sp+ucfXB8EksGiR95RuYGLwQ0cXu/9FTt7ttJWOEEgfSxhs6R6JffVPcTymrVTNn6y4IR0DAYc7lCG4Fi3TxCCZcCiJpsNi6IsbrKZU2sRsUa+8QsEUTNOxIyQsqqImvHRHkG5W9h0M5huflSj2KXGFIKIZRALC1T17cLu2YvoOwq9H4SNn5zy61XMKAghvgG8E+iSUq4qb5sBfA9YCBwHHpJS9gv1r/NV4B1AAfiolPK1Sq3tSuOll15ixYoV1NbW4roue/fuZcuWLZd6WZrLiCGvIONkKAbnNgZhecbAYMGbdLXQ6fQ7vWztfIZXu3+JGzrMSsznhqY7mJtsZlHVtSSsi6uOC6WkJxvS2uuzv83lZJ9Pezk5DDB3hslty+Msm2WzqNEeZQRAVQlFjChxKzlaUE5KROiqbmE3iwidyyIxbAiIWiYRy8AyBXahh1T/XkTnLujYDf3HVKLHtKF+GVRoxGclPYVvAo8B/3Tatt8DnpVS/rkQ4vfKrx8F7gOWlh+bgb8pP1+RTFY6+3wcOXKEz372s0gpCcOQ+++/n/e9730Xu0zNFU4oQwpegQFn4Ly6Q64fUnQDCm4w4Q7ic5Fx+9nTv4PdfTtoyR1GIFg1YyObGrewILXkogTmpJR0DgYc7lTqoYc6PPLOiGTE/DqLNfOjzJthsrDBJhUbHUIxhan6BawEMTMxKkwl/BJWvlWFgi6jRjHbNIjZBjERUlVowWg/CD0H4NRrkG0v75SAppWw8CaYuwEalkOsBhquqciaKiqdLYRYCDx+mqdwANgipWwXQswCXpBSLhNC/G355++cud9Y59fS2ReP/n1deYQypK/Ud95cQdENKLkjU8culs7iKXb1vsqx7AFacoeRSJric1hZez3r62+iJjq5YU5BKGnrV0bgaJfHkc4RI1CTMLim7AHMqjGZVzdSITTEUEhoqEpoVBmrlAi/WA4HDV4WngCofECKIsncMex8G9HB44ju/dB7BMJyyCqahplrYM569TxjERhnNOvZyYsyCpeTdHbTaTf6DqCp/PMc4ORp+7WWt51lFIQQnwY+DTB//vzKrVSjuczIuBkyToa8lx9VRuoHspwbCCbVSXwu8l6WN/u28XrPr2grtGBgMDMxl7fPeRerajfSEJ854XOGUirl0E5lAI51j4jH1aUMVs6NsLjJZnGjTX3aOKfXETMTZWnpGAkrNbpxLAwwnQFMN4vh5y+5N2AKQRSXeN8e4r17sHLtWJmTiL4jIzvZcai/Bla9DxqvVWGh9EwlpnSJuGSJZimlFEJM+K9XSvl3wN+B8hSmfGEazWWElJIBZ4CuQtfofgIJA0WPTNHDvchE8RBu4PBK1wscyezlaPYAgfSZnZjPffMe4rq6t5GyqyZ0Pi+QtPb5HO/2OdmrtIMyRbXWmWXxuCVNNouaLGoSZ8tWGMLANqIkrCQJK0XEiI4yFCJwEX4Bwy9ieOr5UnQOC8AyDRK2QSzfit13kGjmOEb3fujaC6EPwoR0E1TPhYU3w8zVUDMPkg1wCWYmjMV0G4VOIcSs08JHXeXtp4B5p+03t7xtUkgpp22e6pXMlTx172qm5Jco+AVybo68lx/uKRgSmCt6AQUnwAsu/gbYVWxjZ+9WOgonOZ49iBs6NMXnsKnhNq5vuIWZibkXPkkZx5O09CrZiAPtHm39qjIIoCpusLjRYuXcCMvnRM7KB4AyAhEjdponkBw1X0D4pbJ4nFITFedJqFeaiGUQswxipiA2eBi7bTtG124VAiqqpjzMiAr7rP61chhoNVixS7LeiTLdRuGnwCPAn5eff3La9s8LIb6LSjAPXiifcD5isRi9vb3U1dVpwzAGUkp6e3uJxa6MP9S3AgWvMDzU/vSkcclTFUMFd/zzBs6HlJKsN8DRsvz0/oGdGMKkPtrE2rq3saZuMwvT44tVB6HkYLvHrpMux7pVj8DQ94zmBosty+MsaLBYWG9RfQ5PYAhLWKQjNVTZNaO6h4VfLHsBeQy/MO15AUMoD8AyDKKWesRLnVgdr8Op1+HUDigNqJ1nLII518PstSopXD0XprAfYzqpZEnqd4AtQL0QohX4IsoYfF8I8QngBPBQefcnUOWoh1ElqR+b7HXnzp1La2sr3d3dF7H6twaxWGxUlZTm0lDyS3QXu8m4mVHbVeloQMG5uKYpJTp3kr39r7Oz92X6nR4AklaaLbPfyQ2Nd4xLYiLvhBzvHhko09KrcgIRCxY12KxeHWFBvc3CeovkOTyB0zGEQdJKk7KrR8TuwgDDzZZF5IrTqiRqCkE8YpKIWEQtAxMfu2s3ZNog2wGDJ1VVUFbJd5OoU5VA89+mnmNT25l9Kalo9VGlOVf1kUZzpRCEAd3FbnpLvaO2u35IV6ZEyZt8eMgLPVpyhzmWOcAbfdvod7oRCBZVXcu1NWuZk1jAvNTiC3rTRTfkjRMuLx8qcbxHGSdDwOxaVRZ6zSybFXMi2ObY5xEIYlZ8uG8gLqKYQREReJheFuE7iNCFixznOREsU5CMWKSiFglZgM690LkbuvZA1z7wyuEpYUDVHKhdqEJBc9ZD9fxLmgy+mqqPNJq3NFJKcl5ueFbBmWGigbxHbpKeQShDTuQOs737l+zrfx03dMqGYDlbZt3P0upVVF1Aa0hKSUuvz2vHHfaf8ugoTxZrqjZ5x9oEixot5tedrR10JgJBxIwOVwnFzASWX8Rws1j5k5ekY3ioOaw6ZhHNtxI5tV0lgnsPK08AlAGYsRiW3g1zN6rKoFg1WGMMzLnK0EZBo5kmeou9DDgDlIISoBLHgyWPkhtQnMB84tORUnIid5g3ereyt/818n4W24iwru4GltVcx9xk8wVDQ0U35GC7mimwr81joBBiGrC0yeb6ZiUbsahx7HnCMJIoTlhJUnY1ljBUVZCTwyqdmPYSUVMIUjGLSFgiMXgIu3svomuvMgROOVSXnqXyAdfcC00roGGZaha73Dmzb2EK0UZBo6kgfugPaxF55eakMIS+vEO25E86cXwqf5xtXS9waHAXWW8Q24hwbc11rKy9nqXVq4iaYxcQ9GYDth9z2HnCoa08YSxmC5bNsnnH3Air50dIRMZXKmkJi9pYA0krjSGlkpMudGKeERabDiKmIO10Eu/ZTXTgoCoL7TuqLDCoENDCm1UyeM71qifgskeoyqVYlWpsM2ywK1cgoo2CRlMBCl6B3lIvWTc7HCIquhenO9TndLOv/3V29+3gZP4IESPKspo1XFO9hhW168Y0BEGoegYOtKlqoZZeFaJa3GjxjrUJFjfZNDec3TU8FnErQZWZJh1KrGI/htc67d5ARPgkC60kCh3EenZjnHgRCj3lN5Pqm/+6jygj0Lhc3VQvd4SpDICdGHkY09fLoI2CRjNFDM0s6Cp2DXsFSMiWfAaL7qQSx27gsHfgdXb3befAwBtIJI2x2dw//4NcV7f5vKJzoZS0DwQcavc42OFyuHOke3hBvcU71yW4vjnKjNT4wxACQUyYxAKfahEh5noYfivTkRw2BEQExJxOotlWon0HsNtewRg4MSIPYUZh3ib1mLkGai5xMvhCGJZ62AnV2WzHVX/DJc5faKOg0VwEQ4njc0lV5x2f7qwzYdmJgp/jaGY/b/Ru5cDgLkIZUGXXcvPMe9jUuIXaaP1Zxwwpih7vVrmBgx0euZK6bkPa4PrmKEubbBY32VQnxvet05CQEJYyBFIQDwNsP19+9/zzGaYCyxQkLINUsY1I717s9h3QshX8UnkPAbPWwOr3K2+geq6qELLjFV3XpDAsiKSUuqlhKw/Gjlc0L3AxaKOg0UwCJ3DIe3l6ij0jXgGqnDRX8hkoTmxqWShDWnKHeb3nV7zW8xISScJKsbnxdlbUrGNBeulZA+IzxZA3TjjlDmKXIe27VEywfHaEa2bZLGmyJ+QNIAOivksaqAsNTFHZm/8QAohHLGwRkswcJtG2DXH4aciX+40SdbDkLpUMrp4L9Usvww5hob7p23H1bd+KqUfkCkhcn4Y2ChrNBBgoDdBb6h2uIAIouQGDRY+iN3ExOj/0OTDwBi92PM3J/BEsYbG+/mbW19/InOTCs4bTBKHkUIfHtsMlXj/hIqUyAhsXxZhfb7Gg3mJmjYkxkbCJlET8IukQaiREhzpxKxh5MYUgZpskLEjkjmN3voFofx3ad4GXV6WhczfC9Y9A06pyKOjy0gjCsCBapbwAO6bCV+aVf0u98j+BRjMNZN0sA6UBMl65lFGW+woKk+srCEKffQM7+XnrD+l1ukjbNTyw4MOsrbvhnAnjrkzAC3uLvH7coeBK4hHBrdfGuOmaGI1V5qQkXeKBRx0R0oGPwFJGoEKGYMgTSNgGyXwLdseQEXgDnKzaqXouLH47zFmnKoNi1ZVZzGQxoxCvUZ6AMFUu4CowAmdywU8khPgLKeWjF9qm0VyN+KFPV6GLfqd/eFvB8enOupMSpGvNHeO1npfY2/8aOT9DQ2wmH17yeZZWr/vP6BsAACAASURBVDprdrHrS95ocXj5UIkjnT6mAWsXRLluQWRcXcRnYgmLpIR0KIn5LhEkUBkpiYhlEDENTENQHTWI9u2DozvhyHPQf1ztlJ4FC29RekGz1ynF0MsCoW74iRkqHCSEMghW5FIvbFoYj5m7CzUd7XTuO8c2jeaqYqA0MKqSqOQF9ORcSu7Eyi6llBzN7ueX7T/jSGYvthFhadVK1tXfxDU1qzHF6Jh/yQ35+e4iLx4oUfIkdSmD+65LcOM1sbMG0F8IgSBtRKgJApKui6Ayw+ctQ2kHRS2TuG0SMwIlGXHsl9CyDbJtgFDSDLd+AeZsuDx6BIYSv6fnAKzYtJaAXm6c1ygIIT4L/AawSAjx5mlvpYGXKr0wjeZSMFRNlHEzDDhKAVOG0JN3GCxMTJqh6OfZ1bedbV3P0Vk8RcJKce/cX2Nj423nDBENFkJeOVLiF/uKZEuSdQsi3LQsxuIme9w5AlOYxMwECSmIS4gEDpFSdkLrvvA1BLYpsEyDZMQiETWxCl3QtV/JRXTshvadqlLIjMCs6+D6jyrxuNjEZjJMKcJQmkHRlAoBRVKXbQXQpWQsT+HbwM+AP0PNUh4iK6Xsq+iqNJppxg99eoo9o2Srg0AyWPTIlLwJJZBbc8fY2vUcu/texZc+TfE5PNj8MVbVbiBijtSg+4FkX5tLa6/P0W41hEZKWNxk8cnbkyxsuPBg9uGRlBhUhwHxwEM4U99JnIoq4bhYxMQ2hJKM7tqv5KNbXobB1pGdq+fCNfco9dDZ69TNdzoZivfbMWWUTLsc/nlrewDj5bxGQUo5CAwCHxRC3AwslVL+oxCiXgjRLKU8Nm2r1GgqgJSSQWeQvJ8n42RGDbPpK7gMFNxxdx67gcOe/tfY1vUcrfljRIwo6+tv5vqGm5mdWDAqEXyqz2fbkRLbjzrkHYkQ0FRlcsfKOJuXqMTxWAgECTtFtTRIeUVMp1BWGJ06BEo8LmobVMVsYkFGVQad2qEeQwJyZkQ1iq18b3mOwPzpLcE07HLNvzUiBXE59ipcQYwn0fxFYAOwDPhHIAL8M3BTZZem0VSOrJulI98xavC9H0iyJY/B4vg9Az/0eb33Vzx/6t/JeP3Ux2byzvkfYm39DcTMkZtTvhSy9XCJN1tcjveopPHqeRHetkSFhyLWhcNDSTtNCpt06GOXcojTymKnAgGkYhbVcZuoM4DRtgPaXofufSPJYSsGs9bCtfcrBdGmldPUgSvK9f+xcvw/CpH0VVn9c6kZz2/0vcA64DUAKWWbEOIKEBDRaM4m62bpKfZQ8AvD22QIXdkS2dL4k7BZd4AdPS+yresFst4AsxPzec/Ch1lSvXK4ycwPJAc7PF494rC71cH1YU6tyXs2JNm0KHrBQTQRI0LSriIqIiTcHFGngPD7mCpZCQHEbJNoebxkyulEnNiqksOdu9UvJloFjStU49is1dCwXIVjpoMhDaBIWj1P13Xf4ozHKLhSSimEkABCiGSF16TRTClDYaIzm86GZh735MYnRSGl5Fj2AFu7nmP/wBuEMmBR+loebP4oS6pWDoeIBvJKgfRXB0v05kISEcGG5hi3XBtjdu3Y/+UEgriVpMpKkvYczOLglA6ktwxB1DJJRExSUROr/wgcegEOPwO5TrVT3WJY9X5YeifULZn+pjE7qfoBEnU6EXwJGI9R+L4Q4m+BGiHEp4CPA39f2WVpNFODH/q05drIeqdV4EgYLHr05d1xSVf7oc9rPS/xStcLdBRPkrBS3NB4Bxsbb6U+psoqHU+yr00ZgoPlhPHCBosH1idZPS+CdYGegqgZo8aIkw48IsUsTEEthwCitkkyYmKX+wYi7qBqGus+qHoG8l0j3cPXfVAlh6uneUSrGVW5iVjVW26gzeXImEZBqK8+3wOuBTKovMIfSil/Pg1r02gmTRAG9JX66C31EgzJOUvoL7j05cefQD4w8CZPtHyXXqeLmfF5vGfhw1xX9zZsI4LjSV475vD6CYd9p5T2UE3C4I6Vcd62JEbDBRLGhjBI2VVUW1XEnQxWvnPSn1fA8I0/ZpvELIOIZWIaqDnDB55RCeKOXcpFEgbMXg/rH1bzBeLTPGPYjCoZ62S9TgxfZoxpFMphoyeklKsBbQg0VwRe6NGSaRkVKhooeAwU3HEnkPucbp5o+R77B3ZSH5vJw0t/i6XVqxBCcLLX57k9GXa3urg+VMUNblgaY838CIsa7TFnEgzNKk5aadJGjGihEyPXNanPaZmCVMQiZpskI9ZItaWUMHBCSUgceU49I5SI3HUfhOZbYUaz+nY+XQhTJYmjaRUWeot0B1+JjCd89JoQYqOU8tWKr0ajuUgybob2XDu+VEnjohvQl3cpjqMLOe9lOZLZx57+Hewf2IkpLO6Z+z5uaLoLU5ic6PF56s0Ce095JCJKhG59c5RFjdYFm8sMYZCwUsyINmLLEDvfhuGenPDnG6oQqorZJCJlT0RK6D004gl07BoZN5lsgI2fhKV3Qappwte7aGLVKlEcr9WVQlcI4/lX2gx8WAhxAsij/i6llHJNRVem0UyAUIa059uHu5B9X9Kdc8iPQ6xuwOnj1e4XeLnzWdzQIWml2dx4OzfNvJu0VcueVpenduU42euTiAjuWRPn7SvixMYxrtIybKojM0jb1VhuBjPfhukMMtEKIkPAjGSU6pitPAIZQvubcOJFOPHySN9A9dzyuMlVMPs6SM+epkEz5ZGRVkQZgWi6PD9AJ4qvNMZjFO6p+Co0moug6BfpyHcMl5nmSmq4zVhJZDXw/hD/2f4kBwd3AbCidj23zLyHWckFZAvw4p4SrxzpZ6AQUpcyeP+mJBsXRS9oDExhkrZrSNgpYmYcw81iZVowhhRWx8lQorgmbpOIWCP5gRMvwYGfqdnDhq1KRVe/XxmDRN2ErjFphFGWjE4o7aBISncLXyWMxyicSzjlosRUhBC/DXwS9XVpF/AxYBbwXaAO2AF8REo5PRM+NFckg84gg84gWS+L76vGs5zj4/jnL98MZMCBgTd5seNJWnJHiJtJbpl5Lxsbb6M2Wk+mGPL4jiIv7CsigWtn27x3o6ogutD8YktYVEVqSdvVmMLALPVhFU9MuNvYEFAVt6lJRMqSEhnY9QQcfHKkiay2GW57FBbcOD0S00OzA2JV6ll7AFct48opAPOAftSXlxqgQwjRCXxKSrljIhcUQswBfhNYIaUsCiG+D3wAeAfwv6WU3xVCfA34BPA3Ezm35q1BKEO6Cl30lpTGT8Hx6cyM7RmEMmRP/w6ePfVjekqdpKwq7pv3EBsabiVqxhgsBHx/a46th0sEIWxaHOXe6xLUjXNqWcquoj42EwMwvBx2rm1CxkAAybK+UCpqIZwB2P1TOLlNicuFvpKTuOFzsOAmqJo97nNPGjtZzgkklCdwOc871kwZ4zEKPwd+IKV8CkAIcTfwPpTkxV+jcg6TuW5cCOEBCaAdeDvwofL73wK+hDYKmjPoL/XTVejClz5BIOnJORfsRN7Z8zLPtf2UPqebxvhsPrj4s1xbuxZTmORLIU/uzPOf+4sEIdywNMat18aYWXPh/xpDjWa10XriYYCZO4XpZkCOT1o7FbVIRiwiloFtGpheFo4/D0dfgNbt6jy1C2HFu2Hp3WoWcSUZGiIfq1E5AV0h9JZkPEbhbVLKTw29kFI+LYT4ipTyvwohJtxlIqU8JYT4CtACFIGnUeGiASnl0P/uVmDOuY4XQnwa+DTA/PnzJ3p5zRXKkIppb6l3eOpZV9bBPU+oaChn8Oypn3Isu59Zifk8tOjTrJqxAYHgeLfPiwcLvNni4PmwYZHyDOrTF/YMDGGQtNLU2jVEvQJmrh3Du3BEdWgEZTpuEbdNrKFwVL4H9v4Y3vweBJ6aM7DmIWUIZjRP6Pc0Ycyo8gYSM3S/gAYYn1FoF0I8ior3A/w60CmEMJlE770QohZ4N9AMDAD/Ctw73uOllH8H/B3Ahg0bpkYERnPZEoTBsKR1KEP68y6DY0hZhzJk/8AbPN/2U9oLJ0lZVbxj3gfY3LgFMNl+1OGX+0uc7PWJRwTXL4xy24o4sy7gGQz1FyTMJOkQol4OM3/oguu3DGUI4rZJddweicC4BTj2Czj0cyU6h4RFW5QxaFhemVDNkKR0rGrEK7DPnuugeWszHqPwIeCLwI9RieGXyttM4KFJXPNO4JiUshtACPFDlOJqjRDCKnsLc4FTkzi35iqi6BdpybTgS5+849Of9yh55w/N7Ot/nada/42eUge10QbeveAjXFf3NiwR4bXjDj97I0NPNqSp2uT9m5JsWhwjal+ov8CkCoNaaRJzSojgwuWkphCkYhbpmPIIRpHvgTe+A/seh8CBqjmqq3jRlsp4BVZ8RD7CTui8gOaCXNAoSCl7gP8mhEhKKfNnvH14EtdsAd4mhEigwkd3ANuB54H3ozySR4CfTOLcmqsAJ3DozHeS83J4QchAwWVgjKlnvaUunjz5ffYN7KQxPpuHFn2albXrEZjsbnV5/PUBOgcDZtaYfGJLmtXzIhccdK8qKmwa/BArOPPP/tzYpkFdMqISxaefPvSV8uj+J1TSWIYqNLT8AaVAOpU3amGqEtFolcoLaE9AM0HGM0/hRuAfgBQwXwhxHfBfpZS/MZkLSim3CSF+gKpq8oHXUeGg/wC+K4T44/K2r0/m/JorFz/0h9VMvUAJ1vWPMehmwOnjpY6neKX7BQxhcvfc93FT010YwuT14y5P7crSMRAwI2XwyC1p1i6MjN15LEMigUeVMJkhDaxwbGMwXDEUU1IT9uklq1LCya1w8GlofRXcnGokW/V+WPGuqa0eilap5HAkofMCmotmPOGj/41qYPspgJTyDSHErRdzUSnlF1EhqdM5Cmy6mPNqrkyklGTcDJ2FTrzQw/clHdkSpfNIU+S9LE+e/Fd29r4MwPUNt3DH7HeRsKrZecLlF/tynOjxaao2+cjNKdbMj445xMaQkpogoNZ3iRoWKlV27nSZAKrjNumYTdQyzv6S7xbg0NOw+99Ul3G8VmkNLbgJFtwwdTLUVlxJWGhDoJlixiVGIqU8eYa7Pb6aO43mAmTdLG25NnzpE4YwUDi/dxDIgK2dz/LMqR8ThD43Nt3F5qbbiYk6ntlV5OVDfRRcSWOVya9tTnLj0hjGWOJ0gUO1W6RBRLENUyVfz7UfSm8oGbFGC88NEQaqy/jgU9D6iqogalgOW34fltxx3vNOmGiV6heIVeuwkKZijOev9WQ5hCSFEDbwW8C+yi5Lc7XjhR69xd7hBrRs0ac7VyI8xxd0KSWHM3t58uT36SyeYln1Gu6Z934a47PZfdLh+1sHyJRCrpsfYeOiKCvmjhEmkpJEGJIOXGqkwDTPPU/YEJCIWMO9BOdUcHCycOBJ2PsjJT8Rr4Xl74LFb5+6XIEZVYni+IzpnX2secsyHqPwGeCrqL6BU6i+gknlEzQaUN7ByexJJJK849Obc8/bb5D3svz4+LfYN7CT6sgMPrj4s6yoXU+mGPJ/Xsyy/ajDrBqTT9xezYL6849rNLwica9AkxEnZliAqVyAM4haBnHbZEYyqrSGzsQtwJFn4cjzSpJaBkp8btN/VZITUzEy0oworyAxQyWNNZppZLzVRx8eel3uM/gN4E8quC7NVUgoQzryHfQ7/YQh9OYdBs9TVRSEPs+3P86vOn6OF7rcPvsBbpv1Dtr64LGnBznWrfoc71kT5+7ViXNPNpMBppsj5bvUGhHiZuq8VUeGgPpUjOr4Of5LZNqVITj5ihpiH3hQPU/1FCy+XQ2wnwrsJKSbpkfLSKM5D+c1CkKIecD/DcwGfoQqFf0y8DDwnWlZneaqIefmaM+344YuJTegI1M6bwPa7r7tPN36b/Q53ayoXc+dc95DQszk+y/n2XbEIRUTbFke58ZrYuftQI44g6S9EjVWkqh9/pusZQhmJKNUxc4oIw08lSfY97iaU4BUN/8V74Hm26Bp5RSEh4TKEcRrVLJYewWay4CxPIV/An4B/Buq43g7sBNYI6XsmIa1aa4C8l6ezkInRb+oXjs+HYOlcyaSOwonear13zg0uJtZifl8aMnnWJxaywv7ijy3px/Hl9y5Ks4dq+IkziFfnTST1EhByslgijhEzl2VYwhVSlods4nZ5si9PfSVN3DsF9CyDUoDkGyE6x+BZfdN3ZCaeG1ZaC41NeEmjWYKGcsozJBSfqn881NCiF8DPiylnLC0heatRxAGtOfbGXQHh7flSj6dmbMNwoDTy7OnfsLO3peJmXHunfdrbGp4O68c9vnOM31kipI18yPcvzZxTqG6qBGliQip0iDI84vjRS2D6rhqLhvOF8gQOvbC4Wfg6PNQGlTx/DnXwzX3qkH2FysTPTR7QDeUaa4AxswplPMHQ9+jeoFqUQ7KSin7Krw2zRWIlJLOQif9pX7C02r9cyXlIZyOE5R4oe1xXu58BoCbZt7NrTPv42CbzZ/9JEtfLmRxo8XHtyRpbhj9jVppESWoFXGqS72I4PyCdHHbpCpmk4qeVkXkFmDfT1U/Qb5bJXcX3KTGVs7bdPFlpFa8LDSn5xFrrizG+suvRqmXnh44fa38LIFFlVqU5srECzxasi2UgtNu/hJ68y79+ZHZAkHo81rPSzzf9jgZr5+1dTdwx+x309ZTxd88XaClt8SsGpNPv72KFXPsUclhQxik7RqqrTTxUh9mqZ1zaRGdPp8gHTvtzzzboaaW7fmhKimdvR42fUoZhIuN6Q8Ppq/X3oDmiuW8RkFKuXAa16G5wjl9zsEQmaJPX94ZlVA+ktnHkyf/lfZCC/OSi/n1xZ+mkFnAN54t0NKboS5l8IEbUmxaHB016UwgSNppqswUKb+INXj0nHMLYpZBqjzUPmqV3QIZKjXS/Y+rofagjMC6D6t+goslPgOS9VpwTnNVMEWtlpq3Mu25dvqckWhiEEi6sw45Z8RAFPwcT7f+kO3dv6TKruUDiz/DLHstP3y1wO7WDDNSBh+8IcXGM4wBKKXS2bFZJNwsVr6Fc0lQVMUsahKREUMAqqHs6C9UmCjbDjULYOMnVZdxetbFfeih2cS6qUxzlaGNgmbSlPwS3cVuMu7IQPqSG9A+WBoejekEJV7t/gW/6niGnJ/hxqY7uaXxvbyw1+db+weQEu67LsGdq+Jn9RrYRoSElaLKiJLOtJwziRyzDBqrYiPGIPShZSvs+3dVSYRUzWWbP6M0iC7mm7wVU8niZD1YE54vpdFcEWijoJkwoQwZcAboLnSPhIsk9Bdc+vIjukVHM/v5yfF/otfpYl5yER9a8hvkM3P4f/4jR38+ZP3CKPevO3vamUCQsFLUR+qIFbswnY4z3oeobZKOWiODa5wsvPl9Ndw+360SvOsfhmX3XpxXYNiqjyBeq/sING8JxmUUhBA3A0ullP8ohGgAUlLKY5VdmuZypOAVOJk9OSp3EASSrqxDvhwuynkZnjn1Y3Z0/ye10Xo+sez/Yl7yGv799Twv7M3QWGXy3++tprnx7Bp9S1g0RBuocrKYA4c4M4kcs0xmVsewh7yKwVZVQXTwKfCKMH8z3PibSpF0shVEZkSFhYaayjSatxDjmafwRWADsAz4R8AG/hk1LU3zFqKv1EdnvnNUqWnJC+gYHOlO3tv/Oj8+/i1KQZG3Nb6dO+a+h1whwl89NciJHp+bronxng3Js6SsDWFSbcSpk4JIpoUzjcFZMhQDLfDK36uuY2EquYk1vw51iyf34QxLSVHHalRoSCeMNW9RxvNV6r3AOsrlqFLKNiFEuqKr0lxWuIFLd7GbAWdg1PZcyacrq5RNvdDjmdYf8VLn08xKzOeTiz5JjT2LVw6X+OmOARDwyC1p1jePjsWbwqQ2Uketk8Eunt36YpsGNXGbqpitegw698Ab34XjL6qb93UfgFXvU+GiiWInVI4gXqsNgUZTZjxGwZVSSiGEBBBC6MDqWwQv8IaNgTztm3sYQne2RLakwkX9Tg//fOj/o7N4ik0Nt3H3nA+w9ZDPU2+q+QaLGi0eviVNbXIkdyAQ1JhJ6n0PO3uKMz2DdMwiYaseAyFQ08t2fAs6d6tegHUfhpXvnZwxMGyVZ0hO4liN5ipnPEbh+0KIvwVqhBCfAj4O/H1ll6W51Jwub306/XmXwaKHH0iklOzo+U+eaPkeAsFHlvwm+YFl/K9/z9KbC1nUaHH/uiSLG63hBjTbiFBt11DjZImc4RkYgnKzmeozAFTS+JV/gENPQXqmyhcsu28SsX6hjkk2KM9AewUazTkZj3T2V4QQdwEZVF7hD6WUP6/4yjSXhCAMaMu3kXWzowyC64d0ZUqUPJVP6Cl18pPj/8Sx7AEWpJZyW/1H+dm2KEe7ssyuNfnMHVVcO3ukG9kQBrX2DGokRPJdiHCkw9kUguqETVXcHplzXBqEXf8Ku3+oFEvXflgJ05mTkIyIVinPQPcTaDQXZDyJ5t8BvqcNwdXPQGmA9nz7qESy54cMFj0yJY8wVOWoL3U8xbOnfoplWDww/8MEg5v426eKRKyAX3+b6kY+veegJlJLjVckke86q9cgHbOYkYwQGVKo6zsKe36kJpqFnpKp3vBxqF0w/g9iRpQ3YCfUQ2sPaTTjZjzhozTwtBCiD/ge8K9Sys7KLksznfihT1+pj+5i96jtuZJPd9YZbkTLeYN898jfcjx7kEVVy3lg3sf44csme08VWTrT5pFb0qTjIx3FtlugyUxSlevmzC7kVNSiNhEhZpf379qrqonaXldy0tfcA6sehBnjldgSalJZskGXkWo0F8F4wkdfBr4shFgD/DrwCyFEq5TyzoqvTlNx3MClJdOCEzqjtvfnXHpPE7Fry5/gnw89RjHI82Dzx4g7G/ir/8iRLXq8Z0OSW6+NDctTGF6B2sCnBoMoI+cVQFXcJh2ziNum6j4+9KzqMTi1XamKbv6MyhmMZ/qYMCGaUmWk0bSeTaDRTAET6e7pAjpQEtqNlVmOZjrxQ5+T2ZOjDILjhXRnR3IHI/LWz5K00zyy5FH2Hqvn57uyNFWbfGJLioVlWWvhl4gWu2kwU6StkSI1AaRiFnXJqGo68x3Y+zjs/DbkOtUgm02fUtVE9jji/pGyIUjUMaKFrdFopoLx5BR+A3gIaAD+FfiUlHLvxVxUCFED/AOwClWL+HHgACo8tRA4Djwkpey/mOtozk/JL3E8c5zgNKXRoqsa0YbCRfv6d/J4y7cZdPtYW3cDC8138Y2fmwwWiqxbGOHX35YiHjFA+liFHqJ+iZnReqLGSAw/EbGYkbSVZ1Dsh70/VQJ1hV410vLm31bzC8Q4bu6xakjN1AljjaaCjMdTmAf8dynlzim87leBJ6WU7xdCRIAE8AfAs1LKPxdC/B7we8CjU3hNDSpR3FvspafYMyqhfPqYzAGnj/9o+Tb7BnbSFJ/De5Y8yvZ9M/k/hx1m1Qg+dlv18NAb0xnEKnaTMhPUx5owhYllCFIxi6qYrYTq+k/Aq1+HEy8qGeu5G+H2/wmz142vNDSSVgPto7pnUqOpNOc1CkKIKillBvjL8usZp78/2clrQohq4Fbgo+XzuIArhHg3sKW827eAF9BGYUoZdAbpyHeM0i0CVW7amSnhBi5bO5/l+bbHkUjumfs+qv0tfPu5Iv0FhztXxbnvugSWKTD8Enaph6QUJCIzSJtJ0jGbqrhFMlL+s2rdDoeeVqMuzYgKDy1/AGoXjm/BdlIpkiZmXHhfjUYzJYzlKXwbeCdq+ppk9AS2i5m81gx0A/8ohLiufP7fApqklO3lfTqAc05JF0J8Gvg0wPz58ye5hLcePcUeOgtnF415fkj7QImeYjffOfzXtBdOsqx6DW+f9QF+uSvODw7naawy+a17R7wDq9hD1M3SEJlBwowjgKbqGOlo+c+p2A9b/0YZBDuujMHaD42v+9iMjoyx1NPLNJppR8gzp6hX+oJCbAC2AjdJKbcJIb6Kaoz7b1LKmtP265dS1o51rg0bNsjt27dXdsFXAV2FrrPKTUF1J/fmXE5kD/HdI1/DDz0ebP44FFbw3ZdzDBZC7lytvAPTECAlVqmHlF+i3p5BxLBJRizqUuXhNsV+eP1f4MATKpm87r8oOYrxNJwZNlTN1l6BRjMNCCF2SCk3nOu98SSan5VS3nGhbROgFWiVUm4rv/4BKn/QKYSYJaVsF0LMQlU7aS4CKSUd+Y5RU9GGGCx49OZcXu36BY+3fJukVcXDS/4HP345zeHODDOSBr/zjmrm15criwIHu9BNlTCpizQQtUxmVsVUAhngyPPwy6+AX1KKpes/oiadjYkoK5NWq9JSjUZzyRkrpxBDJYDrhRC1jISPqoA5k72glLJDCHFSCLFMSnkAuAPYW348Avx5+fknk72GRs096Mh3UAyKZ703kPdoG8zwVOsP2Nb1PNdUr+KumR/nOy+GnOz1eO/GJDdfE1NdydLHKvZgujnq7Fpq7DSpqOpCjloG5Lpg29fgyHPQuBz+//buPD6q+lz8+OeZSTLZd0hYwo5aBEFEBEGEioi7Vlq3VutSeltvW9vettrF2vtqf1Zv9bbWrVpt9WJVat1oFYqAUpUiIJuCQAxr9j2Zyezz/f1xTkJYEoaQZCbkeb9eeeXM9wwzzzdnkofzXWffDdmdNOuJw+orcGVYTURO3edJqXjS2W/k14E7gcFY7f6tSaEJeOQE3/dbwPP2yKMS4BbAgbX43m3AXqxhsKoLKjwV1PnqjljMLhQ21Lr9VLubeL74UfY07+ScgXOYlL6AR5e68YcMt87OYEKRtby1hP0kespJkwRykwtJcSYd3NMgEoItL8P6Z6wRRZNvtu4OOtrYJiHZSgLJ2brshFJx7Jh9CiLyLWPM73spnuOifQpHqvBUUOurPaI8EIpQ1uBlV/2nvLz7aZoDDVw98hZM82ReWNNMapKDb8zNpDA7OMqgcAAAHKRJREFUAQkHcPrrSQx6yE/MJiMhHVeCg4LWvZArP4H3fgu1u6BoGsz4DmR2sOWlIwFS8zs+r5TqdSfUp2CM+b2IjAfGAcntyp/rvhDViTLGUNlSSZ3vyP4Dty9EaUMzyw+8xvsV/yTXNYAbRv2Q97YMZOv+ZoryErh9TgbZqU4k5CPJU0aGM4UcVyGJjgTSXQkUZCRbk4fX/RE2LrL+0M+911qw7qhzDQSyhloL0zmcRzmvlIpH0W7HORsrKbwJXAy8B2hSiBPBSJAKTwVNgaYjzjV4ghxoqGdR8SPsad7JlAGzODfvGp5Y7qehJcDlk1OZMy4Fp0OQcIAMby15SfmkOJNJcAgDM13WvINIGNb8AbYutharm/GdjpekcLoga0h06xcppeJKNL18C4CJwEZjzC0iUoC1R7OKEx0lBLcvRFlTI4uKH2Gf+zO+NGoheTKZx/7ZRIvf8PULMjl1kNW+L2E/+b4m8pLySE607g6yUhJJEKxJaOueguodcPoXYPo3j9534Ei0OpmTM3u4xkqpnhJNUvAaYyIiEhKRTKyhokU9HJeKUkcJocbtp6Kxked2PcwBdwnXjLqNxuoJ/Hl9A2kuB9+6KIuiPOvySzhAQdBPbmIWuWlJ5KYmWS1CTWXw/u9g/1prZvHnfwZjOhiJnJRhzVTW0URK9WnR/AavtxewewprFJIbWNOjUaljMsawu3H3oUNODbj9IZp8IbZUbeG1Pc/SFKjn6hFfY+v201hX4uG0QYlcOz2d3HSrnd8ZCTE44CPTmUxWaqKVEMJ+awXTTX+xlqOefgeMu7LjSWjpBdaX9h0o1edF09H8TfvwCRFZCmQaY7b0bFjqWOr99UfMQSht8OINhNlat46XS54mKymHOTnf5dV3Cmjy+rl4YirzzkjBYXcMpztcDAz4yUlOY2CmiySHWLOR1z4O/mYYMxemLoT0DlZKT8mxtrlMcPV0dZVSvaSzyWuTOztnjPmoZ0JSx9I6Ma1VKGSocvvxBsKsKlvCitLXGZo2kjF8ndc+gKG5wq2zs9r2PQDIMQ6KQi3kZ6VYaxYFvbDqAShZBYMmWltgDpp49AAciVZHckqnq5Aopfqgzu4UHuzknAE+382xqCg0B5opdZe2TUyLRKC80YsvGObt0td4t/wfTMqdjtQtYMn2IBOKkvjqrIyDeyYbyI+EGZ0YoSAjxe47KIflP4W63dZmNxOv73h/g+QsyBkZ3ZLXSqk+p8OkYIyZ05uBqGPzhryUucvaNsYJhiJUuwP4gmHe2v8SH1S+zZjUGez+9EoO1AY5/3PJXHVWGo7WbTLDQQaHQoxITyEvze4f2PEWrP4fq7/govtg2DkdB+DKhKwiTQhKncSimadw09HKdfJa7/KFfOxr2te2F4IvEKas0Us4bHhtz7NsqHmP7NB5bNxwCVmp8OWZ6Zw9yp5raAyuYAsjcDAsJ53UJCeEA7D+T7D5BRhyFpz/o477DhLTrCUq0qJY+lop1adFM/ro7HbHyVgL2H2ETl7rNZ6gh71Ne9uajPzBCKUNXoyBFaWvs6HmPdJ8cziw+0LmT0zjwvEpbc1FzoCbZG8tYzMGMSQ7jQSHWFthLvspVG+HUy6G87579JFFrkwrUeiOZ0r1G9GMPvpW+8f28NQXeywidQhjDOXu8oOL2xlrDkIoHGJl2RLeLf8HKYGzqNwzjxtmZDB1dHLb8xJ8tWSaFiYUFJGdbJfXlVgJwVsHF9wDo4/SNZSQbA0x1b0NlOp3ujLTyIO1e5rqBZUtlfgj/rbH1W4/Lf4QS/b9hfXVq8kITaGs5Cqum9Y+IRgSWioZmBBhYv5wkpz2MhXbXrdmJiemwmUPwcBxh75ZchakDdS9DZTqx6LpU1gCbWswO7DWQFrck0EpS6239pAVTz3+EI0tAd4ufZX11atJ8Z5H2Z5LuWJyKtPHtiaEMKm+KgpTnIzPKUJErFVN3/8d1OyEIVPg/B8e2n+QnG3dFehaRUr1e9HcKfym3XEI2GuMOdBD8Shbna+OipaDcxE8/hAlNXW8UPwYJc2f4vBMpvbAfL54ThozT02xnmQMAyI1DM50MSQlHwm4Yc2jsHOp1VF8wT0was7B0UPaTKSUOkw0fQrvAtjrHiXYx7nGmCPXaFbdYl/TPpqDzW2P3b4QJbU1LP7sKUqad+CovxxvzbnccWE2owvs7TIxFFBLYXIiRakDcdTvgeU/t9Yvmni9tQFO+1VNXZnWfANHB/MRlFL9UjTNRwuB/wZ8QARrBzYDjOrZ0Pqn6pbqQxJCiz/EZzXVPF/8CAc8e0hqvIKm6ul866Ishtv7J2MiDJJaBiY5GZ5WCPvWwKr7rHOXPXTkzOS0gZA5WOcbKKWOEE3z0Q+A8caYmp4Opr/zBD1UeavaHte7A1S5W3i++FFKPXtJbbyWqooz+NqczLaEIJEQOeEaCtOSGZySDxv+bH3ljrY2wclut6Ct02U1F+l8A6VUB6JJCp8BLT0dSH8XDAc50Hywq6beHaDWE+Dve19gn7uYLPeNlJVN4JbzM/jcEGtOgSPoJjdcw7DMLIa6cpHNL1kJYeT5MOfHhy5Ul5ILmUN0aWulVKei+QtxN/CBiKwF2sZGGmO+3WNR9TPGGPa797fNVnb7QtR6/Czd/zLrqt8l3Teb0v0TuOm8DM4YZv2hdwQ95Js6BmVkMIQEZOkPofQjGDETLvjZwU1wnC7rbkEnoCmlohBNUvgDsBLYitWnoLpZqbsUb8haBjscNlQ1+3i79HXeq1hGZmgapbsv5Ppz05k80koICd4aBiR4GZyRRWF9KY6VvwR/I8z6AZx6ycG+gpRcayc07TtQSkUpmqSQaIz5Xo9H0k9VeCpoDDQCEAobyho9vLn3r7xXsYyM8JmU7rqCy85MZ9qYZDCQHKhmYJKPwanZFOxcDh8+ae1pMP8xyB9rvag4IGMwpA+IYc2UUn1RNEnhLXsE0hIObT46oSGpIuIE1gOlxpjLRGQk1vIZeVg7vH3FGBM4kfeIdzXeGup81o/R2Etg/2XH02yqXYPLO52yvZdy6ZnpzB2fAgZc3jLGZCQzNHkwrjWPwqd/h5GzrMloSfYsZEcC5J8KCR3skqaUUp2IJilcb3+/u11ZdwxJ/Q6wHWjd5f1+4H+NMS+KyBPAbcDjJ/gecavWW0tlS2Xb46pmL2+ULGZT7RoSmubSUD6XW2dlMHG4CwkHcHmrGJuVxghnAo6ld0P5Jpj0ZTj7toPNQ65MyCjUhKCU6rJoJq91+zpHIjIUuBT4FfA9ERGsTXtusJ/yLHAvJ2lSaAm2HDJbucbt57XPXuFfFUtxeafRXDWXb8zNYkxhIhIJkeot49TcXIqaymHFf4Ov0RpdNHae9QLOJGtkUUp2jGqklDpZxGo/hd8CPwRah8TkAQ3G2MNv4AAw5AReP24Fw0H2Nu1te9zQEmTV3tWsKltCZmgqpXuu5JbzMxhTmAjGkNhSwWmpLoZ+tAi2vWbNM7jyEcg/xXqB1HzIGqqdyUqpbtHr+ymIyGVAlTFmg4jM7sK/XwgsBBg2bFhXQoipCk8FEXsQVyhs+HfpOl7d/SfSImPtTuU0Jg23RhkleSs43elj6PL/Zy1X8bkrrO0yXRnWPsnZwyA5s7O3U0qp4xKL/RRmAFeIyCVYSSYT+B2QLSIJ9t3CUKC0g3ieBJ4EmDJlijnac+JVc6CZpmATYHUsf1j6Cc/tfJTMhCHs/+RG5o7P4MIJ1vpEqeEmxidFGLTiPvDWw+W/g8IJ1gslpkLeGHA4Y1UVpdRJqiuroZ3QfgrGmLuNMUONMSOA64CVxpgbgVXAAvtpNwOvd/U94pEv5Ds4Y9nArtoK/rT9EVId2VR8eisjctO5eGIqIpDr9DLO1DBo+d3groJ5vzyYEFLzNSEopXpMPO2n8CPgRRH5JbAReLoH3iNmqlqq2pqN9tTX8PvN9+MLefHs+Q+yktP42uczSXAK+Qk+RkfcDFz9kNWhfNlDUDDeXuI6T0cWKaV6VEz3UzDGvAO8Yx+XAFO743XjjSfoaVv5tN7bzMObHqDBX0ta4224A4P4xuVZpCc7yHa08DlPGTmrH4T63fYdwhmQO0p3Q1NK9YoOk4KIjAEKWvdTaFc+Q0RcxpjPejy6k0SFxxp+GgqHeHD9Q1R4SxnDbWwoHc6XZ6aTn+EkIxGmVW7E9c791kJ2F90Ho86HrCJNCEqpXtNZn8JvgaajlDfZ51QUGv2N+MI+AJ7Zuojixu2clX4jG7aN4txTkjl7VDJOcTClaSeulb+C3JFwzR+thJA3VhOCUqpXddZ8VGCM2Xp4oTFmq4iM6LGITjK1XmuP5bdKlrHywFIm5ZzP2g2nU5Tn5AtnpyHAWUkBMt+4x9r45tIHIWeUtYeydiYrpXpZZ0mhs+mxKd0dyMmoKdCEN+xlf1Mpi7Y/x9jMCezdeQmhCHxlZjqJTmFqgp/CN++0OpUv/Q0MnQrOxFiHrpTqpzprPlovIl87vFBEbsdasE51whhDlacKYwyPb3qSJIeLPN917Ksx3HBuOgVZCYwKeyh869tQvw/m/xpOu1wTglIqpjq7U7gTeFVEbuRgEpgCJAFX93RgfZ0n6MEf8fNm8QpKmnZwXv6NLHvfyaThSUwalkRhJMDpW5+Chr0w/36YfJMON1VKxVyHScEYUwmcKyJzgPF28T+MMSt7JbI+zBhDra+WiuY6Xi5+gaGpY3h/wwSyUh18cWoaeb4Gzip+AcfON+HMr8BZX9WEoJSKC9Esc7EKa7axilJToIkGXzPPfbIIf9iHu/QqWvxw5/xMRjv8TNr4GAn73oUzvgQX/UoTglIqbugu7j2g1lfLR+Xb+KjmA3KCc9lfmc9NMzM4MyPIpPUPkHzgA5j6dZj5XUjOinW4SinVpitrH6lOuANuGn0elux+lURJZ99n53H5malcMiyZCesfshLCtDtg+jchc1Csw1VKqUPonUI38gQ97G/ez7rSbexq/JhA9YWcVpjOJaenM/7dH5BatcnqUJ7+TWsPBKWUijOaFLpJIBzgQPMBals8vLDzj0gom0TPDG65MouJny4mrWojzPyetR9C5uBYh6uUUkelzUfdpMxdhi8U5K2S5dT4K/CUXcX10wcymgCF256FkbNg5p2aEJRScU3vFLpBna8OT9DDzupylu59lbBnFGcMmMD04flMWP1dRAQueVA7lZVScU/vFE5QxESoaqmi2tPCn7Y/RiAcIqlxAddNy2bSJ0/hKl0Ds++CAafEOlSllDomTQonqM5XRzgS5rlP/sR+TzG+8gXcNG0kp9euI3fL03DaZTDjzliHqZRSUdGkcALCkTD1vno+LNvMuup/EaidycVjpnNWWg3j1t4H+afAF54CkViHqpRSUdGkcAKqvdU0+1p4acdiTDiZEQmXMW9MkHP/9WMcDgdcuwiSUmMdplJKRU2TQhf5Qj5qvXX8bec/KG35jEDl5XzpnBzO3fI4ie4Ka6OcAafGOkyllDoumhS6qMZbQ2lDHW/tfZWQ+1SmFJzLnIrXyd2zEqbfAWPnxTpEpZQ6bpoUusAX8tHga2TJ7r8TND6k/iJuH7abkZv/CCNnwwU/j3WISinVJTpPoQvcQTe7aktZXbaMYONEFpw6iFkbvo2kDYQFT4NTf6xKqb5J/3odp2AkSKWnmkXb/0I4IgzhCha2/Bmntw5uXQZp+bEOUSmluqzXm49EpEhEVonINhH5RES+Y5fnishyEdllf8/p7dii0ehvZM2BTexs2kCkYRbfH1NN0e63kTO/DMPOiXV4Sil1QmLRpxACvm+MGQdMA+4QkXHAXcAKY8xYYIX9OK5ETITd9eU8s/U5IsEsLi+axexPn0DS8q3NcpRSqo/r9aRgjCk3xnxkHzcD24EhwJXAs/bTngWu6u3YjqXWW8/ftq2kOXKAvOClLPS9RmrTPph7r65rpJQ6KcR09JGIjADOBNYCBcaYcvtUBVDQwb9ZKCLrRWR9dXV1r8QJ1r7Ln1YdYOm+1zD+Qu4e4mJU8Rsw5kKYdGOvxaGUUj0pZklBRNKBvwF3GmOa2p8zxhjAHO3fGWOeNMZMMcZMGTBgQC9EailrruHhNUsIO2uYnnUR5+/8A2QUwmW/1WUslFInjZgkBRFJxEoIzxtjXrGLK0VkkH1+EFAVi9iOxhjDko93sM3zFq7wCO5tepsUTxUy9xeQrTuoKaVOHrEYfSTA08B2Y8xD7U69AdxsH98MvN7bsXWkwl3PkxuW4Ehs4uv5pzGkdI012uiML8U6NKWU6laxmKcwA/gKsFVENtllPwZ+DSwWkduAvUDc/MV9Zs1mAmkrGZwwmpu2v4zJKkLm/UqbjZRSJ51eTwrGmPeAjv6aXtCbsUSj0l3PX3e8gmT4uc9TTZKvEbl2ESRnxjo0pZTqdrr20TH8fvX7hDJWM83kcVbZx8jsu2DEjFiHpZRSPUKTQifKGxt5Y89fcRrhvgObCA+fAed9P9ZhKaVUj9Gk0In7V71DJHUj17f4yMochvOGl8DhjHVYSinVY3RBvA7Ut3hYUfYSSSnCwtpKHNe+AK6MWIellFI9Su8UOnDP8sWQtpXbm5rIHD4T56nzYx2SUkr1OE0KR9HkDfBO5UsMDCZwe0MjCRc/EOuQlFKqV2hSOIpfr1oGrlIWNlbhnHYHDPxcrENSSqleoUnhMC3+EG+WLCY1Yrg4qRDnBT+NdUhKKdVrtKP5ME9+sI5I+mauanKTefnjkOCKdUhKKdVr9E6hHWMMf/v0cRKIcHPumTDyvFiHpJRSvUqTQjsvbPiYppSNXOFuYfCl/xvrcJRSqtdpUrAZY3hx7cNEBG4omg85I2IdklJK9TpNCrZl28podK1lqi/IKfN0v2WlVP+kHc22V9//A3Uuw5cGz4OUnFiHo5RSMaF3CkBxlZvS8DIKQmEumH1vrMNRSqmY0aQAPLbybfamtnB1wkASUnNjHY5SSsVMv08K3kCY0srHSTCGa2f+LNbhKKVUTPX7pPDShx+zO7OUOeFU8kfH3cZvSinVq/p9Uli96Zd4HcJtk78R61CUUirm+nVS2FvXzLbUTznHD6dP+mqsw1FKqZjr10NSn//gZTxOuDxrGojEOhyllIq5fnunYIxh295FuCIRLpyl+y4rpRT046SwpriEHamVzApnkDrgtFiHo5RScSHukoKIzBeRHSJSLCJ39dT7vPLeL/E5hC9PuL2n3kIppfqcuEoKIuIEHgUuBsYB14vIuO5+H4/Pz6bwOk73w+Szb+3ul1dKqT4rrpICMBUoNsaUGGMCwIvAld39Js+tfJjKRGH+gHngiLcfgVJKxU68/UUcAuxv9/iAXdZGRBaKyHoRWV9dXd2lNzm3cBTX+jK4bt5Puh6pUkqdhPrckFRjzJPAkwBTpkwxXXmNiZOvYeLka7o1LqWUOhnE251CKVDU7vFQu0wppVQviLeksA4YKyIjRSQJuA54I8YxKaVUvxFXzUfGmJCI/CewDHACzxhjPolxWEop1W/EVVIAMMa8CbwZ6ziUUqo/irfmI6WUUjGkSUEppVQbTQpKKaXaaFJQSinVRozp0vyvuCAi1cDeLv7zfKCmG8OJJa1LfDpZ6nKy1AO0Lq2GG2MGHO1En04KJ0JE1htjpsQ6ju6gdYlPJ0tdTpZ6gNYlGtp8pJRSqo0mBaWUUm36c1J4MtYBdCOtS3w6WepystQDtC7H1G/7FJRSSh2pP98pKKWUOowmBaWUUm36ZVIQkfkiskNEikXkrljHEw0R2SMiW0Vkk4ist8tyRWS5iOyyv+fY5SIiD9v12yIik2MY9zMiUiUiH7crO+64ReRm+/m7ROTmOKrLvSJSal+XTSJySbtzd9t12SEiF7Urj/nnT0SKRGSViGwTkU9E5Dt2eZ+6Np3Uo89dFxFJFpEPRWSzXZdf2OUjRWStHddL9rYCiIjLflxsnx9xrDpGxRjTr76wluT+DBgFJAGbgXGxjiuKuPcA+YeVPQDcZR/fBdxvH18CvAUIMA1YG8O4ZwGTgY+7GjeQC5TY33Ps45w4qcu9wH8d5bnj7M+WCxhpf+ac8fL5AwYBk+3jDGCnHXOfujad1KPPXRf7Z5tuHycCa+2f9WLgOrv8CeAb9vE3gSfs4+uAlzqrY7Rx9Mc7halAsTGmxBgTAF4EroxxTF11JfCsffwscFW78ueM5d9AtogMikWAxpjVQN1hxccb90XAcmNMnTGmHlgOzO/56A/VQV06ciXwojHGb4zZDRRjffbi4vNnjCk3xnxkHzcD27H2Q+9T16aTenQkbq+L/bN12w8T7S8DfB542S4//Jq0XquXgQtEROi4jlHpj0lhCLC/3eMDdP4hihcG+KeIbBCRhXZZgTGm3D6uAArs43iv4/HGHe/1+U+7SeWZ1uYW+lBd7GaHM7H+Z9pnr81h9YA+eF1ExCkim4AqrAT7GdBgjAkdJa62mO3zjUAeJ1iX/pgU+qqZxpjJwMXAHSIyq/1JY9039rnxxX017nYeB0YDk4By4MHYhnN8RCQd+BtwpzGmqf25vnRtjlKPPnldjDFhY8wkrP3ppwKn9XYM/TEplAJF7R4PtcvimjGm1P5eBbyK9YGpbG0Wsr9X2U+P9zoeb9xxWx9jTKX9ixwBnuLgbXrc10VEErH+kD5vjHnFLu5z1+Zo9ejL1wXAGNMArAKmYzXVte6S2T6utpjt81lALSdYl/6YFNYBY+0e/SSsDpo3YhxTp0QkTUQyWo+BecDHWHG3jva4GXjdPn4DuMkeMTINaGzXJBAPjjfuZcA8EcmxmwHm2WUxd1hfzdVY1wWsulxnjxAZCYwFPiROPn922/PTwHZjzEPtTvWpa9NRPfridRGRASKSbR+nABdi9ZGsAhbYTzv8mrReqwXASvvurqM6Rqc3e9fj5QtrJMVOrPa6n8Q6nijiHYU1mmAz8ElrzFjthyuAXcDbQK45OIrhUbt+W4EpMYz9Bazb9yBW2+ZtXYkbuBWrw6wYuCWO6vJ/dqxb7F/GQe2e/xO7LjuAi+Pp8wfMxGoa2gJssr8u6WvXppN69LnrApwBbLRj/hi4xy4fhfVHvRj4K+Cyy5Ptx8X2+VHHqmM0X7rMhVJKqTb9sflIKaVUBzQpKKWUaqNJQSmlVBtNCkoppdpoUlBKKdVGk4JShxGRsL2y5scisqR17HgPvt9XReSRnnwPpaKlSUGpI3mNMZOMMeOxFsC7I9YBKdVbNCko1bk12IuJicgkEfm3vcjaq3Jwr4F3RGSKfZwvInvs46+KyCsislSsvQYeaH1REblFRHaKyIfAjHblX7TvUDaLyOperKdSgCYFpTokIk7gAg4ud/Ac8CNjzBlYs2V/HsXLTAKuBSYA14q1Kcwg4BdYyWAm1vr3re4BLjLGTASu6JaKKHUcNCkodaQUe/ni1qWjl4tIFpBtjHnXfs6zWJvuHMsKY0yjMcYHbAOGA+cA7xhjqo21dv9L7Z7/PvBnEfka1sYvSvUqTQpKHclrrOWLh2Ot+XOsPoUQB3+Xkg875293HAYS6IQx5j+An2KtcrlBRPKiDVqp7qBJQakOGGNagG8D3wc8QL2InGef/grQetewBzjLPl7Asa0FzheRPHvZ5y+2nhCR0caYtcaYe4BqDl0CWake1+n/WpTq74wxG0VkC3A91jLFT4hIKtZexLfYT/sNsNjeEe8fUbxmuYjci9WJ3YC1smer/xGRsVh3KCuwVsZVqtfoKqlKKaXaaPORUkqpNpoUlFJKtdGkoJRSqo0mBaWUUm00KSillGqjSUEppVQbTQpKKaXa/H+ovSb9MXyjHgAAAABJRU5ErkJggg==\n"
          },
          "metadata": {
            "needs_background": "light"
          }
        }
      ],
      "source": [
        "# logFGTS = torch.load(\"miss-zeta=%.2f.pt\" % zeta)\n",
        "\n",
        "def plot(log, label):\n",
        "    mean = torch.mean(log, dim=0)\n",
        "    std = torch.std(log, dim=0)\n",
        "    plt.plot(range(log.shape[1]), mean, label=label)\n",
        "    plt.fill_between(range(log.shape[1]), mean-std, mean+std, alpha=0.2)\n",
        "\n",
        "plot(torch.cumsum(logFGTS, dim=1), \"FGTS.LP\")\n",
        "plot(torch.cumsum(logUCB, dim=1), \"LinUCB\")\n",
        "plot(torch.cumsum(logTS, dim=1), \"LinTS\")\n",
        "plt.xlabel(\"Rounds\")\n",
        "plt.ylabel(\"Cumulative Regret\")\n",
        "plt.legend()\n",
        "plt.savefig(\"miss-zeta=%.2f.pdf\" % zeta, bbox_inches=\"tight\")"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 45,
      "metadata": {
        "id": "5nNWflXCk1-x",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 17
        },
        "outputId": "631be511-45d4-4fe4-f6c5-d40d791bdc63"
      },
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<IPython.core.display.Javascript object>"
            ],
            "application/javascript": [
              "\n",
              "    async function download(id, filename, size) {\n",
              "      if (!google.colab.kernel.accessAllowed) {\n",
              "        return;\n",
              "      }\n",
              "      const div = document.createElement('div');\n",
              "      const label = document.createElement('label');\n",
              "      label.textContent = `Downloading \"${filename}\": `;\n",
              "      div.appendChild(label);\n",
              "      const progress = document.createElement('progress');\n",
              "      progress.max = size;\n",
              "      div.appendChild(progress);\n",
              "      document.body.appendChild(div);\n",
              "\n",
              "      const buffers = [];\n",
              "      let downloaded = 0;\n",
              "\n",
              "      const channel = await google.colab.kernel.comms.open(id);\n",
              "      // Send a message to notify the kernel that we're ready.\n",
              "      channel.send({})\n",
              "\n",
              "      for await (const message of channel.messages) {\n",
              "        // Send a message to notify the kernel that we're ready.\n",
              "        channel.send({})\n",
              "        if (message.buffers) {\n",
              "          for (const buffer of message.buffers) {\n",
              "            buffers.push(buffer);\n",
              "            downloaded += buffer.byteLength;\n",
              "            progress.value = downloaded;\n",
              "          }\n",
              "        }\n",
              "      }\n",
              "      const blob = new Blob(buffers, {type: 'application/binary'});\n",
              "      const a = document.createElement('a');\n",
              "      a.href = window.URL.createObjectURL(blob);\n",
              "      a.download = filename;\n",
              "      div.appendChild(a);\n",
              "      a.click();\n",
              "      div.remove();\n",
              "    }\n",
              "  "
            ]
          },
          "metadata": {}
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<IPython.core.display.Javascript object>"
            ],
            "application/javascript": [
              "download(\"download_a8d103e3-a58b-45cb-817e-cd6da6628cdc\", \"miss-zeta=0.00.pdf\", 185627)"
            ]
          },
          "metadata": {}
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<IPython.core.display.Javascript object>"
            ],
            "application/javascript": [
              "\n",
              "    async function download(id, filename, size) {\n",
              "      if (!google.colab.kernel.accessAllowed) {\n",
              "        return;\n",
              "      }\n",
              "      const div = document.createElement('div');\n",
              "      const label = document.createElement('label');\n",
              "      label.textContent = `Downloading \"${filename}\": `;\n",
              "      div.appendChild(label);\n",
              "      const progress = document.createElement('progress');\n",
              "      progress.max = size;\n",
              "      div.appendChild(progress);\n",
              "      document.body.appendChild(div);\n",
              "\n",
              "      const buffers = [];\n",
              "      let downloaded = 0;\n",
              "\n",
              "      const channel = await google.colab.kernel.comms.open(id);\n",
              "      // Send a message to notify the kernel that we're ready.\n",
              "      channel.send({})\n",
              "\n",
              "      for await (const message of channel.messages) {\n",
              "        // Send a message to notify the kernel that we're ready.\n",
              "        channel.send({})\n",
              "        if (message.buffers) {\n",
              "          for (const buffer of message.buffers) {\n",
              "            buffers.push(buffer);\n",
              "            downloaded += buffer.byteLength;\n",
              "            progress.value = downloaded;\n",
              "          }\n",
              "        }\n",
              "      }\n",
              "      const blob = new Blob(buffers, {type: 'application/binary'});\n",
              "      const a = document.createElement('a');\n",
              "      a.href = window.URL.createObjectURL(blob);\n",
              "      a.download = filename;\n",
              "      div.appendChild(a);\n",
              "      a.click();\n",
              "      div.remove();\n",
              "    }\n",
              "  "
            ]
          },
          "metadata": {}
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<IPython.core.display.Javascript object>"
            ],
            "application/javascript": [
              "download(\"download_1978b7ac-6c3f-4801-a200-092771978c98\", \"miss-zeta=0.00.pt\", 60715)"
            ]
          },
          "metadata": {}
        }
      ],
      "source": [
        "from google.colab import files\n",
        "files.download(\"miss-zeta=%.2f.pdf\" % zeta)\n",
        "files.download(\"miss-zeta=%.2f.pt\" % zeta)"
      ]
    }
  ],
  "metadata": {
    "colab": {
      "collapsed_sections": [],
      "provenance": []
    },
    "gpuClass": "standard",
    "kernelspec": {
      "display_name": "Python 3",
      "name": "python3"
    },
    "language_info": {
      "name": "python"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}