{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Evaluations and Methods for Explanation through Robustness Analysis"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "###  Setting Config"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import sys\n",
    "import math\n",
    "import json\n",
    "import argparse\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torchvision.datasets as datasets\n",
    "import torchvision.transforms as transforms\n",
    "from pgd import pgd\n",
    "from expl_utils import *\n",
    "from bbmp import bbmp\n",
    "import matplotlib.pyplot as plt\n",
    "from skimage import feature, transform\n",
    "\n",
    "\n",
    "def parse_args():\n",
    "    parser = argparse.ArgumentParser(description='Example Usage on MNIST')\n",
    "    parser.add_argument('--gpu', type=int, default=0)\n",
    "    parser.add_argument('--seed', type=int, default=1)\n",
    "    parser.add_argument('--norm', type=int, default=2)\n",
    "    parser.add_argument('--pgd-eps-start', type=float, default=10.)     \n",
    "    parser.add_argument('--pgd-niter', type=int, default=100)\n",
    "    parser.add_argument('--pgd-step-size', type=float, default=1.)\n",
    "    args, _ = parser.parse_known_args()\n",
    "    return args\n",
    "\n",
    "# setting up config\n",
    "args = parse_args()\n",
    "np.random.seed(args.seed)\n",
    "torch.manual_seed(args.seed)\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "torch.cuda.set_device(args.gpu)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Preparation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Test Accuracy: 9895/10000 (99%)\n",
      "\n"
     ]
    }
   ],
   "source": [
    "class Net(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(Net, self).__init__()\n",
    "        self.conv1 = nn.Conv2d(1, 20, 5, 1)\n",
    "        self.conv2 = nn.Conv2d(20, 50, 5, 1)\n",
    "        self.fc1 = nn.Linear(4*4*50, 500)\n",
    "        self.fc2 = nn.Linear(500, 10)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = F.relu(self.conv1(x))\n",
    "        x = F.max_pool2d(x, 2, 2)\n",
    "        x = F.relu(self.conv2(x))\n",
    "        x = F.max_pool2d(x, 2, 2)\n",
    "        x = x.view(-1, 4*4*50)\n",
    "        x = F.relu(self.fc1(x))\n",
    "        x = self.fc2(x)\n",
    "        return x\n",
    "\n",
    "\n",
    "class UnnormInputModel(nn.Module):\n",
    "    def __init__(self, model, mean, std):\n",
    "        super(UnnormInputModel, self).__init__()\n",
    "        self.model = model\n",
    "        self.mean = mean\n",
    "        self.std = std\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x_normed = (x - self.mean) / self.std\n",
    "        return self.model(x_normed)\n",
    "    \n",
    "\n",
    "# prepare model\n",
    "model = Net()\n",
    "model.load_state_dict(torch.load('mnist_cnn.pt')) # load pretrained model\n",
    "model = UnnormInputModel(model, mean=0.1307, std=0.3081)  # wrap the model with a normalization layer\n",
    "model = model.to(device)\n",
    "model.eval()\n",
    "\n",
    "# model sanity check\n",
    "test_loader = torch.utils.data.DataLoader(\n",
    "    datasets.MNIST('./mnist', train=False, transform=transforms.ToTensor()),\n",
    "    batch_size=512, shuffle=True)\n",
    "\n",
    "correct = 0\n",
    "with torch.no_grad():\n",
    "    for data, target in test_loader:\n",
    "        data, target = data.to(device), target.to(device)\n",
    "        output = model(data)\n",
    "        pred = output.argmax(dim=1, keepdim=True) # get the index of the max log-probability\n",
    "        correct += pred.eq(target.view_as(pred)).sum().item()\n",
    "\n",
    "print('\\nTest Accuracy: {}/{} ({:.0f}%)\\n'.format(\n",
    "    correct, len(test_loader.dataset),\n",
    "    100. * correct / len(test_loader.dataset)))\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Example Input"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Predicted Class: 4\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAPsAAAD4CAYAAAAq5pAIAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADl0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uIDMuMC4xLCBodHRwOi8vbWF0cGxvdGxpYi5vcmcvDW2N/gAABZtJREFUeJzt3L9LVX8cx3FPhv2ACMKphgQb9D/IJWoSmoyWokmcHHI0nARpa2mvxpr6C9oKh9Bos8UcbAhycalBCU/b94t89XPz+j3n6n09HuN9cz/nLE/f0Cet6roeAPrfmV6/ANAOsUMIsUMIsUMIsUOIs20+rKoq//QPDavrujroc5sdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQpzt9QvQ2fDwcHH+7t27Q2fj4+PF7965c6c4//jxY3HO6WGzQwixQwixQwixQwixQwixQwixQ4iqruv2HlZV7T2sj4yNjRXna2trXZ+9u7tbnF+4cKHrszsZHR0tzn/+/Fmc//jx4/98nb5R13V10Oc2O4QQO4QQO4QQO4QQO4QQO4TwK67hhoaGevbsN2/eFOcjIyPF+f3794vz5eXlo75SX7PZIYTYIYTYIYTYIYTYIYTYIYTYIYR7dnrm0qVLxXmnP6H95MmT4tw9+342O4QQO4QQO4QQO4QQO4QQO4QQO4Rwz86p1ekenv1sdgghdgghdgghdgghdgghdgghdgjhnv0UmJmZaezsV69eNXb2wED5Lvz8+fONPpv9bHYIIXYIIXYIIXYIIXYIIXYIIXYI4Z79BLhx40Zx/uDBg67P3tvbK85XV1e7Pvtv3L1799DZ9evXj3X269evj/X9NDY7hBA7hBA7hBA7hBA7hBA7hHD1dgJcuXKlOL969WrXZ//69as4f/HiRddn/43Lly83dvba2lpjZ/cjmx1CiB1CiB1CiB1CiB1CiB1CiB1CuGc/AYaGhho7e2dnp7Gz/8b8/HxPn8+/bHYIIXYIIXYIIXYIIXYIIXYIIXYI4Z79BFhcXGzs7KWlpcbO5nSx2SGE2CGE2CGE2CGE2CGE2CGE2CGEe/YToKqq4vzMmfLP5Lquu3727du3i/OJiYni/ObNm8X5tWvXjvpK/9jc3CzOv3371vXZiWx2CCF2CCF2CCF2CCF2CCF2CCF2COGevQULCwvF+a1bt4rzvb29rp/97Nmz4rzTHf+5c+eK8053/Mf5PwDr6+vF+cbGRtdnJ7LZIYTYIYTYIYTYIYTYIYTYIYSrtxZMTU0V54ODg409u9PV2Un2+fPnXr9CX7HZIYTYIYTYIYTYIYTYIYTYIYTYIYR79j73+/fv4nxubq44f//+fXH+9OnT4vzevXuHzr5//1787suXL4tzjsZmhxBihxBihxBihxBihxBihxBihxDu2VswPT1dnD9+/Lg4Hx0dLc7fvn176OzTp0/F73b6nfGRkZHifHJysjgvWVlZKc6/fv3a9dn8l80OIcQOIcQOIcQOIcQOIcQOIcQOIdyzt+DLly/F+ezsbEtvcnQPHz4szi9evNjSm3BcNjuEEDuEEDuEEDuEEDuEEDuEcPVGz2xtbfX6FaLY7BBC7BBC7BBC7BBC7BBC7BBC7BDCPTs98/z5816/QhSbHUKIHUKIHUKIHUKIHUKIHUKIHUK4Z6dRHz58OHTm99nbZbNDCLFDCLFDCLFDCLFDCLFDCLFDCPfsNOrRo0eHzra3t1t8E2x2CCF2CCF2CCF2CCF2CCF2CCF2CFHVdd3ew6qqvYdBqLquq4M+t9khhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghhNghRKt/ShroHZsdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQogdQvwBa36iPicySc0AAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "test_loader = torch.utils.data.DataLoader(\n",
    "    datasets.MNIST('./mnist', train=False, transform=transforms.ToTensor()),\n",
    "    batch_size=1, shuffle=True)\n",
    "\n",
    "Xs, ys = next(iter(test_loader))\n",
    "Xs_np = Xs.numpy()\n",
    "plt.imshow(Xs_np.reshape(28, 28), cmap='gray')\n",
    "plt.axis('off')\n",
    "\n",
    "Xs = Xs.to(device)\n",
    "ys = ys.to(device).long()\n",
    "assert Xs.shape == (1, 1, 28, 28)\n",
    "assert ys.shape == (1,)\n",
    "\n",
    "ys_pred = model(Xs).argmax()\n",
    "print('Predicted Class:', ys_pred.item())"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Generating Explanations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Computing explanation by Grad...\n",
      "Computing explanation by IG...\n",
      "Computing explanation by LOO...\n",
      "Computing explanation by BBMP...\n",
      "Computing explanation by SHAP...\n",
      "Computing explanation by Greedy-AS...\n",
      "number of relevant features selected: 0.0\n",
      "number of relevant features selected: 39.0\n",
      "number of relevant features selected: 78.0\n",
      "number of relevant features selected: 117.0\n",
      "number of relevant features selected: 156.0\n",
      "number of relevant features selected: 195.0\n",
      "number of relevant features selected: 234.0\n",
      "number of relevant features selected: 273.0\n",
      "number of relevant features selected: 312.0\n",
      "number of relevant features selected: 351.0\n",
      "number of relevant features selected: 390.0\n"
     ]
    }
   ],
   "source": [
    "# Grad\n",
    "expl_grad = saliency_expl('Grad', Xs, ys, model)\n",
    "# IG\n",
    "expl_ig = saliency_expl('IG', Xs, ys, model)\n",
    "# LOO\n",
    "expl_loo = saliency_expl('LOO', Xs, ys, model)\n",
    "# BBMP\n",
    "expl_bbmp = bbmp(Xs, ys, model, device)\n",
    "# SHAP\n",
    "expl_shap = saliency_expl('SHAP', Xs, ys, model)\n",
    "# Greedy-AS\n",
    "expl_greedy_as = greedy_as(Xs, ys, model, targeted=False, target=None, norm=args.norm,\n",
    "                           eps_start=args.pgd_eps_start, step_size=args.pgd_step_size, niters=args.pgd_niter,\n",
    "                           device=device, reverse_anchor=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def visualize(Xs, expls, percentage=15):\n",
    "    \n",
    "    def select_top_k(expl):\n",
    "        expl = np.copy(expl)\n",
    "        expl += np.random.uniform(-1e-12, 1e-12, size=expl.shape) # for ranking stability\n",
    "        threshold = np.percentile(expl, 100-percentage)\n",
    "        top_k_expl = np.copy(expl)\n",
    "        top_k_expl[expl <= threshold] = 0\n",
    "        top_k_expl[expl > threshold] = 1\n",
    "        return top_k_expl\n",
    "    \n",
    "    # compute edge for overlay\n",
    "    dx, dy = 0.05, 0.05\n",
    "    xx = np.arange(0.0, Xs.shape[1], dx)\n",
    "    yy = np.arange(0.0, Xs.shape[0], dy)\n",
    "    xmin, xmax, ymin, ymax = np.amin(xx), np.amax(xx), np.amin(yy), np.amax(yy)\n",
    "    extent = xmin, xmax, ymin, ymax    \n",
    "\n",
    "    Xs = Xs.cpu().reshape(28, 28)\n",
    "\n",
    "    xi = Xs\n",
    "    alpha = 0.8\n",
    "    dilation=3.0\n",
    "    cmap_xi = plt.get_cmap('Greys_r')\n",
    "    cmap_xi.set_bad(alpha=0)\n",
    "    overlay = None\n",
    "    # Compute edges (to overlay to heatmaps later)\n",
    "    xi_greyscale = xi if len(xi.shape) == 2 else np.mean(xi, axis=-1)\n",
    "    in_image_upscaled = transform.rescale(xi_greyscale, dilation, mode='constant')\n",
    "    edges = feature.canny(in_image_upscaled).astype(float)\n",
    "    edges[edges < 0.5] = np.nan\n",
    "    edges[:5, :] = np.nan\n",
    "    edges[-5:, :] = np.nan\n",
    "    edges[:, :5] = np.nan\n",
    "    edges[:, -5:] = np.nan\n",
    "    overlay = edges\n",
    " \n",
    "    # visualize top-k relevant features\n",
    "    top_k_expls = [select_top_k(expl).reshape(28, 28) for expl in expls]\n",
    "        \n",
    "    n_cols = len(expls) + 1\n",
    "    n_rows = 1\n",
    "    fig, axes = plt.subplots(nrows=n_rows, ncols=n_cols, figsize=(3*n_cols, 3*n_rows))\n",
    "\n",
    "    axes[0].imshow(Xs, cmap='gray')\n",
    "    axes[0].axis('off')\n",
    "    for i, expl in enumerate(top_k_expls, 1):\n",
    "        axes[i].imshow(expl, extent=extent, cmap='Reds', vmin=0, vmax=1)\n",
    "        axes[i].imshow(overlay, extent=extent, cmap=cmap_xi, alpha=alpha)\n",
    "        axes[i].axis('off')\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/usr/local/lib/python3.5/dist-packages/skimage/transform/_warps.py:24: UserWarning: The default multichannel argument (None) is deprecated.  Please specify either True or False explicitly.  multichannel will default to False starting with release 0.16.\n",
      "  warn('The default multichannel argument (None) is deprecated.  Please '\n",
      "/usr/local/lib/python3.5/dist-packages/skimage/transform/_warps.py:110: UserWarning: Anti-aliasing will be enabled by default in skimage 0.15 to avoid aliasing artifacts when down-sampling images.\n",
      "  warn(\"Anti-aliasing will be enabled by default in skimage 0.15 to \"\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1512x216 with 7 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "visualize(Xs, [expl_grad, expl_ig, expl_loo, expl_bbmp, expl_shap, expl_greedy_as])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Evaluating Explanations through Robustness Analysis"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def eval_curve_robust(Xs, ys, model, targeted, targets, norm, eps_start, step_size, niters, expl,\n",
    "                      device, reverse_anchor):\n",
    "    # set up logger\n",
    "    logger = list()\n",
    "    \n",
    "    # for topk stability\n",
    "    expl = np.copy(expl)\n",
    "    expl += np.random.uniform(-1e-12, 1e-12, size=expl.shape)\n",
    "    \n",
    "    for i in range(0, 50, 5):\n",
    "        threshold = np.percentile(expl, 100 - i)\n",
    "        top_k_expl = np.copy(expl)\n",
    "        top_k_expl[expl <= threshold] = 0\n",
    "        top_k_expl[expl > threshold] = 1\n",
    "        anchor_map = top_k_expl\n",
    "        if reverse_anchor:\n",
    "            anchor_map = np.logical_not(anchor_map)\n",
    "        anchor_map = torch.BoolTensor(anchor_map).to(device)\n",
    "        \n",
    "        def binary_search_cond(Xs, ys, epsilons, anchor_map, targeted, targets):\n",
    "            epsilons = torch.FloatTensor(epsilons).to(device)\n",
    "            successes, Xs_pgd = pgd(Xs, ys, model, epsilons, norm, niters, step_size, anchor_map, targeted, targets=targets,\n",
    "                                    box_min=0., box_max=1., verbose=False, multi_start=True, device=device)\n",
    "            return successes, Xs_pgd\n",
    "        \n",
    "        robust_ub, Xs_pgd = binary_search(binary_search_cond, Xs, ys, np.full(len(Xs), eps_start), anchor_map, targeted, targets, tol=0.001, max_steps=25)\n",
    "        robust_ub = robust_ub[0]\n",
    "        logger.append(robust_ub)\n",
    "        print('robustness:', robust_ub)\n",
    "\n",
    "    return logger\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "robustness: 1.2032470703125\n",
      "robustness: 1.58062744140625\n",
      "robustness: 1.6427001953125\n",
      "robustness: 1.88165283203125\n",
      "robustness: 2.0491943359375\n",
      "robustness: 2.0843505859375\n",
      "robustness: 2.52105712890625\n",
      "robustness: 2.6435546875\n",
      "robustness: 2.77099609375\n",
      "robustness: 2.79296875\n",
      "robustness: 1.2021484375\n",
      "robustness: 1.7119140625\n",
      "robustness: 2.0426025390625\n",
      "robustness: 2.3271484375\n",
      "robustness: 2.8402099609375\n",
      "robustness: 3.0775146484375\n",
      "robustness: 3.8970947265625\n",
      "robustness: 4.44586181640625\n",
      "robustness: 4.9495849609375\n",
      "robustness: 5.392333984375\n"
     ]
    }
   ],
   "source": [
    "eval_curve_grad = eval_curve_robust(Xs, ys, model, False, None, args.norm, args.pgd_eps_start,\n",
    "                                    args.pgd_step_size, args.pgd_niter, expl_grad, device, False)\n",
    "eval_curve_greedy_as = eval_curve_robust(Xs, ys, model, False, None, args.norm, args.pgd_eps_start,\n",
    "                                        args.pgd_step_size, args.pgd_niter, expl_greedy_as, device, False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0.5, 0, 'Percentage of Relevant Features (|$S_r$|/d)')"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYoAAAEXCAYAAACzhgONAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADl0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uIDMuMC4xLCBodHRwOi8vbWF0cGxvdGxpYi5vcmcvDW2N/gAAIABJREFUeJzs3Xd8VFX6+PHPnZlUkhAgISG0AIJICR0LYKFZVmlSFMSCYAG/628tgF9Fwa8FRWFBV8XCsrvSUVBsYERcRaqChCa9p2FMISSZzNzz+2PIJEPqTKYled6vF68wd+698+QmmWfOuec8R1NKKYQQQohyGHwdgBBCCP8miUIIIUSFJFEIIYSokCQKIYQQFZJEIYQQokKSKIQQQlRIEoUQQogKSaIQQghRIUkUQgghKiSJQgghRIVMvg7AXc6dO+fScVFRUZw/f97N0dTMOPwhBolD4qgJcfhDDNWNIy4ursr7SotCCCFEhSRRCCGEqFCt6XoSQojaTikFxw+hb1gDSb+QWmiGgEDo3BPDzcMhvi2aprn9dSVRCCFEDaAsFvRF8+C37VBohqIVIswF8OvP6Ek7oUtvDBP+hmZy71u7dD0JIYSfU0oVJwlzQXGSKN7Btv23beiL5uHuZYbqRIvi4sWLWK3WMp/Ly8vDYrF4OSL/jMMfYqhOHKGhoRiNRg9EJISPHT9UnCQqYjbb9jtxGFq1c9vL1/pEYTabAQgPD/dxJMKTlFJkZ2cTFhYmyULUOvqGtbbupqooNKNvWIvx4alue/1a3/WUn59PSEiIr8MQHqZpGhEREVy8eNHXoQjhfkk7S3c3lUcp2LPDrS9f6xOFpmkeGQUg/I/8nEWtZa5ia6JIVVsfVVTrE4UQQtR4AQFO7h/o1pev9fcohBCiplK6FfXDerA6MbhD0yChl1vjkERRgcsnt2A2Q6DnJ7f4s7Zt2/LSSy8xZswYX4ciRK2mTh5F//gd2wgmZwQEYhg8zK2xSNdTOZTFgv7BG+hvPge/brk0LE0VT25541n0D95AeXA4aXp6Oi+88AJ9+vShdevWJCQkMGTIEBYtWkRubq7HXtdbFi5cSPPmzZk9e3aZzy9dupTBgwfTtm1b2rdvz8CBA3nttde8HKUQ3qXyLqIv/wD95Scdk0RwCFQ2kS4wELr0hvi2bo1JWhRlKDW5pfQODpNbDJOecnvL4vTp0wwbNoywsDCmTp3KVVddRXBwML///jvLli2jQYMGDB8+vNRxZrOZwED39k96yvLly5kyZQqrVq3i6aefdhjWunz5cmbMmMHMmTPp27cvFouFgwcP8ssvv/gwYiE8Ryll+xC6/APIzCh+whSA9pdRMHAo6t9vl56ZDbbupoDA4pnZbn4/khZFWVyZ3OJmzzzzDJqm8fXXXzN06FDatWtHixYtGDRoEIsWLWLYMFvTsmnTpixevJiJEydyxRVX2D+dHzp0iPHjx9OuXTsSEhKYPHkyaWlpDq+xYsUKbrzxRlq3bk3fvn15//330XXd/vzx48cZOXIkrVu3pl+/fnz77bcOx48aNYpnn33WYVtOTg5t2rThq6++qvD727lzJxkZGTz55JMEBwezceNGh+c3bNjAbbfdxvjx42nVqhVt27bljjvuYObMmU5dRyFqApWegr7gRfT3XnNMEh26Ypj1Fobb78IQHIJh0lMYnnoZul8HgUG2BBEYBD36YHjqFYwPPe328h0giaJMrkxucaeMjAw2bdrE/fffT2hoaJn7lPzEMHfuXPr37893333H/fffT2pqKiNGjKB9+/Z8+eWXLF++nNzcXCZMmGBPBEuWLGH27Nk89dRTbNq0ieeff5533nmHf/3rXwDous7EiRPRdZ3PP/+cuXPnMnfuXPsERoBx48axdu1aCgqKE+ratWupV68egwYNqvB7XLZsGUOHDiUgIIARI0awbNkyh+ejo6PZtWsXJ0+edO7iCVGDKEsh+ler0F94DPaWaC1HRKJNegrD/5uF1rh43QhN09BatcP4yDSM/1hFzKebMf5jFcaHp6K1cm93U0l1ruvJOmmIe0+oFOz8CevOnyrd1fjB51U65YkTJ1BK0aZNG4ftPXr0IDs7G4ARI0bY++uHDBnC2LFj7fvNmTOHDh06OHzanz9/Ph07duS3336jW7du/P3vf+fZZ5/l9ttvB6BFixacPHmSf/3rXzzwwAP8+OOPHDp0iK1bt9K0aVMAZs2a5dDddeutt/Lcc8/xzTffMHToUMDWShk5ciQBFQzny83NZd26daxatQqAO++8k7feeou0tDQaN24MwBNPPMH+/fu57rrriI+Pp3v37lx//fUMGzaswnMLUVOoQ3vRP34Xkk8Xb9Q0tBtvRRt2D1pomO+Cu0ydSxQ12Zo1a7BarUydOtXhU3xCQoLDfnv27GHbtm20bVv6E8bJkydp0aIF586dY9q0aTzzzDP256xWq72Y2OHDh4mNjbUnCYBu3bphMBQ3QoOCgrjzzjtZvnw5Q4cO5ffff2fXrl3MmzcPgAULFvDWW2/Z99+0aRNNmzbls88+o0mTJnTp0gWA+Ph4unTpwqpVq5gyZQoAMTExrFu3joMHD7J161Z27tzJ9OnT+eCDD/jss89ktr2osVROFmr1YtTP3zk+0aI1hnumeLRl4CpJFH4oPj4eTdM4cuSIw/YWLVoAlHqTvLx7SinFgAEDmDFjRqlzR0dHk5eXB8Ds2bPp2bNntWIdO3YsAwcO5OzZsyxfvpwePXrYE9T48eO544477PvGxMQAttFMR48etX8/YOvq+uOPP+yJokj79u1p3749999/P9u3b2f48OF8/vnnMjxX1DhK11GbE1Gf/Atyc4qfCA5BGzoO7aa/oPlpnbI6lyiq0v1jfe81+PXnqtVW0TTo0cetBbgaNmzIDTfcwOLFi5kwYQL16tVz6vhOnTqxbt06mjVrVmY3TVhYGLGxsZw8eZJRo0aVeY62bduSkpLC2bNn7a2K3bt3O9zsBrjyyivp1q0bS5Ys4dNPP2XatGn25xo0aECDBg0c9i9qdSxbtszezQS2mlzDhg1j69atXHPNNeXGBEg9J1HjqLMnbXMijhxwfKLHdRjGTEJr0Mg3gVVRnUsUVWEYPMy2CEhlo57AI5NbAF555RWGDRvGLbfcwpNPPkmHDh0wGo0kJSWxf/9+brjhhnKPvf/++1m6dCmPPvookydPplGjRpw8eZIvvviC559/nrCwMJ588klmzJhBREQE/fv3x2KxkJSUREpKCv/zP/9Dv379uOKKK3j88ceZOXMm+fn5zJw5E1MZIyrGjRvH9OnTMZlMDBlS8T2gpUuX0qlTJ66//vpSz/Xt25elS5dyzTXXMH36dGJiYujbty9NmjQhNTWV+fPnExISUuaxQvgjVZCPWrcclfgZlFzqICoGw9iH0TpXr0XvLTLqqSyt2tkmrVQ2H8FDk1sAWrZsyfr167npppuYM2cON998M7fccgvvv/8+9913H7NmzSr32NjYWNauXYumadxzzz3079+fZ599lsDAQPsci7Fjx/Lmm2/yySefMHjwYEaMGMGSJUvs3UEGg4EPP/wQpRR33HEHjz/+OI8//niZczSGDBlCQEAAd9xxB2Fh5d+AM5vNfPrpp/zlL38p8/nbb7+dL7/8kuzsbK6//np27drFI488Qr9+/Zg4cSJgGy11+U1+IfyR+m0H+guPodZ/WpwkjCa020ZhmPl2jUkSAJpy91JIPnLu3Lkyt+fk5Li0FkW5yw5C6cktHhi3XJOkpKTQu3dvPvnkE3r1cm+NGWe5+vMuS1RUFOfPn3fLuSSO2hVHRTGojHTbpLldWx2faNcRw7hH0eJalHmcu+OoTFxcXOU7XeJ373AzZ87k8OHD9tE1DRs2ZP78+V6PQzOZMEx6Ck4cRl+/xlYPvmgh84ReGAYP98vRCd5UWFjIn3/+yezZs+nUqZPPk4QQnnJ53bfUoveCEnXf0HXUd+tQny+Fgvzig8PC0UZNQLu2f42tDed3iQJgwoQJDBgwwNdh2H6olya3iNJ27NjBqFGjaNWqFe+9956vwxHCI8rtXSiq+5a0E9pcBdmZcPaEw7Fa30Fod96HFhbh/cDdyC8ThagZrrvuOs6ePevrMITwmCrXfTuw23F7XAsM90xGa9vBO4F6mF8miqVLl7J06VLi4uK466676Nixo69DEkLURVWt+1bEZLLNiRg4tFbdu/S7m9mHDx+mWbNmmEwmNm/ezKJFi3j99deJjY112C8xMZHExETANnHMXM5SgadPn5ZZvHVIXl4ezZs3d8u5TCYTFg+WkZc4/D+OzDnPUbDl+yqvVx3Y8zoaPPuGh6MqVp1r4UyVab9LFJd7+eWX6d69O7feemuF+7l71JOomWTUk8ThTtYpo6remgAIDML4j1WeC+gy3hr15PfzKDRNw89zmRCitiqnp6JcVa06XcP4VaLIzc1l9+7dmM1mrFYrP/74IwcOHKBr166+Dk0IURc5uwhYQM1YNMxZfnW3xWq1smLFCs6ePYvBYKBp06Y8/fTTTjWRhBDCHVR+HoTXhz/SKt8ZbBNxE2rnXCK/ShQRERG8+uqrvg6jTBl5Ft746SxP921KgxC/umxu8+yzz/L777+zevVqX4cihE+powfRP5pb9SQBHqv75g9q5zueB6xIOs/+tDxWJJ3nkd6xlR/gBunp6bz99tskJiaSnJxMWFgY8fHxDBs2jDFjxjhdVVYIUTFlsaC+XIn6ciWoEpWSNYPj48t5sO6bP5BEUQUZeRY2HstCAd8dy2JM5yiPtypOnz7NsGHDCAsLY+rUqVx11VUEBwfz+++/s2zZMho0aOCw2lwRs9ns1LA3IYSNSj1na0UcP1S8MSQU7poEe3+tWt23GlqiozKSKKpgRdJ59Eu/HLpSXmlVPPPMM2iaxtdff+2wMFGLFi0YNGiQfSRY06ZNefnll/npp5/YtGkT9957L88++yxTp05l8+bNpKenExsby7hx43jkkUfsNbSsVisvv/wyy5cvB2DkyJFYS5ZBFqKOUEqhftyAWvGh41DYdh1tb/6NGqOu7V+n675JoqhEUWvCcqnVadE936rIyMhg06ZNTJ8+vdTqdUVKfnKZO3cu06dPZ8aMGWiahq7rxMbG8t5779GoUSN27drFtGnTaNCgAXfffTcACxcuZOnSpbz++ut06NCBxYsXs2bNGjp37uyR70kIf6RystD/9ZattVDEaEIbNg5t8DA0g23FucvrvvnLnBJvqXOJYuiSg9U+h9mquP/TI5XveJnPxrWv0n4nTpxAKVVq3YUePXqQnZ0NwIgRI3jttdcA23oQY8eOddj36aeftv+/efPm7N27l7Vr19oTxYcffsijjz5qX2joxRdfZNOmTU5/T0LUVCppJ/riBbZifkWaNMcw8Qm0FrLmSUl1LlHUZGvWrMFqtTJ16lQKCoqbyAkJCaX2/fe//82yZcs4c+YM+fn5WCwW+5Km2dnZpKam0qNHD/v+BoOBbt26kZyc7PlvRAgfUgUFqNX/RG36ymG71v92W6XXwCAfRea/JFH4ofj4eDRN48gRx1ZL0epzl9euurx76rPPPmPmzJnMmDGDnj17EhYWxuLFi/nmm288G7gQfk6dPIL+4ZuQUqLqcf0GGO5/HK1Td98F5ufqXKKoavdPRp6Fhz87itlafvmQQKPG+0PbuP1eRcOGDbnhhhtYvHgxEyZMcHoY7I4dO+jWrRsPPPCAfdvJkyft/4+IiCAmJoZff/2Vvn37ArYbert37yYmJsY934QQfkTpVtTXn6DWLXNcu7rbNRjGP4YWXrPXi/A0vyrh4U9KjnQqT9EIKE945ZVX0HWdW265hbVr13Lo0CGOHj3K2rVr2b9/P0ajsdxjW7duTVJSEhs3buTYsWPMmzePrVsdl2V88MEHeffdd/niiy84cuQIL7zwAmlpTkwuEqKGUOkp6HP+F7X24+IkERSCdv9fMTz6jCSJKqhzLYqquHykU3k8OQKqZcuWrF+/nrfffps5c+Zw7tw5TCYTbdu25b777nNoLVzunnvuYd++fTz22GMopbjtttt4+OGH7UNhAR5++GHS0tLsN73vvPNOhg8fXqq7S4iaSimF2vI9atlCyM8rfqJNe9uw18ZNfBdcDeP3Zcaryp1lxt/dnkLi0cxKEwWAyQCD2kR6bba2qJiUGZc4AFRuDuo/76B+2Vy80WBAu+MutFtHoVXQIndXDN7grTLj0qIow+/n86qUJMDWqjh4Pq/yHYUQXqH270L/53zIzCje2LgJholPorVq57vAajBJFGX4+22tfB2CEMJJqtCM+vTfqMTPHbZr19+MNmoCWrCsdOkqSRRCiBpPnT5uG/Z67lTxxvD6GO77H7QuvX0XWC0hiUII4deUUnD8EPqGNZD0C6lFdZY690QbNBR1ZD+s/RhKrh3duSeG+/8HLaKB7wKvRSRRCCH8lrJY0BfNK1251VwAv/6M+vVnx2qugYFoox5Eu+GWWlvJ1RckUQgh/JJSqjhJlKzqWryD4+MWbTBMehIttpl3AqxDqjThbsyYMZ6Ow2OUUtSSEcCiEvJzrmWOHyo/SVzOaES7+yFJEh7i1pnZSikyMzMr39GLgoODycuT4au1nVKK7Ozscsuyi5pH37DW1t1UpZ111HfrPBtQHebWridN05g1axbz5s1z52mrJTAwEIvFQk5OTpnPm0wmLCVvgvmIP8ThDzFUJ46wsLAKS5uIGiZpZ+nupfIoBXt2eDaeOszt9yhiY2PJy8srVeHUlyr6lFkbZljWphj8KQ7hY+YqtiaKVLX1IZzm9qKAoaGhvPbaaw7VSoUQwmkBTn6ODZC14j3F7S2KmJgYcnNzefnll9F1nTZt2tCmTRtGjx7t7pcSQtRCtjWs1zvOi6iMpkFCL88FVce5PVGUTAgZGRkcO3aMY8eOuftlhBC1kCrIR338Lmrr984dGBCIYfAwzwQlnE8UO3fuJDk5mTvuuKPSfRs2bEjDhg3p2bOnS8EJIeoOlXwG/b3ZjmU4QuqBxQyFheUfGBgIXXpDfFvPB1lHOZ0o1q9fD1AqUWzcuJENGzZQWFhIly5dGDlypAxVFEJUib7jR9S/3oaC4qHsWp8BMGYi6j/vlJ6ZDbbupgBbkjBM+JvMxPYgpxPFqVOnGDVqlMO2gwcPsnDhQgwGA/Xr1+fLL79k3759vPTSSwQEBLgtWCFE7aIshahV/0Rt/KJ4Y0Ag2tiHMfQdZNtn0lNw4jD6+jW2IbNFtZ4SemEYPBytlbQkPM3pRHHhwgWaNHFcGSoxMRFN03jppZdo06YNhw4d4tVXX+Xrr79myJAhbgtWCFF7qD/S0Re+ZpuBXaRxEwyPTEdrXlzqX9M0aNUO4yPTABk+7QtOD48NCgpymOmslGLXrl1cccUVtGnTBoB27dpx88038/PPP7svUiFEraH2/oL+f//PMUl0vxbDs3MdkoTwD04nijZt2rBr1y7740OHDnHhwgUSEhIc9rvyyivLXZ5UCFE3Kd2KvvZj9AUvQu6laglGI9roB20tidB6vg1QlMnprqdbbrmFOXPm0LBhQzp37syqVasA6NGjh8N+QUFBUqRNCGGnsjNtiwsd+K14Y2RDDA9PRbuig+8CE5VyOlH06NGDESNGsHLlSlauXAlAhw4d7N1ORc6cOUNERIR7ohRC1GjqyH70ha87rmN9VRfbOtYRkb4LTFSJSxPuRo8ezXXXXcfevXsJDg7muuuuK7XP5s2biY+Pr258QogaTCmF+nYt6pN/ga7bNmoa2l/GoN0xBs0gRRxrApdnZjdr1oxmzcqu/Z6ZmYnRaKR3b1mrVoi6Sl3MRV88H3ZtLd4YFo7hwSfQOvUo/0Dhd6qUKFasWOHUSSMjI3n++eddCkgIUfOpU8dss6zTU4o3tmqH4eFpaI2ifReYcIlbqsfqus6YMWOkppMQAv3HDeivPu2QJLQBd2CY+qokiRpK1swWQriFKihALXsPtfm74o1BIWj3/Q+GXn19F5ioNrevR+EuycnJjBs3jgULFvg6FCFEJVTqOfTZTzsmibgWGJ57U5JELeC3LYqPPvqo1JBbIYT/Ub9sRl+8APJLFPS75ia0ex5FCwr2YWTCXdySKAwGAy+88AJxcXHuOB2bN28mNDSUdu3akZKSUvkBQgivU5ZC1Cf/QiV+XrzRFIB290No/QZLNddaxOVEoes6Sin7YvYdOnRg9+7dnD59mk6dOtGqlWv1Wi5evMjKlSt5/vnn+e677yo/QAjhEUopOH4IfcMaSPqF1KKqrZ17ol17E+qrVXDs9+IDomJsZThaSk9AbeNyopg/fz4BAQE89thjAGzYsIGPPvoIAKPRyPTp00vVf6qKFStWcNNNN9GoUaMK90tMTCQxMRGA2bNnExUV5fRrAZhMJpePdSd/iMMfYpA4/CMOZbGQPf9F8nf8BOaC4nUgzAXw68+oXzY77B/Uqy8Rf30OQ5jnqzH4w8/FH2LwZhwuJ4rDhw8zbtw4++N169bRv39/7r33XhYuXMiaNWucThQnTpwgKSmJ119/vdJ9Bw4cyMCBA+2PXS077C8li/0hDn+IQeLwfRxKKfQP3rAtFmQuKGsHx8d33kfhzSPIyDdDvufj9Iefiz/EUN04nLlV4HKiyMrKomHDhgCkpKSQlpbGLbfcQkhICDfddJNLo5X27dtHeno6jz76KAD5+fnous60adN47bXXXA1VCOGM44fKTxKXCwjAcGVnuR9Ry7mcKEJDQ8nJsZUJ3rdvH+Hh4bRs2RKw3dw2m81On3PgwIH06dPH/vjzzz8nPT2dSZMmuRqmEMJJ+oa1tlXkqsJiQd+wFuPDUz0blPAplxNFu3bt+OyzzzAajXz11Vd069bN/lxKSkql9xjKEhQURFBQkP1xcHAwAQEBUoVWCG9K2lm6e6k8SsGeHZ6NR/icyxPu7rnnHnJycnj99dcxm80O62j//PPPtGvXrtrBjR49mr/+9a/VPo8QwgnO9gZUtfUhaiyXWxRNmjRhwYIF5OTkEB4e7vDcAw88QGSk1JgXokYKDKza/YkiAYGei0X4hWqX8Lg8SQC0aNFCuouEqKk6O1ECXNMgoZfnYhF+wW0T7gC3TLgTQviYM4sJBQRiGDzMc7EIv+B3E+6EEL6jJ34OO36s2s6BgdClN8S39WxQwudc7no6fPiww0inogl3ixcvpnfv3qxZs8YtAQohvEP9ugW18qPiDQ0aQWCQrXupJE2zbe9yNYYJf5M5FHWAX024E0L4hjp6EP3DN4uHxba+Eu2J/0M7dwp9/RrbkNmiWk8JvTAMHo7WSloSdYVfTbgTQnifSjuH/vZLxcNcGzfB8NhzthLhrdphfGQa4D9lK4T3+dWEOyGEd6mcbPT5s+BCtm1DWASGx19AC6/v28CEX/HrCXdCCM9R5gL0f7wEacm2DQGBtpZEY/esKyNqD5lwJ0QdpHQr+kdz4ehB2wZNwzDxCbQ27X0bmPBL1V7hrrwJd0II/6VWLYZft9gfa6MfROt+ne8CEn6tWoni+PHjfPLJJxw4cIDc3FxeeeUVWrduzdKlS+nQoQNdu3Z1V5xCCDfREz9HJX5mf6wNHIJh4BAfRiT8ncv3KA4ePMhzzz3H2bNn6dOnj23ZxKKTGgxs2LDBLQEKIdyn1FyJ7teijXrAdwGJGsHlRLFkyRK6dOnC3Llzue+++xyea9WqFcePH692cEII9ylrroThwSfQnCnZIeoklxPF8ePHGTx4MJqmlZqZGR4eTnZ2drWDE0K4h0pLdpwrER1rG+EUGFTxgUJQjUQREBBAQUHZpYgzMzMJDQ11OSghhPuUnisRjuHxmTJXQlSZy4miffv2fPXVV+i6bt9W1LLYuHEjnTp1qn50QohqKZ4rcc62ISAQw2Mz0GJkroSoOpcTxZgxYzh27BhPP/00q1evBuCHH35g1qxZHD582GECnhDC+5Suo380z3GuxIMyV0I4z+VEER8fz6xZs6hfv769Uuw333wDwMyZM4mLk08sQviSWv1P+PVn+2Nt9AS0HjJXQjivWvMoWrduzfPPP4/ZbObChQvUq1ePoCC5OSaEr+nfrUN9W2KuxIA7MAwc6sOIRE1W7ZnZAIGBgfaS40II31K7tqJWfFi8ods1aKMn+C4gUeNVK1Fs2rSJzZs3c/78eQoLCx2e0zSNt956q1rBCSGco44eRP/gjcvmSjwpcyVEtbicKFavXs2qVato3rw58fHxBAQEuDMuIYSTyp0rId3BoppcThTff/89t956K/fff78bwxFCuELmSghPcnnUU05ODj179nRnLEIIF8hcCeFpLieKDh06cOLECTeGIoRwltJ19EUyV0J4lsuJ4v7772fTpk388MMPZGdno+t6qX9CCM9SnyyGX0rMlRglcyWE+7l8j+Lxxx8H4J133inzeU3TWL58uaunF0JUQv/uC9SGtfbH2oA70GRdCeEBLieKO++8s1TVWCGEd6jdW1ErPije0NU2V0L+JoUnuJwoRo8e7c44hBBVpI797jhXolU7DBNlroTwHJfvUTz22GPl3sw+deoUjz32mKunFkKUwz5XwlxirsT/zJC5EsKjXG5RpKenY7FYynyusLCQ9PR0l4MSoi5TSsHxQ+gb1kDSL6QWmiEgEK7qAqeOQU6WbcewcAx/fUHmSgiPc0utp8sdPXqUevXqeeLUQtRqymKxDXf9bbtthnVR95K5wLatiNGEYcpzaLFNfROoqFOcShRffPEFX375pf3xa6+9hsnkeIqiSrJ9+vRxT4RC1BFKqeIkYS579Ui7Vm1B5koIL3EqUcTExNC5c2fAtkhRmzZtiIiIcDyhyUSzZs0YMGCA+6IUoi44fqhqSQJsXVAnDkOrdp6PS9R5TiWKXr160atXL/vjkSNH0rhxY7cHJURdpG9YW1zQrzKFZvQNazE+PNWzQQlBNe5RPPLII7abbiXs3r2b06dP07lzZ+Lj46sbmxB1S9LO4nsSlVEK9uzwbDxCXOJyopg/fz4BAQH2YbAbNmzgo48+AsBoNDJ9+nQSEhKcPu+CBQvYu3cvBQUFREZGMmTIEOnGErWeuphbtS6nkqra+hCimlxOFIdOlT4BAAAgAElEQVQPH2bcuHH2x+vWraN///7ce++9LFy4kDVr1riUKIYPH86jjz5KQEAAZ8+eZebMmbRq1YrWrVu7GqoQfkldvIDavR31y2bYv8v5EwQEuj8oIcrgcqLIysqyL3+akpJCWloat9xyCyEhIdx0000sWLDApfM2b97c/n9N09A0jZSUFEkUolZQuRdQu7ddSg67wVr2XKRKaRok9Kp8PyHcwOVEERoaSk5ODgD79u0jPDycli1bAmAwGDCbXW8Wf/jhh2zatAmz2UyrVq3o3r27y+cSwtdUbo5tHetffoYDv5WfHGLi4HwqWK2VnzQgEMPgYe4NVIhyuJwo2rVrx2effYbRaOSrr76iW7du9udSUlJo1KiRy0FNnDiRCRMmcOjQIfbt21dqrgZAYmIiiYmJAMyePZuoqCiXXstkMrl8rDv5Qxz+EENtiUPPzqJg+3/J/3kj5j07y33zN7VpT/B1NxF07U0YY5uSPfcF8rf/WPH9isAggnv3I6LntV4tAlgbfi61KQZvxqGpy4cuVVFycjKvvvoqqampNG7cmBkzZtiHys6aNYvo6GgmT55c7QDff/99mjVrxm233VbhfufOnXPp/FFRUZw/f96lY93JH+LwhxhqchwqJxu1a4utW+ngHihvTZb4tmg9+6B1vw4tOtbxHOXNzAZbd1NAIHTpjWHC39DK+ADlSTX151JbY6huHHFxVV8B0eXftCZNmrBgwQJycnIIDw93eO6BBx4gMjLS1VM70HWd1NRUt5xLiIqUW2Opc08MNw+3vcFf9gleZWde6lbaDL8nlZ8cWrVD69EHrcd1aFEx5cagmUwYJj0FJw6jr19jGzJbFEdCLwyDh6O1auvOb1uISlX7I8nlSQKgRYsWLp0rKyuLvXv30qNHDwIDA9mzZw+bN2+2L5IkhKdUWGPp15/Rk3baP8lzMQf16xbbPYff94IqJzm0vvJScuiD1ii6yrFomgat2mF8ZBrgP59eRd3lcqKYNWtWhc9rmsbzzz/v1Dk1TWPDhg188MEHKKWIiorivvvuo2fPnq6GKUSlKq2xpFRxwji4B3KygXJ6bNu0L+5Walj15CCEszLyLLyweg//7+rGNAjxbDeky2dXSpVqhufk5HDu3DkiIiJo0qSJ0+eMiIioNAEJ4XZVrbFktRaX+C6iadDmquLk0MD1QRyiZvDmG3RFViSd57ez2axIMvBI79jKD6gGl7/LmTNnlrk9JSWFOXPmMHz4cFdPLYRXOVVjqUi7jrYupe7XokVKcqhLvPkGXZ6MPAsbj2WhgO+OZTGmc5RHk5bLK9yVJzY2lmHDhvHxxx+7+9RCeIYzNZYAAgIxPv0qhv63S5KoYy5/g/4zz8UJk9W0Iuk8+qXfWV0pViR59h6WR1JQREQEycnJnji1EO7n7ORQS6Fn4hB+r6w36Kq2KpRSFOoKs1VRaFWYrTqFetH/S2wreqxftu3S45wCC98ezUK/9NnGonu+VeH2s+bk5PDFF18QE1P+EEAh/IrJ5Nybv9RYqnOUUhzJyCfxaCaWS4PcLDqsP5JJco7tg4bjm7ui0Krb39wLLz3nKc4mLWe5nCimTJlS6ma2xWIhK8t2s+/JJ5+sXmRCeJiyWFDrljuXJKTGUq13sdDKyT8LOJFZwMkS/3ILSw+D1hXsTrnogygdebpV4fIZO3ToUCpRBAQEEB0dzTXXXENsrG9u8ghRFSotGf3DN20jnpwhNZZ8yp0jjqy64lyOmRN/2hJBUWJIy/Vc16LJoBFo1AgwagQaNAKMBvvjAPtz5W/blXyBoxkFlNU48WSrolotCiFqGqUUasv3qKULoSCv+InwSMi/WPHop0Bb+QziZWa0r7gy4kgpRWa+9VIiyLcnhtNZZrd0Bxk16NakHsM7NLrszV0j0Gi49FXDZNAwVKM2V0aehbUHMspMEuDZVoXvBgEL4WXq4gXUx++idvxYvNFoQht+D/S/HfXP+VWrseTFQnyiWFWGhBZYdE5lXWohlGgpZBdUoSLvJSYDNI0IIj4yiJaRQTQKMfH2tpRyk4pVwZ7Uizx2TROPDlEteSO9PJ5qVVT7u7p48SKnT58mIyODhg0b0qJFC0JCQtwRmxBuow7vR/9oLvyRVrwxpimGSU+itbzCto/UWPJrl484+uevqVzbIqLE/YR8knMKy5szX6ZGoSZ7Qij62jQiiABj8YeBd7enoCo5q6dvJhclSUs51WKKeKpVUa0zrV69mnXr1pGfn2/fFhwczJAhQ7jzzjurHZwQ1aWsVtQXK1BfrnSoyaT1G4w2ZiJaUHDxNqmx5HeUUmTlW9mbdpFvj2RiLTEk9IcTOfxwIqdK5wk2GWgZGUh8ZLBDUggLMlZ4nK/foItUpTVRxBNJy+XvaOXKlXzyySf079+fPn36UL9+fbKysvjpp59YuXIlVquV0aNHuy1QIZyl0lNsrYijB4s3hoZhuPcxtB7X+S4wUaYLBVZOZhVwKrOAU/avZqe6jQwaxIUHOiSDlpFBNA4LcOn+gK/foIv8fj6v0mRVxKLDwfN5le/oBJcTxXfffcftt9/O+PHj7duaN29Op06dCA0N5bvvvpNEIXxG3/o9asl7kF/iD+bKzrZ7DA19v+BMTePO0UZ5hTqnswrs9xJOZZk5lVlAhguznDVg8BX1uTIqhPgGwTSLCCTI5L6CE75+gy7y99talbndW61el3/iFy9epEuXLmU+17VrV7799luXgxLCVepiLmrpe6htPxRvNBrRht6DdvMwNEPFXQ2ibK6MNjJbdc5kmUu0Dgo4mWl2evipUcPe5VTqOQMYNI0Bbdyz/s3lfP0G7S9cThRXXHEFR48eJSEhodRzR48e5YorrqhWYEI4Sx05YJsbUfKGdeMmGCY+JTeiq6Gy0UYWXZGcY2sV2LqObMkhOcdc7lDOsgQYNJrXD6RFZBAt6tu6jCKCjDybeAprOZnCG+UrhJOJQi+xetcDDzzAG2+8gdFo5JprriEyMpLMzEy2bNnC999/z9SpU90erBBlUVYr6suVqC9XOKwwp/UZiHbXJLRgGYVXHSX76a26YsGWZDo0DuFUppmTWQWczTZjcSIjGDWIiwi0J4OixBAbFoDR4Hgf4d3tKT4bEiqKOZUo7r777lLblixZwpIlS0ptf+qpp1i+fLnrkQlRBep8qu2G9ZEDxRtD62EYPwWtZ1/fBVZL/HGxkG+PZmK9lH+tCn5NzuXX5NxKj9WA2PAAWtS3JYIWl24sx4UHOgw/LY+/jDgSTiaKO++8UyYbCb+hb/sBteRdyCtRa6ddRwwTnnBq6VFRtrPZZmYknrQniYpEh5rsLYOir83rV+/Gsr+MOBJOJgoZxST8gcq7iFq6ELX1++KNBgPakLFot94pN6yrqcCis3rfH3yy749ybyIbNLi3azRXRYfSvH4g9QLdf839ZcSRqOaEuz///JN169Zx4MABLly4QFhYGB07duT2228nMtIzoxBE3aaO/W67YZ2eUrwxOhbDpKfQWrXzXWC1xM6zF3h/ZyqpFyoemWTQIPVCIcM7eO7+j4w48h8uJ4pz587x/PPPk5ubS/v27YmJiSErK4uvvvqKH374gRdffNGldbOFKIvSraivVqPWLXO8YX1tf7SxD6EFh/owupovPbeQD3amsu3MhSrtL/cF6haXf8JLliwhNDSUV155hcaNG9u3p6en89JLL7FkyRKeeuoptwQpajelFBw/hL5hDST9QmpRjaXOPTHcPBwVHolaNBcO7y8+KKQe2vjJGHr1813gtUChVfH5wQxWJJ2noEQ/U4DBduO6osFMcl+g7nA5Uezbt49JkyY5JAmA6OhoRo0axUcffVTt4ETtpywW9EXzSldtNRfArz+j/7bdVqPJWqKMwxUdMEx8Aq1R47JPKqokKTWX97ancibbsbR6v5bhbD19odIbydKqqDtc/ulaLJZyq8SGhIRgsfhm0XFRcyilipOEuaCsHRxXn9M02w3r20bKDetq+DPPwuJf09h0Itthe3xkEI/0jmHT8exKq6UWkVZF3eByooiPj+frr7+ma9euGAzFQ+CUUmzYsIH4+Hh3xCdqs+OHyk8SZdDGT8HQb7CHg6q9rLrim8OZLPkt3WFZz2CTgXFdovhLuwYYDRoLd6TKaCPhwOVEMXLkSGbPns3f/vY3rrvuOho0aGCfmZ2SksL06dPdGaeohfQNayteUa4kTUPt3w2SKFxy6Hwe7+1I4WiGY1Lu2zKcCd0b0yg0wL5NRhuJy7mcKLp27cr06dNZvnw5n376qX1769atmTZtWrkFA4WwS9rpuJJcRZSCPTs8G08tdKHAyn9+S2f94UyHzqS48AAe7hVL1yb1fBabqDmqdQeqa9eudO3alYKCAnJzc6lXrx5BQUHuik3UYiojvcpdTnZVbX0IlFJsPJbFv3alk1ViPYdAo8aojo0Y3qEhAUb3leMWtZtbhioEBQU5JIjCwkK+/fZbbrvtNnecXtQSymqFpJ3oP26ApF+cP0FAoPuDqoVO/JnPwh2p7E93vHfQI64eD/WMITZcrqNwjsuJIjs7m/DwcIfaT2azmfXr1/PFF1+QmZkpiUIAoP5IQ/30LeqnbyEzw7WTaBok9HJvYLVMXqHO8qTzfH4ww2H+Q1SoiUk9Y7i6WZjUahMucSpRFBYW8vHHH/P9999TUFBAaGgod999N4MHD+a///0vH3/8MVlZWbRp04YpU6Z4KmZRAyiLBfZst7Ue9u0q+15E/BVw5qTjENjyBARiGDzM/YHWAkoptpzO4cOdafxRYpU4owZDr2rImM5RBLtx1TdR9ziVKFavXs0333xD586dadWqFWlpafzzn//kzJkzrF+/niZNmvDQQw/Rs2dPT8Ur/JxKT0H9uAG1ORGyM0vvEF4fre9AtL6DILoJ+gdvwG/bwFzB/YfAQOjSG+Jl8aHLJeeYeX9Haqmy350ah/Bw71ha1Jd7hqL6nEoUP//8M4MHD+bBBx+0b9u4cSMLFy4kISGBadOmYTLJDM26RlkKUbu2oX5cDwd+K72DpkGHrhj63QxdeqGZiodiGib8reyZ2UXHBdiShGHC3+pkt0l5a1WbrTqf7stg9b4/KCzRz1Q/2MiE7o25IT6iTl4v4RlOvav/8ccf9O7d22Hb1VdfzcKFC/nLX/4iSaKOUSlnba2HLRshJ6v0DvUb2lsPWlRMmefQTCYMk56CE4fR16+xDZktqvWU0AvD4OF1ehnTstaq/vXcBRbuSCWlRIVXDbi1XSTjukQT5oGS36Juc+qd3Wq1lirbUfQ4IiLCfVEJr6isGB/xbUt9KlWFZtSvW1D/XQ+H9pY+qWaATt0xXD8YOvdCM1b+pqVpGrRqh/GRaYB/TOwq75O8t2MouVb1wDb1Wb0vgy2ncxz2a9somEd6xXJFo2CfxClqP6f/AjIyMkhNTbU/LlpHOyMjg3r1HCfvxMSU/SlS+F6lxfiSdhZ3+ZhMqHOnLrUevofcnNInbBBlazn0HYjWsOavLlfWJ3lfxFBUmM+iK6auP+mwkFC9QAPju0Qz+IrIUmtNC+FOTieKN998s8ztc+bMKbVtxYoVzkckPK5KxfjMBbB7G/rr08FggKMHS+9nMNi6h/oNhk7da02hvss/yXuzOqpSCrNVkZxt5rujxetFX17uu3/rCO7r1pjIYOnuFZ7n1G/Zo48+6qk4hDdVtRhfodm27+UaNba1HvoMRGvQyDMx+lDJT/KVVUe16oo8i06+RSevsPhr3mWPi7+q4ufL2Cffole4BkRYoIH/vb4ZHWNkoSbhPU4lihtvvNFDYQhvcqoYXxGjEbpcbWs9dOiKZqid4/LTLphJLPFJ3qLD+iOZnM4qwKoolQzM5S0q7SFmqyIuQmZWC+/yq3ZrYWEhH374IUlJSVy4cIGYmBjGjh1Lt27dfB1a7eJMMT4AoxHDa4vQ6jfwXEw+csFs5WB6HgfS8ziQfpEDaXlcXmFbV7A3zXultA1QKobiWGT9B+F9fpUorFYrjRo1YubMmURFRbFr1y7mzZvHG2+8UWolPVENFU1uK4uu14okoZQiPdfC/vSLHEzPY396HqcyC6q4RE/ZNGzrOQQHGAgxGQgJ0C59NRBc1tdL/y9vn7xCncnrjpXbUpFV5YQv+NVvWnBwMKNHj7Y/7tGjB40bN+bYsWOSKNxEnTwCBq3ixZAvV0OL8Vl1xamsAvan2VoL+9Pz+OOiaysvGjXoGRfGXQlRDm/sgUYNgxsntv1nd3qlS5BKq0J4m18listlZmaSnJxM8+bNSz2XmJhIYmIiALNnzyYqKsql1zCZTC4f606ejEMphfm37VxcswTznp3OHaxpBPXqS6SXrtH5XDOPfZLErFuupFE95xJUfqGV/ak57DmXzZ5z2exNziHXbK3wGKMGbaPDaBtdj68PpGEpJ4FaFexOvcizTRs7HVdVnc81s/HYoUpXl7PosPF4No/e0M5jsZSlLvyt1KQYvBmHppQzndXeY7FYePXVV4mJieGhhx6qdP9z58659Dr+MLnLU3EoqxW18yfUN5/CmeOunSQwCMNTL6O1aufW2Mrz7vYU1h/O5Ja2kZV+Ys7Mt9juLaTZWgvHMvKp7N5ysMnAlVHBdIgO5arGIbRrFEJIgIF3t6eQeDSzwjdpkwEGtak8LldVJQZvxVKW2vy3UhNjqG4ccXFxVd7XL1sUuq7z9ttvYzKZmDBhgq/DqXFUQT7qp0TUt2vhjzTHJzUD9OwDFy/A4X1+VYyvovkLSinO5RTabjin57E/LY9zOZXfa2kQYqJDdAhXRYfQoXEo8ZFBpSanFb1uVT7Je/L+wO/n82StauGX/C5RKKV47733yMrK4plnnpH6UU5QOVmojV+ivv+y9OzpwEC0voPRBg5Bi44tf2Y2+KwY3+XzFxbuSOGq6FD2XxqNVHKltvI0rx9oay1Eh9ChcQiN6wVUGn/J162MJ+8PyFrVwl/53bvwBx98wNmzZ5kxYwaBgTXzJqq3qbRk1Lef2Up7Xz4/IiwCrf/taDfehhZeXI/L34rxZeRZHGYiW3TYcvoCW05fKPcYk0GjbaNgW1KIDuXK6BAigpyfHS6f5IWomF8livT0dBITEwkICGDSpEn27Q899BD9+vXzYWT+SZ08gvrmU9QvP4O67J0uKgZt8DC06wailbOOuT8U4zubbWbzqWw+P5DhUC67LGGBBq6KDqF9dCgdokO4olEwgW5Y91k+yQtRMb9KFNHR0axcudLXYfg1pRTs24W+/lM4uKf0Di3aoN0yAq37dVWq3OoLRcnh51M5HP+z4jIiBg3Gd4miZ9NwmtUPdOtQVCFE1fhVohDlq3QEU4duGG4ZAe0T/HLBGmeSQ0kGDdJyLbSIlJXahPAVSRQ+UtW1IGwjmL5FfftZ6RFMBgNaz35oNw9Ha9HaN99IBaqSHEyabY5CeZ1OMhNZCN+TvzwfqNJaEB26QVxz+OGbMkYwBaH1uzSCqZyV43ylKskhwKDRPa4efVqEsyc1l03Hsyu8mSwzkYXwLUkUXlb1tSC22v6VFBaBNuDSCKYw/1lR0Nnk0KtZGKEBRjLyLLy9LcXn8xeEEBWTvzpvq+paECVFx6INGoZ23YByRzBVl7NLf7qaHEryl/kLQoiKSaLwMqfXgmjdHsPUVz0+gqkqS3+6IzmUJPMXhKgZJFF4m7NrQZw57vEkUVHpDHcnh5Jk/oIQNYMkCm8ro7ZSRmA4czuM48n9H9PAfNlMZGdXonPB5aUzFv2SSvPIII8kByFEzSOJwttMRrA4romwquUADtSPZ1XLgTx0eK3j/h5cC6LAonMiM7/U0p//PZkDJ3NK7S/JQYi6SRKFl6iLuahlC0sliYzAcL6P7YXSDGyM7cmok4nFrQpNg4ReVX4Nq67IMVvJyreSlW8h89LXrHwrmfkWsgpKPraSX4UbBJIchBCSKLxAHdmP/uHc0hPmsLUm9EszqXXN4NiqCAjEPGAoWTlmsgouvdmXkQSy8q1kFljIKbA6tXBdRTTgoZ6NubF1fUkOQtRxkig8SFmtqC9WoL5c6Vi0r1EMZP9JhhbExtheWAy2H4PFYGJ93DX8HtGC3IB6ZAWFU7BNAcc8Ep9G+TOijQY4lWWWJCGEkEThKSotGf2juXDs9+KNofXQ7pmC1u0asha9zcsqgUKD449AaQaOhzdz+XXDAg3UDzZRP8hI/WATkcFGIoNN1A82Xvpn+79S8MTXJzCXsyScTHITQhSRdwA3U0qhtmxELX0fCkqM+2/XCcODfyM/vBHrDmbwaf1bybNU3k9kMmhEBhe/6Rd/NVI/yPamX5QIIoJMBBirVhDw3e0plU52k0luQgiQROFWKvcC6uN3UDt/Kt5oNKINvQfLwCF8fSyHld8fJSu/4pXajBr0bRnOI71jCTEZ3F4N1l+W/hRC1Azy1+8m6vck9I/mwZ8lJorFNEVNfIJN1miWf3mStFxL+Scowapsq7s90F0RGuD+kuFSOkMI4QxJFNWkLIWoz5fa1oko+eZ7/c1s63MXS5KyOJOd4nBMkFGjUFcVjlDy5Bu0lM4QQjhDEkU1qJQztmGvJ48UbwsLJ2nE3/j4QhRHtqQ77B8RZOS2dpF8sj+j0mGsnuz2kdIZQghnSKJwgVIK9eMG1IoPHarA/p4wgKVtbyfpVCGQb98eYjIwrENDhrRvwL92pdsWLaoC6fYRQvgDSRROUjnZ6P9+22GtiJMRTVl2zQS2m8Phj0L79gCDxl+ubMCdHRoSEWy71NLtI4SoaSRROEHt34W+aD5kZQCQEtyQFVcN47/126NK1O4zaDCwTX3GdI4iKjTA4RzS7SOEqGkkUVSBKixErfm3bd1qbPWZVrccwLdNr8GKwWHffi3DuTshmqYRnivmJ4QQ3iSJohLq3Cn0D96EM8e5YAphTYsb+bJZX8wGx5ZCj7h63NMlmtYNg30UqRBCeIYkinIopVCbvkatWkSeDl+2uIm1LW7koinEYb8O0SGM7xpNh8ahPopUCCE8SxJFGVR2JvriBRTu3cWGuKtZ3XIAWYHhDvu0ahDE+C7RdI+r5/aZ00II4U/qXKJQSsHxQ+gb1pBx8HeeaTuKJw+vokH79hhuHo7KycayeAH/DW3D8qufJj24ocPxceEBjE2Ipk/LcAySIIQQdUCdShTKYkFfNA9+2w6FZlZdMdS2slxcPx769TOsu7ayreFVLG3/IGfqxTgc2yjUxF2doxjQuj5GgyQIIUTdUWcShVKqOEmYC0qtLNcx8yifNb+BIxEtHI4LDzIyqmMjbm0XSaDRUM7ZhRCi9qoziYLjh+xJAhxXljMbTLzZcbzD7iEGxbCO0Qy5qoEs3iOEqNPqTKLQN6yFQtusuIzAcL5rUryyHFpxSyFAL+SWs1u4s8FFGiT8P1+EKoQQfqXOJAqSdtqruy6NvwWLdtm3rhTNclN5PukjogqyIDAIkEQhhBB1p9PdXNya+CmmK1w+YknTSAtpiFFdWlSo0IwQQoi6lCgCbSU1St6buJyuGVjVcqDtQYCU4BBCCKhLiaJzTzKCIvg+tsS9ictYDCY2xvbkz8BwSOjl5QCFEMI/1ZlEYRg8jFXxg8ptTRTRNQOrWg3GMHiYlyITQgj/VmcSRUZsa76P7Vlua6KIvVURU3Y5cCGEqGvqTKJYufcPdGPV5kPoRiMr9/7h4YiEEKJm8Kvhsd988w2bNm3i1KlT9OnThylTprjt3LKynBBCuMavEkWDBg0YMWIEv/32G2aze4enyspyQgjhGr9KFFdffTUAx44d448/pOtHCCH8gV8lCmckJiaSmJgIwOzZs4mKinLpPCaTyeVj3ckf4vCHGCQOiaMmxOEPMXgzjhqbKAYOHMjAgQPtjwMDXZ8gV51j3ckf4vCHGEDiuJzE4cgf4vCHGMA7cdSZUU/lmT59uq9DAPwjDn+IASSOy0kcjvwhDn+IAbwXR51PFEIIISrmV4nCarViNpvRdR1d1zGbzVitVl+HJYQQdZpx5syZM30dRJHVq1fz0ksvcfDgQU6dOsWnn36Kpml07NjRo6/bunVrj56/qvwhDn+IASSOy0kcjvwhDn+IAbwTh6bUpUUahBBCiDL4VdeTEEII/yOJQgghRIVq7DyK6rpw4QLvvvsue/bsITw8nLFjx9K3b1+vxzFz5kwOHz6MwWDL2Q0bNmT+/Pkefc2KamolJSXx0Ucfcf78edq2bcvkyZOJjo72ahxpaWk89thjBAUF2fcdOnQoI0eOdHsMhYWFfPjhhyQlJXHhwgViYmIYO3Ys3bp1A7x3PSqKw5vXA2DBggXs3buXgoICIiMjGTJkCAMGDAC8+/tRXhzevh4AycnJPPXUU1x99dX89a9/BeCnn35i6dKl5OTk0LlzZyZPnkxYWJjHYigrjn379vHiiy86zKV48MEHufHGG937wqqOmjdvnpo7d67Ky8tTBw4cUPfee686deqU1+N44YUXVGJioldfc+vWrWrbtm3q/fffV2+//bZ9e1ZWlrr33nvVzz//rAoKCtS///1v9b//+79ejyM1NVWNGjVKWSwWj712kby8PLVixQqVmpqqrFar2rlzpxo/frxKTU316vWoKA5vXg+llDp16pQym81KKaXOnDmjJk6cqI4ePer134/y4vD29VBKqf/7v/9TM2bMUPPnz7fHNn78eLVv3z6Vl5en/v73v6t58+Z5PY69e/eqhx9+2OOvWye7nvLz89m2bRtjxowhODiY9u3b07NnT/773//6OjSvuPrqq+nduzfh4eEO27dv307z5s259tprCQwMZNSoUZw4cYKzZ896NQ5vCg4OZvTo0TRu3BiDwUCPHj1o3Lgxx44d8+r1qCgOb2vevDkBAQEAaJqGpmmkpKR4/fejvDi8bfPmzYSGhtKpUyf7th9//JEePXrQoUMHgoODGTNmDNu2bSMvz3NVp8uKw1vqZBC6frEAABSFSURBVNdTcnIyRqORuLg4+7aWLVuyf/9+n8SzdOlSli5dSlxcHHfddZfHhwOX5/Tp07Rs2dL+ODg4mNjYWE6fPk3Tpk29Hs/kyZPRNI2EhATuueceIiIiPP6amZmZJCcn07x5czZs2OCz61EyjiLevB4ffvghmzZtwmw206pVK7p3786yZcu8fj3KiiM7OxvwzvW4ePEiK1eu5Pnnn+e7776zbz9z5gzt2rWzP46NjcVkMpGcnOyR4arlxQGQlZXFpEmTCAwMpFevXtx1110EBwe79fXrZKLIz88nJCTEYVtoaCj5+flej2XcuHE0a9YMk8nE5s2bee2113j99deJjY31eiz5+fml/th8cV0iIiJ49dVXiY+PJycnh48++oi33nqLZ5991qOva7FYeOutt7jhhhto2rSpz65HWXF4+3pMnDiRCRMmcOjQIfbt24fJZPLJ9SgrDm/+fqxYsYKbbrqJRo0aOWzPz88nNDTUYVtoaKjHWhTlxdG0aVPmzJlDXFwc58+f5x//+Af//ve/eeihh9z6+nWy6yk4OLjUDzQvL8/tWbgq2rZtS0hICAEBAdx4441ceeWV7Nq1y+txQNnX5eLFi16/LsHBwbRp0waj0UhkZCQPPvggv/32m0eb9bqu8/bbb2MymZgwYYI9Dm9fj/Li8Pb1ADAYDLRv354//viDDRs2+Oz3o6w4vHE9Tpw4QVJSErfffnup58p7D7n8A6in44iMjKRZs2YYDAYaN27MuHHj2LZtm9tjqJMtiiZNmmC1WklOTqZJkyYAnDx50qGZ7yuapqF8NAeyefPm/PDDD/bH+fn5pKam+sV1ATx2XZRSvPfee2RlZfHMM89gMtn+LLx9PcqLo6L9vUHXdfv37cvfj6I4yuPu67Fv3z7S09N59NFHAdv3q+s606ZNo0uXLpw8edK+b2pqKoWFhfb3E2/F8dprrznsq2kaul7FpTydUGdbFFdffTUrVqwgPz+fgwcPsmPHDq6//nqvxpGbm8vu3bvtNa1+/PFHDhw4QNeuXT36uuXV1OrduzenTp1i69atmM1mVq9eTcuWLT3W/1xeHIcPH+bcuXPouk5OTg7//Oc/6dixY6mmvrt88MEHnD17lmnTpjkMM/T29SgvDm9ej6ysLDZv3mx/M9q9ezebN2+mc+fOXr0eFcXhresxcOBA3nrrLebMmcOcOXMYNGgQ3bt359lnn6Vfv3788ssvHDhwgPz8fFasWMHVV1/tkRZFRXHs3buX9PR0lFKcP3+epUuX0qtXL7fHUGdLeFy4cIF33nmHpKQkwsLCGDdunNfnUWRnZ/Pqq69y9uxZDAYDTZs2ZcyYMSQkJHj0dVeuXMnq1asdto0cOZLRo0ezZ88eFi1aRHp6un2cfOPGjb0aR1xcHMuWLSM7O5uQkBD7zcrIyEi3x5Cens6UKVMICAiwz2UBeOihh+jXr5/XrkdFcWia5rXrkZ2dzZtvvsnJkydRShEVFcWtt95qX/vFW9ejojh++uknr12PklauXElKSorDPIolS5Zw4cIFr82juDyOL774gnXr1pGbm0t4eDi9evXi7rvvdnvCqrOJQgghRNXUya4nIYQQVSeJQgghRIUkUQghhKiQJAohhBAVkkQhhBCiQpIohBBCVEgShRBCiApJohBCCFGhOlHradOmTbzzzjv2x8HBwcTExDBgwAAGDRqE0Wj0YXSu2759O2lpaWUWC6uJNm7cyNq1a0lPTycoKIjFixeXud/lP0+j0UhUVBTXXnstI0eOdCh/UVVFs8RXrlzpavhe5czP/vLrVdJzzz3nkUoANf13c9GiRaSlpTF9+nT7ttGjRzN58uQyV4/75Zdf+Oabbzhy5AgFBQWEh4fTvHlzBg8eTO/evct9nfLO+eeff/LII48wa9Ys2rdvX+7xM2fOdPj65ZdfsnHjRubMmeMww7+66kSiKPLEE0/QsGFD8vLy2LJlC4sWLSIrK4sxY8b4OjSX7Nixo9yqkjVNRkYG77//Pn379mXy5Mn2BWsqUvTzzM/PZ/v27axdu5b8/Hx71dXazJWffdH1KqlZs2buDg2o2b+bKSkpfPvtt7z00ktV2n/lypV88skn9O/fn0GDBhEaGkpycjJbt251eaGlHTt2EBER4bDmRVUMGjSItWvX8sMPP3DTTTe59NplqVOJIj4+3r7OQ5cuXUhNTeWrr76qdqIoLCys0hubKF9KSgq6rnPjjTdW+AmqpJI/z4SEBJKTk/n++++5//773fppqrYoeb1qIm/9nX311VfEx8fTpk2bSvc9c+YMn3zyCSNHjmTUqFH27Z06dWLQoEEuV3LdsWMHPXr0cPr3ODAwkBtuuIF169ZJonCXNm3asG/fPrKysqhfvz4nTpxgxYoVHDx4ELPZTOvWrRk7dixXXXWV/ZiiLoo33niD//znPxw8eJDOnTszdepUwFY7ftWqVRw4cICCggKioqK48cYbGT58uP35qr7G/PnzWbx4Mfv37yc8PJz+/fszYsQIDAYD//jHP+wln0ePHg1AdHQ0//jHPwDbG++qVas4ePAgmZmZNGjQgC5dunD33Xc7FC776aefWL16Nenp6cTGxnL33XfzxRdfAMXN2SJVib08u3fvZtWqVZw4cQKTyUTHjh255557iIuLc/heXnzxRQBuuOEGpkyZUvUfJtCqVSuSkpLIycmhfv361Y67suO2bNnCvHnzmDNnjsPKbwCvvvoqGf+/vXOPaep84/i3hZYCs0G5NYyB1EIZCgNxc7gwhk7mlWlY1bhYydyybJljztvGVEiwwazELZj9t4DLvEFxTtycw0QZlcRiUOc6Mi5yyRbAMkopl0GB9veH60lPe1rOafn9/CW8n7/ac/q+7/Ocp+953vtjNEKtVgNgb4+5sL23sHlObPTwJN9XX32FlpYWF1mdh1DY1LPZZO3t7cXZs2fR2tqKf/75B2KxGPHx8SgoKHA73Dw1NQWtVou8vDxWz0yv18NmsyE5OZnxvjcNlvHxcfz+++/Yv38/7XpjYyM0Gg0MBgMkEgl27NjBmH7VqlWora1Fa2sr5HI55/KZmNeOwmAwgM/nQyQSobOzE0VFRVi8eDHeffddCIVCXL9+HSUlJTh+/LhLeEO1Wo3s7Gy8/vrr4PF4AICOjg4UFxdDIpFg9+7dCA0NRV9fH3VuPdcyysrK8Morr2Djxo1obm5GdXU1QkNDkZ2djby8PJjNZjx8+JCqPI6tLaPRiNDQUOTn5yM4OBgGgwGXLl1CaWkpVCoVgMcngZ46dQrp6elQKpUwm804ffo047n6XGV35P79+ygtLcWyZcuwb98+6ljmo0ePQq1WIy8vD1KpFJWVldizZw/i4uK8Cms5MDCAoKAgWgxub+Vmky49PR1BQUHQarU0R2EymfDrr7/izTff5GQPR3yxvTusVitmZmao7zwej3qRsX1ObPTwVj4mmOoZW1lPnDiB4OBgvP322xCLxTAajbh7967HuBXt7e0YGxtj1fgBQAVt0mg0UCgUSEhI8Lk3e+/ePfj7+9Ocz4MHD1BeXo60tDSqrlZWVmJmZoYW0hl43HMMDAzE/fv3iaPwBntFsc9R6HQ6pKenIyAgAGfOnEFYWBiKioqogDGpqanYv38/ampqqD+8nfXr12PDhg20a99++y0WLFgAlUqFgIAAAKAFQudaxqZNm6juY0pKCvR6PRobG5GdnQ2JRAKxWAx/f3/GccykpCQkJSVR3+VyOSQSCY4dO4auri7ExcWhuroa0dHROHjwIFUJY2Ji8Mknn7g4Cq6yO3LhwgVERkaisLCQasklJCSgoKAAV65cwe7du6mYBtHR0azHZR3t2dTUBJ1O5zLs5K3cbNIJhUK8+OKLuHXrFnbu3EmV29jYCAC0Y+vZ2MMRX2zvjo8++oj2XS6Xo6SkhNNzYqOHt/IxwVTP2MhqNpvR39+PQ4cOYcWKFVTa2UIJtLW1gcfjufQQ3bFq1SrU19fjt99+o0IWLF++HGvWrGHtbJy5c+cOnnvuOdqiDI1Gg6ioKBw6dIj6n0VFReHIkSMujoLP5yM2Nhbt7e1elc/EvHIUjhWFx+MhMzMT+fn5sFgsaGlpwdatW8Hj8WitruTkZGi1Wpe8nFcyTE5O4o8//kBubi7lJBzxpozly5fTvj/zzDPo7u5mpev09DRqa2vR0NCAgYEBTE1NUfd6e3sRGxuLhw8fUvLYkUqlLvEFvJHdzsTEBLq6urB161Zadz8iIgJyuRwtLS2s9GHC+cWXk5ODdevW+Sw3l3RZWVm4ceMG9Ho9tXqooaEBycnJWLhwIfW72ezh7Ch8sb07Dhw4QIu5bI9ZwEVfrnr4inM9YyvrggULEBkZibNnz8JkMmHp0qWsos8NDQ0hMDBw1uiCdoRCIYqLi9HR0YHm5mY8ePAAWq0WWq0W77zzDhXHgy3T09O4d+8e9uzZQ12zWq3o6OjAli1baI2ghIQEhIeHM+YjFovR19fHqWxPzCtHYa8ogYGBCAsLozy20WiE1WrFxYsXcfHiRca0VquVZiTnICljY2Ow2Wwuwc/tjI6Oci7DOQiKQCCgVUxPnDt3Dj/99BPeeOMNyOVyiEQiGI1GlJWVYWpqCmazGTMzM4xDPI7j+97Kbsf+XJiCyoSEhGBgYICVPkzY7Wk2m/HDDz+grq4O8fHxyMrK8kluLukSExMRHh6OhoYGpKSk4K+//kJXVxf27t1L+/1s9nDGF9u7IyYmhnEym4u+XPXwFef/DRdZjxw5Ao1Gg/Pnz2NkZAQRERHIzc1FTk6O2/K8nTCXyWSQyWTYvn07Ojo6cPToUVy/fp3RUdhsNthsNsb6otfrYbFYkJ6eTl2z11Xnegm4Ph87QqEQFouFsx7umFeOwl1FCQoKAo/Hw2uvvUa9ZJxxNqpjKxwAgoODwePxYDQaGdN7U4YvNDY2IisrizYpp9frqc9isRh+fn4wm80uaYeHhxEWFjYnstufi8lkcrlnMpl8igjmaM9ly5bhwIEDOHPmDFauXAmRSOS13FzS2XumV69exeTkJBoaGiASiVxawrPZ40nCRV9f9RAIBJienna5PjIyQptbsuNcz7jIGhkZiQ8++AA2mw09PT24du0avv76a4SHhyMtLY0x7VNPPYWxsTHW+jAhk8kgFotpL+qqqio8evQI4+Pj6O/vx7Fjx1yWKgOP958kJSUhODiYumavq8PDwy6/N5lMjL2K0dFRxufpLfPKUbhDJBLh2WefRU9PD+Li4rx6YQcEBCAxMRFarZZx09dclOGMQCBw22qwWCwuKzvq6+upz3w+H0uWLIFOp4NCoaBNFBoMBpqj8EV2kUgEqVSK27dvY9u2bVTagYEBtLa2Yv369azz8oRAIMCuXbvw+eefo66uDrm5uV7LzTXdyy+/jO+++w46nQ63bt3CCy+84DL8OJs9uOLJ9lzhoi9bPdzJFx4eDpPJBLPZTPVm+/v70dvby2ri1Rub8ng8LF68GEqlEjdu3MCff/7p1lE8/fTTmJ6exuDgoNvRATtDQ0O04UU7bW1tGBoaQkZGBnWts7MTExMTOHz4sNvY3jabDc3NzdQKSTt8Ph8ymQy3b9+GQqGgdG5vb8fAwACjozAYDJDJZB7l5wJxFP+iVCpRVFQElUqF1atXY+HChTCbzejq6oLVaqWtYHHHrl27UFxcjM8++wybN2/GokWLYDAY0N3djbfeemtOynAkOjoao6OjqKurg1QqhVAoRExMDIDH+0R++eUXqtWt0+nQ2tpKS79t2zYcP34carUar776KkZGRqDRaBASEuLSkvNF9u3bt6O0tBQnTpxATk4OJiYmoNFoEBQUNKcbslasWIElS5bgypUrWLduHYRCoddyc0kXFRWF+Ph4nDt3DkajkbGly8YeXPBke29gqy9bPdzJl5GRgaqqKpw6dQobN27EyMgIvv/+e06r3NjI2tPTg9OnTyMjIwMSiQRWqxX19fXw8/OjLTBxxj4B3dHRMaujKC8vx8zMDF566SVER0djcnISbW1tuHbtGmJjY2n7Kjo7O1FYWOjWSQCPX/xDQ0N4/vnnXe4pFAqoVCqo1WqsXbsWZrMZ1dXVjENPY2Nj6Ovrw+bNmz3KzwXiKP5FKpWitLQUNTU1qKysxPj4OMRiMeLi4rB27VpWechkMpSUlKCqqgoVFRWYmppCeHg4tT1/LspwZPXq1Whra8P58+cxNjZGW0tv35184cIFAEBaWhoKCgpQWFhIpU9JScGHH34IjUaDsrIySCQSKJVK1NTUuPyhfZE9NTUVn376KTQaDb744gvaPgqm7rcv7NixAyqVCnV1ddi0aZPXcnNNl5mZiYqKCixatAhLly51uc/GHlzwZHtvYKsvWz3cySeRSPDxxx+jqqoKarUaUVFRUCqVuHTp0pzKGhISgtDQUPz4448YHByEQCBATEwMDh8+7HEpd0REBGQyGZqbm7Fy5UqPcmRlZaGpqQmXL1/G8PAw+Hw+oqKisGXLFmzYsIEaVRgcHITVap11ov/OnTuQSqWMDiolJQV79+6l1dX8/HxcvXrV5bd3796Fv7+/x6NDOGMjEBz4+++/bTt37rRpNJonLQqB8ES4efOmTalU2iYmJmjXFQqF7ebNm5zza2pqsqlUKsZ7jnkWFBTYampqOOfvjEqlspWXl/ucjyOkRzGPsVgs+Oabb5CcnAyxWIxHjx6htrYWQqEQa9asedLiEQhPhMzMTFy+fBk///wzcnNzfc6vs7OT1XEgX375pc9ldXd3Q6/X4+TJkz7n5QhxFPMYPp8Pk8mEiooKjIyMQCQSITExEfv27WOcpCMQ5gN+fn5477330NXVNSf5/S8PHTWZTHj//ffn/Ewv4ijmMf7+/jh48OCTFoNA+L8jISHB513lT4LU1NT/Sr48m83DwScEAoFAmPeQs5gJBAKB4BHiKAgEAoHgEeIoCAQCgeAR4igIBAKB4BHiKAgEAoHgEeIoCAQCgeAR4igIBAKB4BHiKAgEAoHgkf8An8JVejbDMTYAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.style.use('ggplot')\n",
    "markers = ['o', '^']\n",
    "fig = plt.figure()\n",
    "ax = plt.axes()\n",
    "\n",
    "methods = ['Greedy-AS', 'Grad']\n",
    "curves = [eval_curve_greedy_as, eval_curve_grad]\n",
    "\n",
    "for i, (method, curve) in enumerate(zip(methods, curves)):\n",
    "    steps = len(curve)\n",
    "    ax.plot(np.arange(steps), curve, label=method, linewidth=3, marker=markers[i], markersize=12)\n",
    "\n",
    "\n",
    "leg = plt.legend(framealpha = 1, loc = 2, fontsize=14)\n",
    "for text in leg.get_texts():\n",
    "    plt.setp(text, color='black')\n",
    "\n",
    "plt.xticks(np.arange(10), np.arange(10)*5)\n",
    "plt.tick_params(labelsize=12)\n",
    "plt.ylabel(r'Robustness-$\\overline{S_r}$', fontsize=16)\n",
    "plt.xlabel(r'Percentage of Relevant Features (|$S_r$|/d)', fontsize=16)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.5.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
