{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import urllib.request\n",
    "import tarfile\n",
    "\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "import pandas as pd\n",
    "\n",
    "from sklearn.linear_model import Ridge\n",
    "from sklearn.metrics import mean_squared_error\n",
    "from sklearn.model_selection import KFold\n",
    "\n",
    "from tqdm import tqdm\n",
    "\n",
    "from gcv.ensemble import PreSketchedEnsembleRidge, EnsemblePreSketchTransform\n",
    "from gcv.sketches import CountSketchFactory, SketchSizeParams"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# download UCI gene expression cancer RNA-Seq Data Set\n",
    "# https://archive.ics.uci.edu/ml/datasets/gene+expression+cancer+RNA-Seq\n",
    "URL = 'https://archive.ics.uci.edu/ml/machine-learning-databases/00401/TCGA-PANCAN-HiSeq-801x20531.tar.gz'\n",
    "DATA_DIR = os.path.expanduser('~/data')\n",
    "FILE_NAME = 'TCGA-PANCAN-HiSeq-801x20531.tar.gz'\n",
    "EXTRACT_DIR = 'TCGA-PANCAN-HiSeq-801x20531'\n",
    "\n",
    "# download data\n",
    "if not os.path.exists(DATA_DIR):\n",
    "    os.makedirs(DATA_DIR)\n",
    "\n",
    "if not os.path.exists(os.path.join(DATA_DIR, EXTRACT_DIR)):\n",
    "    urllib.request.urlretrieve(URL, os.path.join(DATA_DIR, FILE_NAME))\n",
    "\n",
    "    # extract data\n",
    "    tar = tarfile.open(os.path.join(DATA_DIR, FILE_NAME))\n",
    "    tar.extractall(DATA_DIR)\n",
    "    tar.close()\n",
    "\n",
    "# load data\n",
    "df = pd.read_csv(os.path.join(DATA_DIR, EXTRACT_DIR, 'data.csv'), index_col=0)\n",
    "targets_df = pd.read_csv(os.path.join(DATA_DIR, EXTRACT_DIR, 'labels.csv'), index_col=0)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "classes = ['BRCA', 'KIRC']\n",
    "dataset_idx = np.where(targets_df.values == classes[0])[0].tolist() + np.where(targets_df.values == classes[1])[0].tolist()\n",
    "\n",
    "X = df.values[dataset_idx, :]\n",
    "y = (targets_df.values[dataset_idx, :].ravel() == classes[0]).astype(int) * 2 - 1\n",
    "\n",
    "rng = np.random.default_rng(0)\n",
    "idx = rng.permutation(X.shape[0])\n",
    "n_train = int(0.8 * X.shape[0])\n",
    "train_idx = idx[:n_train]\n",
    "test_idx = idx[n_train:]\n",
    "\n",
    "X_train, X_test = X[train_idx, :], X[test_idx, :]\n",
    "y_train, y_test = y[train_idx], y[test_idx]\n",
    "\n",
    "X_mean = X_train.mean(axis=0)\n",
    "X_std = X_train.std(axis=0)\n",
    "\n",
    "X_train = (X_train[:, X_std > 0] - X_mean[X_std > 0]) / X_std[X_std > 0]\n",
    "X_test = (X_test[:, X_std > 0] - X_mean[X_std > 0]) / X_std[X_std > 0]\n",
    "\n",
    "# save data\n",
    "np.savez('data/rna-seq.npz', X_train=X_train, X_test=X_test, y_train=y_train, y_test=y_test)\n",
    "\n",
    "n_train, p = X_train.shape\n",
    "n_test = X_test.shape[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(99, 20223)\n",
      "6572.211796660964\n"
     ]
    },
    {
     "data": {
      "image/png": "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vfXWmDdvXmzatCkuuuiimDZtWrz11luVeDkAoA+pSHwsXrw4vvWtb8V1110XZ555Ztxzzz3R2NgYS5YsqcTLAQB9SNmv7XLw4MHYuHFjzJkzp9P+qVOnxrPPPtvl+Pb29mhvb++43draGhERbW1t5R4tIiIOt79Xkeftqe7+nMcy44cf15PHfJLH5dLdfL05T2/78HoM5LXoL3yNU2m5vsaOPGdK6eMPTmX2y1/+MkVEeuaZZzrt/5u/+Zs0ZsyYLsffcccdKSJsNpvNZrP1g23Hjh0f2woVu6ptoVDodDul1GVfRMTcuXNj9uzZHbcPHz4c7777bpx66qndHt8TbW1t0djYGDt27IiampqyPGd/YF2Oztp0z7ocnbU5OmvTvf62Liml2LdvXzQ0NHzssWWPj9NOOy0GDRoULS0tnfbv2rUrhg8f3uX4YrEYxWKx077f+Z3fKfdYERFRU1PTL/4PLjfrcnTWpnvW5eiszdFZm+71p3Wpra09puPK/oHTk046KcaPHx/Nzc2d9jc3N8ekSZPK/XIAQB9TkW+7zJ49O775zW/GhAkT4vzzz49//Md/jLfeeituuOGGSrwcANCHVCQ+rr766tizZ0/89V//dezcuTPGjh0bq1evjqampkq83McqFotxxx13dPn2zkBnXY7O2nTPuhydtTk6a9O9gbwuhZSO5WdiAADKw7VdAICsxAcAkJX4AACyEh8AQFb9Jj7uu+++GDVqVJx88skxfvz42LBhw0cev27duhg/fnycfPLJcfrpp8c//MM/ZJo0r1LW5Sc/+Ulceuml8elPfzpqamri/PPPjyeffDLjtHmV+jVzxDPPPBNVVVXxu7/7u5UdsJeUui7t7e0xb968aGpqimKxGGeccUb80z/9U6Zp8yp1bZYvXx7jxo2LU045Jerr6+Paa6+NPXv2ZJo2j/Xr18f06dOjoaEhCoVCPPbYYx/7mIHy/lvq2gyo9+CyXNCllz388MPpxBNPTPfff3/avHlzuuWWW9KQIUPSm2++2e3xW7duTaecckq65ZZb0ubNm9P999+fTjzxxPTjH/848+SVVeq63HLLLel73/te+u///u/02muvpblz56YTTzwx/fznP888eeWVujZH7N27N51++ulp6tSpady4cXmGzagn6/K1r30tTZw4MTU3N6dt27aln/3sZ12u7dQflLo2GzZsSCeccEL6u7/7u7R169a0YcOG9MUvfjFdccUVmSevrNWrV6d58+alf/mXf0kRkR599NGPPH6gvP+mVPraDKT34H4RH+eee2664YYbOu37/Oc/n+bMmdPt8bfffnv6/Oc/32nf9ddfn84777yKzdgbSl2X7nzhC19Id955Z7lH63U9XZurr746fec730l33HFHv4yPUtflP/7jP1JtbW3as2dPjvF6Valr87d/+7fp9NNP77TvBz/4QRoxYkTFZuxtx/IX7EB5//2wY1mb7vTX9+A+/22XgwcPxsaNG2Pq1Kmd9k+dOjWeffbZbh/z05/+tMvxv//7vx8vvPBC/PrXv67YrDn1ZF0+7PDhw7Fv374YOnRoJUbsNT1dm6VLl8Ybb7wRd9xxR6VH7BU9WZfHH388JkyYEHfffXd85jOfiTFjxsRtt90W77//fo6Rs+nJ2kyaNCnefvvtWL16daSU4n//93/jxz/+cVx++eU5Rj5uDYT333Lpr+/BERX6Dac57d69Oz744IMuF60bPnx4l4vbHdHS0tLt8YcOHYrdu3dHfX19xebNpSfr8mGLFi2KAwcOxB/+4R9WYsRe05O12bJlS8yZMyc2bNgQVVV9/j+bbvVkXbZu3RpPP/10nHzyyfHoo4/G7t2748Ybb4x33323X33uoydrM2nSpFi+fHlcffXV8X//939x6NCh+NrXvhZ///d/n2Pk49ZAeP8tl/76HhzRjz5wWigUOt1OKXXZ93HHd7e/ryt1XY546KGHYv78+bFy5coYNmxYpcbrVce6Nh988EFcc801ceedd8aYMWNyjddrSvmaOXz4cBQKhVi+fHmce+658dWvfjUWL14cDz74YL87+xFR2tps3rw5/uzP/iz+6q/+KjZu3BhPPPFEbNu2zTWuYuC8/34S/f09uM//E+60006LQYMGdfnXx65du7rU9RF1dXXdHl9VVRWnnnpqxWbNqSfrcsTKlSvjW9/6VjzyyCPxla98pZJj9opS12bfvn3xwgsvxKZNm+Kmm26KiN/8pZtSiqqqqlizZk1ccsklWWavpJ58zdTX18dnPvOZTpfRPvPMMyOlFG+//XaMHj26ojPn0pO1WbhwYVxwwQXxF3/xFxER8aUvfSmGDBkSF110UXz3u98dsP/CHwjvv59Uf38PjugHZz5OOumkGD9+fDQ3N3fa39zcHJMmTer2Meeff36X49esWRMTJkyIE088sWKz5tSTdYn4TW3PnDkzVqxY0W+/N13q2tTU1MRLL70UL774Ysd2ww03xOc+97l48cUXY+LEiblGr6iefM1ccMEF8c4778T+/fs79r322mtxwgknxIgRIyo6b049WZv33nsvTjih81vsoEGDIuL//0t/IBoI77+fxEB4D46I/vWjtg888EDavHlzuvXWW9OQIUPS9u3bU0opzZkzJ33zm9/sOP7Ij3r9+Z//edq8eXN64IEH+uWPepW6LitWrEhVVVXphz/8Ydq5c2fHtnfv3t76I1RMqWvzYf31p11KXZd9+/alESNGpKuuuiq9/PLLad26dWn06NHpuuuu660/QsWUujZLly5NVVVV6b777ktvvPFGevrpp9OECRPSueee21t/hIrYt29f2rRpU9q0aVOKiLR48eK0adOmjh9BHqjvvymVvjYD6T24X8RHSin98Ic/TE1NTemkk05Kv/d7v5fWrVvXcd+MGTPSxRdf3On4tWvXprPPPjuddNJJaeTIkWnJkiWZJ86jlHW5+OKLU0R02WbMmJF/8AxK/Zr5bf01PlIqfV1eeeWV9JWvfCUNHjw4jRgxIs2ePTu99957mafOo9S1+cEPfpC+8IUvpMGDB6f6+vr0J3/yJ+ntt9/OPHVl/dd//ddHvm8M5PffUtdmIL0HF1IawOf/AIDs+vxnPgCAvkV8AABZiQ8AICvxAQBkJT4AgKzEBwCQlfgAALISHwAwQKxfvz6mT58eDQ0NUSgU4rHHHiv5OVJK8f3vfz/GjBkTxWIxGhsbY8GCBSU9R5+/sBwAcGwOHDgQ48aNi2uvvTa+8Y1v9Og5brnlllizZk18//vfj7POOitaW1tj9+7dJT2H33AKAANQoVCIRx99NK644oqOfQcPHozvfOc7sXz58ti7d2+MHTs2vve978XkyZMjIuKVV16JL33pS/E///M/8bnPfa7Hr+3bLgBARERce+218cwzz8TDDz8cv/jFL+IP/uAP4rLLLostW7ZERMS//du/xemnnx7//u//HqNGjYqRI0fGddddF++++25JryM+AIB444034qGHHopHHnkkLrroojjjjDPitttuiwsvvDCWLl0aERFbt26NN998Mx555JFYtmxZPPjgg7Fx48a46qqrSnotn/kAAOLnP/95pJRizJgxnfa3t7fHqaeeGhERhw8fjvb29li2bFnHcQ888ECMHz8+Xn311WP+Voz4AADi8OHDMWjQoNi4cWMMGjSo032f+tSnIiKivr4+qqqqOgXKmWeeGRERb731lvgAAI7d2WefHR988EHs2rUrLrroom6PueCCC+LQoUPxxhtvxBlnnBEREa+99lpERDQ1NR3za/lpFwAYIPbv3x+vv/56RPwmNhYvXhxTpkyJoUOHxmc/+9n40z/903jmmWdi0aJFcfbZZ8fu3bvjqaeeirPOOiu++tWvxuHDh+Occ86JT33qU3HPPffE4cOHY9asWVFTUxNr1qw55jnEBwAMEGvXro0pU6Z02T9jxox48MEH49e//nV897vfjWXLlsUvf/nLOPXUU+P888+PO++8M84666yIiHjnnXfi5ptvjjVr1sSQIUNi2rRpsWjRohg6dOgxzyE+AICs/KgtAJCV+AAAshIfAEBW4gMAyEp8AABZiQ8AICvxAQBkJT4AgKzEBwCQlfgAALISHwBAVuIDAMjq/wHypjwCD/KDwgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sf = CountSketchFactory(p)\n",
    "Sh = sf(SketchSizeParams(150))\n",
    "print(Sh.shape)\n",
    "\n",
    "# compute the sketch of the training data\n",
    "Xh_train = Sh @ X_train.T\n",
    "\n",
    "XX = Xh_train @ Xh_train.T\n",
    "w, v = np.linalg.eigh(XX)\n",
    "print(min(w))\n",
    "\n",
    "plt.hist(w, bins=100)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "X_train_norm = X_train\n",
    "X_test_norm = X_test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "def classification_error(y_true, y_pred):\n",
    "    return np.mean(y_true != np.sign(y_pred))\n",
    "\n",
    "def gcv_error(estimator, y, y_hat, error_metric):\n",
    "\n",
    "    div = estimator.gcv_div()\n",
    "    z = (y_hat - div * y) / (1 - div)\n",
    "\n",
    "    return error_metric(y, z)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "1/10: 100%|██████████| 2/2 [00:00<00:00,  5.49it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 1226.08it/s]\n",
      "2/10: 100%|██████████| 2/2 [00:00<00:00,  3.86it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 1047.57it/s]\n",
      "3/10: 100%|██████████| 2/2 [00:00<00:00,  4.38it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 844.62it/s] \n",
      "4/10: 100%|██████████| 2/2 [00:00<00:00,  5.05it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 1140.72it/s]\n",
      "5/10: 100%|██████████| 2/2 [00:00<00:00,  6.21it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 1273.53it/s]\n",
      "6/10: 100%|██████████| 2/2 [00:00<00:00,  3.73it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 1020.08it/s]\n",
      "7/10: 100%|██████████| 2/2 [00:00<00:00,  5.62it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 1030.32it/s]\n",
      "8/10: 100%|██████████| 2/2 [00:00<00:00,  6.39it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 1448.07it/s]\n",
      "9/10: 100%|██████████| 2/2 [00:00<00:00,  4.61it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 1378.35it/s]\n",
      "10/10: 100%|██████████| 2/2 [00:00<00:00,  5.31it/s]\n",
      "100%|██████████| 500/500 [00:00<00:00, 1200.34it/s]\n"
     ]
    }
   ],
   "source": [
    "# sketches = ['gaussian', 'orthogonal', 'countsketch', 'dct']\n",
    "# sketches = ['countsketch', 'dct']\n",
    "sketches = ['countsketch']\n",
    "q = 99\n",
    "n_ens = 5\n",
    "# Ks = [2, 5, 10]\n",
    "Ks = [2]\n",
    "rng = np.random.default_rng(0)\n",
    "n_trials = 10\n",
    "\n",
    "lamdas = np.concatenate([-np.logspace(2, 1, 101), np.linspace(-10, 10, 201)[1:], np.logspace(1, 3, 201)[1:]])\n",
    "\n",
    "cv_reg_errs = np.zeros((n_trials, len(sketches), len(Ks), len(lamdas)))\n",
    "cv_cls_errs = np.zeros((n_trials, len(sketches), len(Ks), len(lamdas)))\n",
    "test_reg_errs = np.zeros((n_trials, len(sketches), len(lamdas)))\n",
    "test_cls_errs = np.zeros((n_trials, len(sketches), len(lamdas)))\n",
    "gcv_reg_errs = np.zeros((n_trials, len(sketches), len(lamdas)))\n",
    "gcv_cls_errs = np.zeros((n_trials, len(sketches), len(lamdas)))\n",
    "\n",
    "for t in range(n_trials):\n",
    "\n",
    "    for i, sketch in enumerate(sketches):\n",
    "        sketch_transform = EnsemblePreSketchTransform(n_ens=n_ens, q=q, sketch=sketch, rng=rng)\n",
    "        XS_train = sketch_transform.fit_transform(X_train_norm)\n",
    "        XS_test = sketch_transform.transform(X_test_norm)\n",
    "\n",
    "        # perform k-fold cross validation\n",
    "        for k, K in enumerate(Ks):\n",
    "\n",
    "            y_hat_cv = np.zeros((len(lamdas), n_train))\n",
    "\n",
    "            for train_idx, val_idx in tqdm(KFold(n_splits=K, shuffle=True, random_state=rng.integers(0, 1000)).split(XS_train), total=K, desc=f'{t + 1}/{n_trials}'):\n",
    "                XS_cv_train = XS_train[train_idx, :]\n",
    "                y_cv_train = y_train[train_idx]\n",
    "                XS_cv_val = XS_train[val_idx, :]\n",
    "                y_cv_val = y_train[val_idx]\n",
    "\n",
    "                ens = PreSketchedEnsembleRidge(n_ens=n_ens, lamda=lamdas[0], which_ridge='svd').fit(XS_cv_train, y_cv_train)\n",
    "                y_hat_cv[0, val_idx] = ens.predict(XS_cv_val)\n",
    "\n",
    "                for j, lamda in enumerate(lamdas[1:]):\n",
    "                    ens.refit(lamda)\n",
    "                    y_hat_cv[j + 1, val_idx] = ens.predict(XS_cv_val)\n",
    "                \n",
    "            # compute the CV error\n",
    "            for j in range(len(lamdas)):\n",
    "                cv_reg_errs[t, i, k, j] = mean_squared_error(y_train, y_hat_cv[j, :])\n",
    "                cv_cls_errs[t, i, k, j] = classification_error(y_train, y_hat_cv[j, :])\n",
    "        \n",
    "        # compute the test error and gcv error\n",
    "        ens = PreSketchedEnsembleRidge(n_ens=n_ens, lamda=lamdas[0], which_ridge='svd').fit(XS_train, y_train)\n",
    "        y_train_hat = ens.predict(XS_train)\n",
    "        y_test_hat = ens.predict(XS_test)\n",
    "        test_reg_errs[t, i, 0] = mean_squared_error(y_test, y_test_hat)\n",
    "        gcv_reg_errs[t, i, 0] = gcv_error(ens, y_train, y_train_hat, mean_squared_error)\n",
    "        test_cls_errs[t, i, 0] = classification_error(y_test, y_test_hat)\n",
    "        gcv_cls_errs[t, i, 0] = gcv_error(ens, y_train, y_train_hat, classification_error)\n",
    "\n",
    "        for j, lamda in enumerate(tqdm(lamdas[1:])):\n",
    "            ens.refit(lamda)\n",
    "            y_train_hat = ens.predict(XS_train)\n",
    "            y_test_hat = ens.predict(XS_test)\n",
    "            test_reg_errs[t, i, j + 1] = mean_squared_error(y_test, y_test_hat)\n",
    "            gcv_reg_errs[t, i, j + 1] = gcv_error(ens, y_train, y_train_hat, mean_squared_error)\n",
    "            test_cls_errs[t, i, j + 1] = classification_error(y_test, y_test_hat)\n",
    "            gcv_cls_errs[t, i, j + 1] = gcv_error(ens, y_train, y_train_hat, classification_error)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cv[0,0]: 152\n",
      "test[0]: 74\n",
      "gcv[0]: 74\n"
     ]
    }
   ],
   "source": [
    "# detect when lamda is too negative and set to nan\n",
    "lamda_0_i = np.argmin(np.abs(lamdas))\n",
    "BIG_VAL = 1\n",
    "\n",
    "for i in range(len(sketches)):\n",
    "    \n",
    "    for j in range(len(Ks)):\n",
    "        cv_idx = np.max(np.argwhere(np.mean(cv_reg_errs[:, i, j, :lamda_0_i], axis=0) > BIG_VAL))\n",
    "        print(f'cv[{i},{j}]: {cv_idx}')\n",
    "        cv_reg_errs[:, i, j, :cv_idx] = np.nan\n",
    "        cv_cls_errs[:, i, j, :cv_idx] = np.nan\n",
    "\n",
    "    test_idx = np.max(np.argwhere(np.mean(test_reg_errs[:, i, :lamda_0_i], axis=0) > BIG_VAL))\n",
    "    print(f'test[{i}]: {test_idx}')\n",
    "    test_reg_errs[:, i, :test_idx] = np.nan\n",
    "    test_cls_errs[:, i, :test_idx] = np.nan\n",
    "\n",
    "    gcv_idx = np.max(np.argwhere(np.mean(gcv_reg_errs[:, i, :lamda_0_i], axis=0) > BIG_VAL))\n",
    "    print(f'gcv[{i}]: {gcv_idx}')\n",
    "    gcv_reg_errs[:, i, :gcv_idx] = np.nan\n",
    "    gcv_cls_errs[:, i, :gcv_idx] = np.nan\n",
    "\n",
    "to_save = {\n",
    "    'cv_reg_errs': cv_reg_errs,\n",
    "    'cv_cls_errs': cv_cls_errs,\n",
    "    'test_reg_errs': test_reg_errs,\n",
    "    'test_cls_errs': test_cls_errs,\n",
    "    'gcv_reg_errs': gcv_reg_errs,\n",
    "    'gcv_cls_errs': gcv_cls_errs,\n",
    "    'lamdas': lamdas,\n",
    "    'sketches': sketches,\n",
    "    'Ks': Ks,\n",
    "    'n_ens': n_ens,\n",
    "    'q': q,\n",
    "    'n_trials': n_trials,\n",
    "    'p': p,\n",
    "    'n_train': n_train,\n",
    "    'n_test': n_test\n",
    "}\n",
    "\n",
    "np.savez('results/rna-seq.npz', **to_save)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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MXDgQB588EHMZnOFxyh37doVs9nM7bffzurVq8ucj7Rx40a6d+/O+fPnWbhwIcuXL+fqq69m+PDhdsfbt28fffr04fz58yxZsoSFCxeya9cuXnzxxQq1qTTlvW5Wp0+fpmfPnuzZs4f58+fz0UcfceHCBR5//HGHspMnT2by5MkMGDCA5cuXM2rUKB5++GEOHTpkVy47O5uePXvywQcf8MQTT/Djjz/yzDPPsGTJEoYMGWL3AV7Zc9LpdDRu3Ni27eWXX2bEiBG0atWKzz//nI8++oisrCx69OjBvn37bOXeeecdHn30Udq1a8fXX3/NlClTmD59eqnzDd544w02bNjAnDlz+PHHH2nRokWFf6fjx49nwYIFPPHEE6xatYqPPvqIf/3rX6SlpdnKDBo0iN9//51Zs2axdu1aFixYQIcOHTh//nyp55+bm0vv3r358MMPGT9+PD/88AP33HMPs2bN4vbbb3co/8MPP/Dmm28yY8YMvvrqK4KDg7nttts4cuRIxV90ITyMXHOq5nK95uzYsYNz585xyy23OMyrqqhz584BMHXqVH744Qfef/99GjduTK9eveyuMUuXLmXMmDH07NmTb775hm+//ZZx48bZBYkVuX6UlJCQwKBBg/D29iYhIYGEhARuuukmp2WzsrK47rrrePvtt3nggQdYsWIFCxcupFmzZiQnJ9vKHT16lMcee4zPP/+cr7/+mttvv53//Oc/vPDCC7Yy33zzDY0bN6ZDhw62437zzTeltnPz5s3ccMMNZGRksGjRIj777DP8/f25+eabnd7QePjhh9Hr9Xz66afMmjWLTZs2cc8995S6f1EKt/a/iTrPOhxk69atakFBgZqVlaWuWrVKjYyMVK+//nq1oKDArnzPnj3V1q1bq6qqquvWrVMB9f/+7/9UVS19OMhPP/2kAurEiRNVVVVVs9msxsXFqQ0bNrQbTlIas9msPvbYY6pGo1EBVVEUtWXLluq4cePUxMREu7ItWrRQO3To4NDuwYMHq1FRUbahI8OHD1e9vb3VlJQUWxmTyaS2aNFCBRz2W5J1OEhKSopaUFCgpqenq59//rnq6+urjhgxwqF8w4YN1fvuu8/28zPPPKMqiqLu3r3brlzfvn3thoOcO3dO9fLyUocPH25XLiEhQQXshiDMnDlT1Wg0DkN7vvzySxVQV65cWeY5WYcgFhQUqAUFBWpqaqq6YMECVaPRqM8++6ytXFJSkqrT6dT//Oc/dvWzsrLUyMhI9Y477lBV1TJMJzIyUu3SpYtduWPHjql6vV5t2LChbZv1vdOkSRM1Pz/frnxFf6dt2rRRb7311lLPLzU1VQXUefPmlfk69OzZ0+51XbhwoQqon3/+uV056/CZNWvW2LYBakREhJqZmWnblpKSomo0GnXmzJllHlcITyDXHLnmqKqqLl26VAXUhQsXlnnel6KcIYgmk0ktKChQ+/Tpo95222227Y8//rhar169Mvdd3vVDVYt/B5cqbeh+ydd/xowZKqCuXbu2zGNcqrCwUC0oKFBnzJihhoSE2L13SxuCaP2beP/9923brr32WjU8PFzNysqybTOZTGqbNm3UBg0a2PZr/dscM2aM3T5nzZqlAmpycnKF2y5kCKKooGuvvRa9Xo+/vz8DBgwgKCiI5cuXo9OVnselT58+9OvXjxkzZjgdxmBlnQj94IMPApZu+/vvv59jx46xfv16WzmTyWT3UIvunimKwsKFCzly5Ajz58/ngQceoKCggNdee43WrVuzefNmAA4fPsyBAwe4++67HfY3aNAgkpOTOXjwIGC5a9mnTx8iIiJsx9dqtRWepG0VGRmJXq8nKCiIO+64g/j4eD744INy623cuJHWrVvTvn17u+133XWX3c9bt24lLy+PO+64w277tddeazeED+D777+nTZs2XH311Xbn3r9//wpnubp48SJ6vR69Xk9oaCijR49m+PDhvPTSS7Yyq1evxmQyce+999odx2g00rNnT9txDh48SEpKikPbY2Nj6d69u9PjDxkyxG4ieWV+p507d+bHH39k4sSJbNq0iZycHLt9BwcH06RJE2bPns3cuXPZtWsXZrO53Ndkw4YN+Pr6MmzYMLvt1uEll76HAXr37o2/v7/t54iICMLDwys8/EkITyDXHLnm1ISFCxfSsWNHjEYjOp0OvV7P+vXr2b9/v61M586dOX/+PCNGjGD58uWkpqY67Ke860d1/fjjjzRr1owbb7yxzHIbNmzgxhtvJDAwEK1Wi16v5/nnnyctLY0zZ85U+rgXL17kt99+Y9iwYbYEIWB5740cOZITJ07Y3qNWQ4YMsfu5Xbt2AHINqyQJwESFfPjhh2zfvp0NGzbw2GOPsX//fkaMGFFuvVdffZXU1NRS0wBnZWXxxRdf0LlzZ8LCwjh//jznz5/ntttuQ1EU24USsH3xtz5KXlQaNmzI6NGjWbRoEX///TfLli0jNzeXp59+GigeAz9hwgSHfY0ZMwbA9sGblpZGZGSkQ3udbSvLunXr2L59O6tXr2bo0KH89NNP/Oc//ym3XkWPbx3+cOlF26rkttOnT/Pnn386nLu/vz+qqjq96JTk7e3N9u3b2b59OytWrKBXr1589tlnvPLKK3bHAbjmmmscjrVs2TK717iibbeKiopyOCeo2O/0jTfe4JlnnuHbb7+ld+/eBAcHc+utt9pSWyuKwvr16+nfvz+zZs2iY8eOhIWF8cQTT5T5Zc76uyo5TCY8PBydTucwRCUkJMRhH15eXjV+QRficibXHAtPvebExsYCxUMoq2Lu3LmMHj2aLl268NVXX7F161a2b9/OgAED7D5vR44cyeLFizl27BhDhw4lPDycLl26sHbtWluZ8q4f1XX27FkaNGhQZplt27bRr18/wDJ3+pdffmH79u1MnjwZoErXkPT0dFRVdbi2AkRHRwOUew3z8vKq8vE9maShFxXSsmVL22Te3r17U1hYyHvvvceXX37pcOf/UldffTUjRoxg7ty5DBo0yOH5zz77jOzsbLZt2+Y0tfA333xDeno6QUFBbN++3e65uLi4Mtt8xx13MHPmTPbs2QNAaGgoAJMmTXI6NwegefPmgOUDxtnaKc62laV9+/a24/bt25f+/fvzzjvv8NBDD3HNNdeUWq+ix7d+EDqbmJ6SkmJ3RzI0NBRvb28WL17s9JjWdpZFo9HY3gdgOaf4+HimT5/O3XffTUxMjG0/X375JQ0bNix1X+W13ZmSQU5lfqe+vr5Mnz6d6dOnc/r0advdzJtvvtk2yblhw4a2L2CHDh3i888/Z9q0aeTn57Nw4cJSz+O3335DVVW79p05cwaTyVSh11UIYU+uORaees3p1KkTwcHBLF++nJkzZ1ZpHtjHH39Mr169WLBggd12ZzfUHnjgAR544AEuXrzITz/9xNSpUxk8eDCHDh2iYcOGFbp+VEdYWBgnTpwos8zSpUvR6/V8//33GI1G2/Zvv/22yscNCgpCo9HYzTOzsibWkGuYa0gPmKiSWbNmERQUxPPPP1/uMK0XX3yR/Px8pk+f7vDcokWL8Pf3Z/369WzcuNHuMXv2bPLy8vjkk08AywfypQ/rhcDZBwfAhQsXOH78uO0uTvPmzWnatCl//PGHw76sD+vQsN69e7N+/Xq7i0xhYWGFM2w5oygKb731FlqtlilTppRZtnfv3uzdu5c//vjDbvunn35q93OXLl3w8vJyaNfWrVsdhgMMHjyYf/75h5CQEKfnXnL4SEV4eXnx1ltvkZuba5ss3r9/f3Q6Hf/880+przNYfh+RkZEO2cCSkpL49ddfK3T8yvxOLxUREcH999/PiBEjOHjwINnZ2Q5lmjVrxpQpU2jbti07d+4stQ19+vThwoULDhfBDz/80Pa8EKJ65JpTeZfzNUev1/PMM89w4MABuwQTlzpz5gy//PJLqftQFMXWO2P1559/kpCQUGodX19fBg4cyOTJk8nPz2fv3r0OZSpy/aisgQMHcujQIds6m85YFybXarW2bTk5OXz00UcOZSs6qsLX15cuXbrw9ddf25U3m818/PHHNGjQoNwELqJqpAdMVElQUBCTJk3iv//9L59++mmZGXDi4uIYPXo0r7/+ut32PXv2sG3bNkaPHu108czu3bvzv//9j0WLFjnNxGT10ksv8csvvzB8+HCuvvpqvL29SUxM5M033yQtLY3Zs2fbyr799tsMHDiQ/v37c//991O/fn3OnTvH/v372blzJ1988QUAU6ZM4bvvvuOGG27g+eefx8fHh7feeqta6doBmjZtyqOPPsr8+fP5+eefS02hO3bsWBYvXsxNN93Eiy++SEREBJ988onDnbbg4GDGjx/PzJkzCQoK4rbbbuPEiRNMnz6dqKgou3TIY8eO5auvvuL6669n3LhxtGvXDrPZTFJSEmvWrOGpp56iS5culT6nnj17MmjQIN5//30mTpxIXFwcM2bMYPLkyRw5csQ2f+P06dNs27bNdidRo9Ewffp0HnvsMYYNG8aDDz7I+fPnnba9LBX9nXbp0oXBgwfTrl07goKC2L9/Px999BFdu3bFx8eHP//8k8cff5x//etfNG3aFIPBwIYNG/jzzz+ZOHFiqce/9957eeutt7jvvvs4evQobdu25eeff+bll19m0KBB5Y7pF0KUT645VXM5X3Oefvpp9u/fz9SpU9m2bRt33XWXbSHmn376iXfeeYfp06eXOmd48ODBvPDCC0ydOpWePXty8OBBZsyYQVxcHCaTyVbukUcewdvbm+7duxMVFUVKSgozZ84kMDDQ1mtY3vWjusaOHcuyZcu45ZZbmDhxIp07dyYnJ4fNmzczePBgevfuzU033cTcuXO56667ePTRR0lLS2POnDkOQSZA27ZtWbp0KcuWLaNx48YYjUbatm3r9NgzZ86kb9++9O7dmwkTJmAwGJg/fz579uzhs88+q3IWSlEOd2YAEXVfaYtiqqqq5uTkqLGxsWrTpk1Vk8mkqqp9RqpLnT17Vg0ICLDLSDV27FgVcMi6dKmJEyeqgPr777+XWmbr1q3qv//9b7V9+/ZqcHCwqtVq1bCwMHXAgAFOsyz98ccf6h133KGGh4erer1ejYyMVG+44QaHbEu//PKLeu2116peXl5qZGSk+vTTT6vvvPNOpTJSlVwUU1VV9fTp06qfn5/au3dv27aSGZFUVVX37dun9u3bVzUajWpwcLD60EMPqcuXL3dYFNNsNqsvvvii2qBBA9VgMKjt2rVTv//+e7V9+/Z2mZ5UVVUvXLigTpkyRW3evLlqMBjUwMBAtW3btuq4cePssm85U1o2J1VV1b/++kvVaDTqAw88YNv27bffqr1791YDAgJULy8vtWHDhuqwYcPUdevW2dV955131Kuuuko1GAxqs2bN1MWLF6u33HKL2qFDB1uZshZUVdWK/U4nTpyodurUSQ0KClK9vLzUxo0bq+PGjVNTU1NVVbX8Xu6//361RYsWqq+vr+rn56e2a9dOfe2112zvb1V1zIKoqqqalpamjho1So2KilJ1Op3asGFDddKkSQ4LRQPqv//9b4f2O/v9C+GJ5Joj15ySli9frt50001qWFiYqtPp1KCgILV3797qwoUL1by8PFs5SmRBzMvLUydMmKDWr19fNRqNaseOHdVvv/1Wve++++yy7H7wwQdq79691YiICNVgMKjR0dHqHXfcof7555+2MuVdPy79HVyqolkQVVVV09PT1SeffFKNjY1V9Xq9Gh4ert50003qgQMHbGUWL16sNm/e3NaGmTNnqosWLXJ4jxw9elTt16+f6u/vrwK283WWBVFVVXXLli3qDTfcoPr6+qre3t7qtddeq65YscKuTGl/m+5crPtypqhqFRf/EcIDLVmyhAceeIDExMQqDdmrLYmJibRo0YKpU6faLSZ5OTh//jzNmjXj1ltv5Z133nF3c4QQwm3kmiPElUmGIApxmfvjjz/47LPP6NatGwEBARw8eJBZs2YREBDAQw895O7mlSklJYWXXnqJ3r17ExISwrFjx3jttdfIysriySefdHfzhBBClHA5X3OEqCskABPiMufr68uOHTtYtGgR58+fJzAwkF69evHSSy+Vms69rvDy8uLo0aOMGTOGc+fO4ePjw7XXXsvChQtp3bq1u5snhBCihMv5miNEXSFDEIUQQgghhBCilrg9Df38+fOJi4vDaDQSHx/Pli1bSi2bnJzMXXfdRfPmzdFoNIwdO9Zpua+++opWrVrh5eVFq1at+Oabb1zUeiGEEJebylx3ADZv3kx8fDxGo5HGjRs7rAn37rvv0qNHD4KCgggKCuLGG29k27Zt1T6uEEKIK5NbA7Bly5YxduxYJk+ezK5du+jRowcDBw4kKSnJafm8vDzCwsKYPHky7du3d1omISGB4cOHM3LkSP744w9GjhzJHXfcwW+//ebKUxFCCHEZqOx1JzExkUGDBtGjRw927drFs88+yxNPPMFXX31lK7Np0yZGjBjBxo0bSUhIIDY2ln79+nHy5MkqH1cIIcSVy61DELt06ULHjh3tVilv2bIlt956KzNnziyzbq9evbj66quZN2+e3fbhw4eTmZnJjz/+aNtmXYfos88+q9H2CyGEuLxU9rrzzDPP8N1337F//37btlGjRvHHH3+UuqBrYWEhQUFBvPnmm9x7771VOq4QQogrl9uScOTn5/P77787LHDar18/fv311yrvNyEhgXHjxtlt69+/v0Ogdqm8vDzy8vJsP5vNZs6dO0dISIgsQCeEELVMVVWysrKIjo6u8ILcFVGV605CQgL9+vWz29a/f38WLVpEQUEBer3eoU52djYFBQUEBwdX+bgg1yYhhKgravq65LYALDU1lcLCQoeMOREREaSkpFR5vykpKZXe58yZM5k+fXqVjymEEKLmHT9+nAYNGtTY/qpy3SntmmIymUhNTSUqKsqhzsSJE6lfvz433nhjlY8Lcm0SQoi6pqauS25PQ1/yLp6qqtW+s1fZfU6aNInx48fbfs7IyCA2Npbjx48TEBBQ5XaMXtKZXQYNd+YqjH3Q+VCVK1WbqasB+Pyxa2kVHejm1ggh6oIfXhvFTbnfc19YDAd9VAbFDeL5rs87lMvMzCQmJgZ/f3+XtKOy1whn5Z1tB5g1axafffYZmzZtwmg0Vuu4NX1tWrH2v7x09ifbz1vv2lrpfVyuHnx/O9uOniMu1JcV/7nO3c0RQlxmavq65LYALDQ0FK1W63D378yZM9VaRyIyMrLS+/Ty8sLLy8the0BAQLUCMIO3Fq1Bg1FRqrWfy5HGywcAHz9/jzt3IYRzQWQQ4KVwLkCHVm+mTf02ZX4+1PQwu6pcd0q7puh0OkJCQuy2z5kzh5dffpl169bRrl27ah0Xav7a5OPrhfaC1m4/nsLbzw+NVy4aLx+POm8hRM2qqeuS27IgGgwG4uPjWbt2rd32tWvX0q1btyrvt2vXrg77XLNmTbX2WVXFvyLPXWqt0Oy55y6EsBdQeB4zcF5nBqBtWNtaPX5VrjulXVM6depkN/9r9uzZvPDCC6xatYpOnTpV+7iiZmmKvjSZ5ZokhKgD3DoEcfz48YwcOZJOnTrRtWtX3nnnHZKSkhg1ahRgGX5x8uRJPvzwQ1ud3bt3A3DhwgXOnj3L7t27MRgMtGrVCoAnn3yS66+/nldffZVbbrmF5cuXs27dOn7++edaPz9rCObJH/dyrRNCWIVynjNaLeaiu1OtQ1rXehsqe90ZNWoUb775JuPHj+eRRx4hISGBRYsW2WXVnTVrFs899xyffvopjRo1svV0+fn54efnV6HjCtfSaYoCMPclfhZCCBu3BmDDhw8nLS2NGTNmkJycTJs2bVi5ciUNGzYELAsvl1wjpUOHDrb///7773z66ac0bNiQo0ePAtCtWzeWLl3KlClTeO6552jSpAnLli2jS5cutXZeopgbVzkQQtQh2bn5RClpHNJZLjtaRYtRZyynVs2r7HUnLi6OlStXMm7cON566y2io6N54403GDp0qK3M/Pnzyc/PZ9iwYXbHmjp1KtOmTavQcYVraYsCsEK5JAkh6gC3J+EYM2YMY8aMcfrckiVLHLZV5Av9sGHDHC6Ewj1kCKIQAmDv/j1co+RwQu8LgG/Rv+5Q2etOz5492blzZ6n7s94ArM5xhWvptEUjUuSmoBCiDnDbHDBP4skf93KtE0IAnDrwGwCHdJbAK9gY7M7meCYPXjvM1gMmNwWFEHWABGAu5LmXumIy3l4IAZCd/DcAR3SWrH7RftHubI7wMLqihVMl/hJC1AUSgAmXKpQATAgB6C4mA3BCb7nsNApo5MbWCE8jSTiEEHWJBGCuJF1gkvJXCAGAX8E5AM4UZW5vE9rGja3xTJqSFyWz2T0NcQNrACZzwIQQdYEEYMIlrJd5yTglhADLIswXFIVsreVDoVNEp3JqCJdTPScA09p6wNzcECGEQAKwWqF6cBoOudsohDifnU8kaSTpLYl3FRSi/KLc3CrhSXRay9cduSYJIeoCCcBcSEYgysVOCAEJB09SX0kjSW8Zf+ij83FzizxUySyIHtQDZk1DLz1gQoi6QAKwWuDJn/eFnnN9F0KU4sT+beiVQv7RGQAI9pYU9HWCBwVgeo30gAkh6g4JwFzIo3vAik7e7NHhpxACIP/UXgAO6r0BqO9X353N8VgO1yQPCsBkDpgQoi6RAKw2eOAHvi0Jh2ThEMLjeV08AcBRg2UI4lX1rnJnc4SVBwVgel1RFkRPvCALIeocCcBcSrH7xxPJOmBCiHqmVABSLDk4aBva1o2tEZ5IZxuC6OaGCCEEEoAJF1GKos48k+fcYRVCODKbzYRzjkyNQo7Wsq1TpKSgrxM8qAfMoLWuA+bmhgghBBKA1QpPHPJgTbZVIFk4hPBoR85eJEY5w3GdZfihgkK4T7ibW+WZVA/Ogqi19oB54PVYCFH3SADmQh488tAWgOVLD5gQHm3b3yeIUc5yrGgNMB+9pKB3F6XkVcmDAjDpARNC1CUSgNUCT/y8V4oiMJP0gAnh0U4f+h2dYuYfnRcAIcYQN7dIeCKtdSFmN7dDCCFAAjDhItY3Vr5kQRTCo6lnDwBw0GBJQd/Av4E7myMu5YE9YNIFJoSoC3TubsCVzLOHIFrOXuaACeE5zmTmciYrz26b98WToIGjRUMQJQV9HeJBwYhOesCEEHWIBGAu5Mkf9DIHTAjP88lvSby+/m+7bbN0qaCB05YcHLQLbeeGlgmnPKkHTCcBmBCi7pAAzIVsPWAe+ImvkR4wITzO3V1i6dsqgtyCQoYtTAAgSjlHhkZDbtG4ZElB7z4aDx6WobeevAdej4UQdY8EYLXAEz/vNbY09J549kJ4pvAAI+EBRrLzTbZtDZXTtgyIGjSEeEsSDvfx3CyIehmCKISoQyQAEy6h0UgWRCE8TlYKZKWgmApprSSSr2qpr6Tyh86SgMOn6F9RR3hgACaEEHWBBGAu5MGjPYqHIJrlfqMQHmPH+7D5FbyBH7xgmaknWkXlH31RCnpJvFu3eFAAptN68hVZCFHXSABWC1QPHPRgHYIoPWBCeJBOD0DzgeTkXMD7o5u4iBGAv/WWfxsEN3dn60RJHhSAeekk+BdC1B0SgAmXKE7C4XnBpxAeyz8S/CNRL2TYbU40WC41zcLauqNVoojiweMyZAiiEKIukU8kl7Jc7DwxBNFa54CZPecOqxDCotCsklDYEm/yUYGUohT08RHxbm2XKMlzrk6GSwIwkyyPIoRwMwnAaoEn3nO0BWDSAyaER1m1J5kb/287IwqeI0Y5w2mtlgJFQVXhmqhr3N08z6aUuBp50A0y/SVDEPM96LyFEHWTBGAu5ImBl5VtDphc6ITwGKv2JDP6452czsoHoJHmNEf0lu4vc34YPx3IKKu6qHWec4Ps0h6wAukBE0K4mQRgtcBzLnHFiocgeuLZC+F5Cs0qC7/bTCslkdZKIh2UQ0STapv/5Zfvy8LvNlNYBz4T5s+fT1xcHEajkfj4eLZs2VJm+c2bNxMfH4/RaKRx48YsXLjQ7vm9e/cydOhQGjVqhKIozJs3z2Ef06ZNQ1EUu0dkZGRNnpYog15XfEs0TwIwIYSbSRIOF/LsHjBLbF8oQxCF8Ai79/zF0rzHMXoV2G239oDdXfgHj+UlsHtPK+LbtXNHEwFYtmwZY8eOZf78+XTv3p23336bgQMHsm/fPmJjYx3KJyYmMmjQIB555BE+/vhjfvnlF8aMGUNYWBhDhw4FIDs7m8aNG/Ovf/2LcePGlXrs1q1bs27dOtvPWq225k+wDA7XJA8aoeClK36tCzzovIUQdZMEYLXAE0MQ62gPWQdMCM+Qde40RqXAYXtiUQAWV2DCqBSQde50bTfNzty5c3nooYd4+OGHAZg3bx6rV69mwYIFzJw506H8woULiY2NtfVqtWzZkh07djBnzhxbAHbNNddwzTWW+W0TJ04s9dg6na6O9Xp5TiCi1xSHn/kFnnPeQoi6SYYgupAnhx7WIYiFcqdRCI8Q7GNwut3aA9Y4v6DMcrUhPz+f33//nX79+tlt79evH7/++qvTOgkJCQ7l+/fvz44dOygocAw4y/L3338THR1NXFwcd955J0eOHKncCdQ0D7pBZrgkCYcMQRRCuJsEYC7kyUMQdTIHTAiP0rp+gMO2DI1CWtHQr7iiYMVZudqSmppKYWEhERERdtsjIiJISUlxWiclJcVpeZPJRGpqaoWP3aVLFz788ENWr17Nu+++S0pKCt26dSMtLa3UOnl5eWRmZto9qsdzL/kaTfG5m1W5Lgkh3MtzP42FS2k1MgdMCE+iLZninOLhh+EmE75FX3qdlattSok2qKrqsK288s62l2XgwIEMHTqUtm3bcuONN/LDDz8A8MEHH5RaZ+bMmQQGBtoeMTExFT5exXhmT1BBoWeetxCi7pAAzIUUD+4Ds/aAFcqdRiE8ljUAa1zJoXquEhoailardejtOnPmjEMvl1VkZKTT8jqdjpCQkCq3xdfXl7Zt2/L333+XWmbSpElkZGTYHsePH6/y8cDJqAzVMwORArkxKIRwMwnAXEgt8a8nKZ4D5olnL4SASwKwfJObW2JhMBiIj49n7dq1dtvXrl1Lt27dnNbp2rWrQ/k1a9bQqVMn9EXnVxV5eXns37+fqKioUst4eXkREBBg96hRHhqAyXVJCOFuEoAJl7D1gHnm9V0IARwxWDMg1o0eMIDx48fz3nvvsXjxYvbv38+4ceNISkpi1KhRgKXX6d5777WVHzVqFMeOHWP8+PHs37+fxYsXs2jRIiZMmGArk5+fz+7du9m9ezf5+fmcPHmS3bt3c/jwYVuZCRMmsHnzZhITE/ntt98YNmwYmZmZ3HfffbV38iV5aAAmQxCFEO4maehdyDrcwxPvtWm1kgVRCI/iEwI6LzDlAbDK14ct3kbgkiGIOi9LOTcaPnw4aWlpzJgxg+TkZNq0acPKlStp2LAhAMnJySQlJdnKx8XFsXLlSsaNG8dbb71FdHQ0b7zxhi0FPcCpU6fo0KGD7ec5c+YwZ84cevbsyaZNmwA4ceIEI0aMIDU1lbCwMK699lq2bt1qO66oPZIcSgjhbhKAuZLnTgGz9YCZ5UInhGeoFwOP/86ew0c4sGIa/ws5Y0vxoBk8D4JbWYKvejWdSKLyxowZw5gxY5w+t2TJEodtPXv2ZOfOnaXur1GjRrbEHKVZunRppdroEpoSFyUPm6OrYLkhKkMQhRDuJkMQXciD4y/01iyIcp0TwnPUi+HH1AhWhJ8jW6NAUZbAj89ug+ir60Tw5ck0Ja9KHjoE0SQXJiGEm0kAVgtUDxyEqCsagijrrQjhWTadXMtOPzPmS1K0r09az6qjq9zYKuGchwVgRW9JmQMmhHA3CcCES+i0RT1gMtRDCI+RlpPGcc0HKCVuvCgozEiYQVpO6YsOC+Fq1lsCsjyKEMLdJABzIVkHTHrAhPAUqqoyZct0VCUfteTCxahczL/Icz9Pd1PrhFMedoPM+q40SQ+YEMLNJAATLiE9YEJ4lsPnD/Nz8kZQnP/NmzGz5dRGDqcfdvq8qA0lL/meFohYQjDJgiiEcDcJwIRL6It6wKQDTAjPcFW9q2hHg9LiL7SqSi+v+lwVdFXtNkyUzsOScFg7ZiUAE0K4mwRgtcATP+oNOstbS4YgCuEZFEWhgfEJjGaczAEDX70v0/q85p7GCXEJGZkhhHA3CcBcyHNngIHelgXRzQ0RQtSaA2f8+G/qBSdzwOC57tMJCWvpnoYJi5I9Xp7aAyZp6IUQbiYBWC3wxI96nVYLSA+YEJ4kN+04Q7PP0Si/wLZNq2i5MfZGBjQa4MaWCYsSn8ce9vlsTYxVaPaswFMIUfdIAOZCnnVps2ftAfOw67sQHi0i/xgKEGYq/oLro/NhyrVT3NcoYaMp+XnsoT1gkgRRCOFuEoC5kGcPQZQ5YEJ4kn2nMohTkgE4ZtDZtk/qMokQ7xB3NUvYKdkD5lmRiC0NvfSACSHcTAIw4RIG6QETwqOs2Xea5spxchSFM9riS0uf2D5ubJUQl5AsiEKIOsLtAdj8+fOJi4vDaDQSHx/Pli1byiy/efNm4uPjMRqNNG7cmIULFzqUmTdvHs2bN8fb25uYmBjGjRtHbm6uq06hVJ7cA2bQWeaAqR49EFMIz/H7sXSaak5wRK8vHusl6hYPvyNmnQMmSTiEEO7m1gBs2bJljB07lsmTJ7Nr1y569OjBwIEDSUpKclo+MTGRQYMG0aNHD3bt2sWzzz7LE088wVdffWUr88knnzBx4kSmTp3K/v37WbRoEcuWLWPSpEm1dVoOPDEIsQ5B9PDrvRAe4++UTJorJ/jboLfbfuDcAfal7WNf2j7OZp91U+uEM+ZCk7ubUKus9wVkaLwQwt105Rdxnblz5/LQQw/x8MMPA5aeq9WrV7NgwQJmzpzpUH7hwoXExsYyb948AFq2bMmOHTuYM2cOQ4cOBSAhIYHu3btz1113AdCoUSNGjBjBtm3bauekBCBDEIXwNPqLpwgwZHPIUM9u+32r7rP9f3T70Yy5ekwtt0xYleyYNKuF7h8GU4uKk3DIHDAhhHu5LQDLz8/n999/Z+LEiXbb+/Xrx6+//uq0TkJCAv369bPb1r9/fxYtWkRBQQF6vZ7rrruOjz/+mG3bttG5c2eOHDnCypUrue+++5zu05Wswx08MQbRFy3E7Im9f0J4muTzOTTiFAB7DV4A9Kjfg8c7PG5XLsw7rNbbJi5R4o6YSTW79y5sLdNYhyBK/CWEcDO3ffampqZSWFhIRESE3faIiAhSUlKc1klJSXFa3mQykZqaSlRUFHfeeSdnz57luuuuQ1VVTCYTo0ePdgj0LpWXl0deXp7t58zMzGqcWTHrpc4TZ0MYbAGYEOJKZ03AAfBPUQbE3jG9aRXSyp3NEiUoJQIws7nQTS1xj+IeMLkyCSHcy+2jD5QSYyJUVXXYVl75S7dv2rSJl156ifnz57Nz506+/vprvv/+e1544YVS9zlz5kwCAwNtj5iYmKqejn1ba2QvlyeDNQuaXOeEuOIl/JNKS00S5zQaMooWYe8Z09PNrRLl87QAzLoQs1yYhBDu5bYALDQ0FK1W69DbdebMGYdeLqvIyEin5XU6HSEhlnVmnnvuOUaOHMnDDz9M27Ztue2223j55ZeZOXMm5lLGfU+aNImMjAzb4/jx4zVwhsU88aPeS3rAhPAY+1OyaKkkcbAoAYdO0RHuE+7mVgkHDj1gnjUWT+aACSHqCrcFYAaDgfj4eNauXWu3fe3atXTr1s1pna5duzqUX7NmDZ06dUKvt1z4s7Oz0WjsT0ur1aKqqq23rCQvLy8CAgLsHqJ6rEMQhRBXvnMZWVylnOCQwQBAsHewm1sknCn5qaziWYFI8ULMcmtQCOFebv2WPH78eN577z0WL17M/v37GTduHElJSYwaNQqw9Ezde++9tvKjRo3i2LFjjB8/nv3797N48WIWLVrEhAkTbGVuvvlmFixYwNKlS0lMTGTt2rU899xzDBkyBG3R0JjaYv2w98SPei8JwITwCBnZ+dQvPIFBKWSvlyUAiwuIc3OrhHMlesA8LE2tRpHsvEKIusGtCZCGDx9OWloaM2bMIDk5mTZt2rBy5UoaNmwIQHJyst2aYHFxcaxcuZJx48bx1ltvER0dzRtvvGFLQQ8wZcoUFEVhypQpnDx5krCwMG6++WZeeumlWj8/T6aXAEwIj7B632laKpbP6T1elgyIV4df7cYWiVKViDxU1dPWAbNmQZQITAjhXm7PQDtmzBjGjHG+LsySJUsctvXs2ZOdO3eWuj+dTsfUqVOZOnVqTTWxGor6wDzws95LV9zbmG8yy5BEIa5Qmw+eoZ0miTwFThb93V/f4Ho3t0o459lzwDSSBVEIUUfIt+La4IHpEP29imP7C7medZdVCE+y52QmLZUk/tHrMSsKCgptQtq4u1nCKbWMn6581iQcnjb0UghR90gAJlwiwFgcgGXlFbixJUIIV0rOyKWl5hgHixJw+Bv8HRIhibpBKXE30NPWAdNIGnohRB0hV0kX8uQkHEbDJQGY9IAJcUU6k5lLYGEaYUomB4oCsPp+9d3cKlEq1X7IoSrrgAkhhFtIAOZKHjj00BkZgijElenHPSm01FgScPxVlAGxdUhrdzZJlKVkEg6ZAyaEEG4hAZgLeXr8ZT3/zFwZgijElWjL32dpqSShAn8XLcJ8bfS17m2UKFXJtTBNqmf1gGlkDpgQoo6QAKwWeOpHvXXC88U8z7rIC+Ep9idn0VJzjJM6LblF8756RPdwc6vKN3/+fOLi4jAajcTHx7Nly5Yyy2/evJn4+HiMRiONGzdm4cKFds/v3buXoUOH0qhRIxRFYd68eTVy3JrmuPCyZ12dbHPAJAATQriZBGAupHp4H5j1YndRknAIcUU6nZlLG+Uo+4rmf3lpvfAx+Li5VWVbtmwZY8eOZfLkyezatYsePXowcOBAuzUnL5WYmMigQYPo0aMHu3bt4tlnn+WJJ57gq6++spXJzs6mcePGvPLKK0RGRtbIcV1BLTkHzMOScFjngJllCKIQws0kAHMhzw6/QFs03uNivmdd5IXwBMfTsjGYc2isJLOvaP5XlG+Um1tVvrlz5/LQQw/x8MMP07JlS+bNm0dMTAwLFixwWn7hwoXExsYyb948WrZsycMPP8yDDz7InDlzbGWuueYaZs+ezZ133olX0WLU1T2uK5SMOzxtKJ71pqDEX0IId5MATLhMcQ+YJOEQ4kqz/I+TtFKOolFU/igKOlqFtHJzq8qWn5/P77//Tr9+/ey29+vXj19//dVpnYSEBIfy/fv3Z8eOHRQUVKx3vyrHBcjLyyMzM9PuUR0lhyCaVc/6bNYWfeORJBxCCHeTAMyFitPQe+aHvU5reQWyC6QHTIgrzZa/U2mjOYoKth6w6+pf595GlSM1NZXCwkIiIiLstkdERJCSkuK0TkpKitPyJpOJ1NRUlx0XYObMmQQGBtoeMTExFTpeaQod0tB7Fus6aJ7W8yeEqHskABMuYx2CmCNDEIW44hw6nUVbTSIndVqyixJw9I7p7eZWVYx1LpCVqqoO28or72x7TR930qRJZGRk2B7Hjx+v1PEclQjAVA9LQy89YEKIOkJXfhEhqkZXdLWTAEyIK4vJZCY9u4DWhqPsLUrAYdQa8TP4ubllZQsNDUWr1Tr0Op05c8ahd8oqMjLSaXmdTkdISIjLjgvg5eVV6pyyqigZd3hcACZzwIQQdYT0gLmQ7b6mh37Y64uGIObIEEQhriibDp3BSB5NlRPFCTj86n4CDoPBQHx8PGvXrrXbvnbtWrp16+a0TteuXR3Kr1mzhk6dOqHX6112XFcoGXCZPSwAs47KkCGIQgh3kx4w4TKGohnPuRKACXFFWb33NC2VJLSKyp9FPTStQ1q7uVUVM378eEaOHEmnTp3o2rUr77zzDklJSYwaNQqwDPs7efIkH374IQCjRo3izTffZPz48TzyyCMkJCSwaNEiPvvsM9s+8/Pz2bdvn+3/J0+eZPfu3fj5+XHVVVdV6Li1wVzibqCnzU+2jvaUIYhCCHeTAMylLJ/2qofmo9frLAFYnsmz7rIKcaX7/Vg63TWJdgk4utfv7t5GVdDw4cNJS0tjxowZJCcn06ZNG1auXEnDhg0BSE5OtlubKy4ujpUrVzJu3DjeeustoqOjeeONNxg6dKitzKlTp+jQoYPt5zlz5jBnzhx69uzJpk2bKnTc2mBW7W+GqR4WiGiLIjBVesCEEG4mAZhwGUPREMQ8k/SACXElOZGeQxvlKCcuScBxQ4Mb3NyqihszZgxjxoxx+tySJUsctvXs2ZOdO3eWur9GjRpV6Et9WcetDSXbWDIgu9IVD0F0c0OEEB5P5oC5kof2fFkZdFoAcgukB0yIK8XJ89nkF5ppq0lkb9HwQ6PWiI/Bx80tE+VxDMA867O5OAmHRGBCCPeSAMyFitcB80xGveXtVVDoWRd5Ia5ky3efwkgezZTj7Ckafljfv76bWyUqwlwyDT2e9dlsDcDkkiSEcDcJwITLeBX1gOXLHDAhrhibDp6ltXIUnWJmZ1EPWPuw9m5ulaiIkj1gnpaGvigvlMwBE0K4nQRgLuThIxAx6qw9YHKxE+JKcSA5k6s1hykADhT1gN0Ye6N7GyUqpGQWRE8biqeRNPRCiDpCAjDhMt4GS44XGYIoxJXhQq6JzFwTV2v+4ZBBT0HRkK6u0V3d3DJRESXTzsscMCGEcA8JwITLeMscMCGuKCv+PAVAR+Vv/ioafljPqx46jSTUvRyUHHqneFgAVpyG3s0NEUJ4PAnAXKrow97NrXAXb4NlDpgseinElWHN3hRCyKC+Jo0/igKwJvWaVGofJpOJ6dOnc/z4cVc0UZShZI+XZ4VfkoZeCFF3SAAmXMY6BNEkVzshrgh/ncygneYIALuMlgCsc2TnSu1Dp9Mxe/ZsCgs9aw2quqDkJ7GnDUHUyhwwIUQdIQGYCyke3gPm6yU9YEJcKUwmM2kX8rlac5gMjYaTessNlpvibqr0vm688UY2bdpUwy0U5XEIuDwsACuKvyQAE0K4nQzcrwWemg3Rz8vy9pIATIjL39oDp1Gxzv+yZD/UKToaBjas9L4GDhzIpEmT2LNnD/Hx8fj6+to936tXrxposSjJIQmHhw1C1Gos95wl/hJCuJsEYLXAUz/rQ30tQ5QK5WonxGXvhz+TAZUOmsN8VBSAhfuGV2lfo0ePBmDu3LkOzymKQnp6epXbKUqXHd4cjh22/VwyLf2VTuaACSHqChmC6EKe2vNlFeJn+ZIm8ZcQl7+dx9KJU1LwU3JtCzC3DmldpX2ZzeZSHzI3zHVy/SPsflbNntUDptNYsyDKRUkI4V7SAyZcJtTfy/b/3HwTRoO83YSo685k5nImK89um9lsJjkjl2HagxQCu4sScPRt2NcNLRRVViLu8LQesKIs9B521kKIuki+EdcCT/2wD78kADt7MZ8YCcCEqPM++S2J19f/7fS5a5QD/G3Qk1s0l6ZPTJ8qH2fz5s3MmTOH/fv3oygKLVu25Omnn6ZHjx5V3qcom05r/xmselgSDp1kQRRC1BEyBFG4jL9Rb/t/aok76kKIuunuLrF8/5/r+HJUV9u2zo2CAeim3WcbfljPqx4GnaFKx/j444+58cYb8fHx4YknnuDxxx/H29ubPn368Omnn1b/JIRTj7R9BH+Dv+1nTwvAJAmHEKKukC4JF7LOAfPkz3oFy/mflQBMiMtCeICR8AAj2fkm27YjqReoRxYNlFR2GUMAaBbUrMrHeOmll5g1axbjxo2zbXvyySeZO3cuL7zwAoMHD676CYhSGXVGfh3xKz0Wt+K8VuuQFfFKp5E5YEKIOkJ6wGqBJyfjsF7w0i7mu7klQoiqSr2QT7zmECqw3WgEoGeDnlXe35EjR7j55psdtg8ZMoTExMQq71dUjO3moIcFIrYkHG5uhxBCSADmUp4cellYL3jnJAAT4rLWRXOAZJ2WNJ1lgfWbGld+AWarmJgY1q9f77B9/fr1xMTEVHm/onI8bQiirugbj6cFnkKIukeGILqQDEEEvVZDnsnM+WwJwIS4nJRcQL2H9i/b/C9vnTch3iFV3vdTTz3FE088we7du+nWrRuKovDzzz+zZMkSXn/99Wq1W5TPU3vArHPAZB0wIYS7SQAmXMqg00AenM8pcHdThBAVtGpPMlO/22v7WY+JOJL50ugHQFxAXLX2P3r0aCIjI/nf//7H559/DkDLli1ZtmwZt9xyC5mZmdXav6gYT0tDr1FkVIoQom6QAKwWeNYlzp5X0ZiPLAnAhLgsrNqTzOiPd9p9brVREjEqBWw3egNwbdS1Vd6/yWTipZde4sEHH+Tnn3+uZmtFVSi2X65nXZ10WmsSDjc3RAjh8WQOmHApb71lvkhWrqmckkIIdys0q0xf/pfD1/JrNAc4p9GQWLSW3+AmQ6p8DJ1Ox+zZsyksLKxGS0V1FA9B9LQ5YJIFUQhRN0gAJlzK22AJwC7kSwAmRF23LfEcyVmOvdXXavazw2iZ/1WYG0HquXrVOs6NN97Ipk2bqrUPUX1mPCsA00gWRCFEHSFDEF1IkfHm+HpZ3mLZeXK3W4i67kxWrsM2LYV00hzk/7wt6ecLs5s4LVcZAwcOZNKkSezZs4f4+Hh8fX3tnu/Vq1e19i/KZr0yeVoyCr1GhiAKIeoGCcCES/kXBWA5BRKACVHXhfsbHba1Uo4RoOSw1RgEQGF2Y6flKmP06NEAzJ071+E5RVFIT0+v1v5FeSwRiKcNQbRmQfS0BaiFEHWPBGAuJGnooZ6PHoDsfAnAhKjrOscFc3VAFqasVNvn1i2aX0jTaDhaNP+ruSaEznHB1TqO2Vz2F3/Jguhaxdcmz7o6SRIOIURdIQGYC6mKhGARAZY75bnSAyZEnafNPMFXpv+g9bJft2+Vtw8AzfLy+bxgEtrM/lCvagsmm0wmjEYju3fvpk2bNtVus6g8T03CodXItAAhRN0gSThcyvJhr3hu/EVkoCUAyzd51oVeiMtSdhpas+Oi6duLEnB0zs21PJ+dVuVD6HQ6GjZs6NYsiPPnzycuLg6j0Uh8fDxbtmwps/zmzZuJj4/HaDTSuHFjFi5c6FDmq6++olWrVnh5edGqVSu++eYbu+enTZuGoih2j8jIyBo9r4qyhSEe1hWkkyQcQog6QgIwV1K0Rf/x3I/7+vUs6waZPG22txBXkO1Gy42UTrl5NbK/KVOmMGnSJM6dO1cj+6uMZcuWMXbsWCZPnsyuXbvo0aMHAwcOJCkpyWn5xMREBg0aRI8ePdi1axfPPvssTzzxBF999ZWtTEJCAsOHD2fkyJH88ccfjBw5kjvuuIPffvvNbl+tW7cmOTnZ9vjrr79ceq6lsSXhcMvR3ccagHla4CmEqHtkCKILmRUF1EvuNnqghsE+tv/n5pswGuQtJ8TlYpWvDy+HBJGu1aKoao0FYG+88QaHDx8mOjqahg0bOmRBdGWK+rlz5/LQQw/x8MMPAzBv3jxWr17NggULmDlzpkP5hQsXEhsby7x58wBo2bIlO3bsYM6cOQwdOtS2j759+zJp0iQAJk2axObNm5k3bx6fffaZbV86nc5tvV7OeNoQRJ3WmoRDCCHcS74Nu5CiSAdjgyBv2/+T0nNoFuHvxtYIISoqTaNhRkgwWUW9Bs3y8wksJ3lGRd166601sp/Kys/P5/fff2fixIl22/v168evv/7qtE5CQgL9+vWz29a/f38WLVpEQUEBer2ehIQExo0b51DGGrRZ/f3330RHR+Pl5UWXLl14+eWXady4cfVPrJKsw+I9LQmHVpaGEULUERKAuZC5KABTPOwid6lLe7yOn8uWAEyIy4AKvBAaTLZGgaIvrXk1eENp6tSpZT7vqiyIqampFBYWEhERYbc9IiKClJQUp3VSUlKcljeZTKSmphIVFVVqmUv32aVLFz788EOaNWvG6dOnefHFF+nWrRt79+4lJCTE6bHz8vLIyyvudazp10X1sKF41qy8ZtWSiVOjkZukQgj3kE8fV5IeMKB43P3J8zlubokQoiJW+/qw3teHwkt6DI4a9Kzy9SmjVvm2bdtml3yjZACQl5fH559/Xq1jVIRSoidEVVWHbeWVL7m9vH0OHDiQoUOH0rZtW2688UZ++OEHAD744INSjztz5kwCAwNtj5iYqmWeLKm4VZ4VgLWPqWf7/+GzF93XECGEx5MIwZUUyYIIYNBZ3mbJ53Pd3BIhRHmsQw+VEsGRoqrMCAkirRq9Bl27diUtrTiDYmBgIEeOHLH9fP78eUaMGFHl/ZcnNDQUrVbr0Nt15swZhx4sq8jISKfldTqdreeqtDKl7RPA19eXtm3b8vfff5daZtKkSWRkZNgex48fL/P8Kqo4CYdnXZz8jXqseTh2Jcli30II95EhiC6kaixZED15CCKAUa8lO7+QM1kSgAlRlx3PMzI7JIRsjXLJOoYWqqKQrdHwQlgo83ycD5krT8keL2dD4Fw5LM5gMBAfH8/atWu57bbbbNvXrl3LLbfc4rRO165dWbFihd22NWvW0KlTJ/R6va3M2rVr7eaBrVmzhm7dupXalry8PPbv30+PHj1KLePl5YWXl1eFzq1KPGwIIoCPQceFPBP7k7Pc3RQhRCWcyczlTFbpiaDC/b0IL1p7tibq7kvO4IUV+8k1FTJxQAtahuqr1vBSuD0Amz9/PrNnzyY5OZnWrVszb968Mi9ImzdvZvz48ezdu5fo6Gj++9//MmrUKLsy58+fZ/LkyXz99dekp6cTFxfH//73PwYNGuTq07EnQxAB8PPScu4ipF5wXF9ICFF3vLrjHzb7eZf6fKGisN7HyGE1j6tc1IayhgLWhPHjxzNy5Eg6depE165deeedd0hKSrJdRyZNmsTJkyf58MMPARg1ahRvvvkm48eP55FHHiEhIYFFixbZZTd88sknuf7663n11Ve55ZZbWL58OevWrePnn3+2lZkwYQI333wzsbGxnDlzhhdffJHMzEzuu+8+l56vM9ZXuNDjEtFDkI+eC3kmjqRecHdThBCV8MlvSby+vvQRA0/2acq4vs1qrO74ZX9wIMVyo2ZfciYtQ6t247E0bg3ArOuxzJ8/n+7du/P2228zcOBA9u3bR2xsrEN563osjzzyCB9//DG//PILY8aMISwszJYOOD8/n759+xIeHs6XX35JgwYNOH78OP7+tZ/8QZUADIAgHwNJ53JIvVAzKayFEK6RcNBAQWBrDP77UJ2MndYqWnrH9OaqIFeFX643fPhw0tLSmDFjBsnJybRp04aVK1fSsGFDAJKTk+3WBIuLi2PlypWMGzeOt956i+joaN544w3bNQegW7duLF26lClTpvDcc8/RpEkTli1bRpcuXWxlTpw4wYgRI0hNTSUsLIxrr72WrVu32o7rDp6WhAMgKtDI8fQcTsmQeCEuK3d3iaVvqwhyCwoZtjABgC9HdcWot4w2C/cvfbRAVeoeT88GQKNAcxckkHNrAOaK9VgWL17MuXPn+PXXX23DQ9x1gVNtWRA9W0SAEcgg/aL0gAlRV53OzCXtQj5Kzm3U89vLRVTbPFawfI75ar2Y0ubRah1n3759tvlSqqpy4MABLlyw9EakpqZWa98VNWbMGMaMGeP0uSVLljhs69mzJzt37ixzn8OGDWPYsGGlPr906dJKtdGVrMPiPS/8goYhvmw7mk6a3BAU4rISHmAkPMBIdr7Jtq1VdAA+FVhftrJ1zWYzF/MsCaMW3B1Pt6tCazwLrdsCMFetx/Ldd9/RtWtX/v3vf7N8+XLCwsK46667eOaZZ9BqtU7367JUv9YAzBOvcpewrgWWlWsqp6QQwl3mb/oHgCAzTD2byn8jQu2eV4HnTiURsudb6N2yysfp06ePXc/L4MGDAcvQw/KyEYqaYXuFPbAHzDrPI8/kecMvhRClO3k+m21H0m3/t+p+VWhpVarFbQGYq9ZjOXLkCBs2bODuu+9m5cqV/P333/z73//GZDLx/PPPO93vzJkzmT59es2c2KVs2cI87yJ3qUahvgDkFBSWU1II4S6r9yQD0CfkHP0vZDPVbCan6DNMi4be4R0Z0PdJ8I+s8jESExNrpK2iemxZED0wAAsuWgusoFACMCGExYVcEz1e3Yi5xEeiQavBz+iaUMntSThqej0Ws9lMeHg477zzDlqtlvj4eE6dOsXs2bNLDcAmTZrE+PHjbT9nZmbWyHorKs573DxNs3DL2FmTWZXFL4Wog85dyCcl0zIK4BG/nzmWp7MFXwA+el+m9JoD3tWbhFzR4eCuWohZlOR5AViIn2WuR2HJb1pCCI/1y+FUW/CltYYZisLgdlEuO6bbAjBXrccSFRWFXq+3G27YsmVLUlJSyM/Px2AwOOzXVal+Va01Db1naxFVPHnxZEYuMUHVW8xVCFGz3v7pMABajULTjF/50Ns+E+KkLpMIqWbwJeoO67B4T+wDCvWzfAeQ+EsIz3PpjZdtiefo0TQMrUbh1yOW9Sl9vbTsnT6gVtritq6IS9djudTatWtLXTvFutbKpUqux9K9e3cOHz6M2Vx8aTl06BBRUVFOgy9XkiyIFvV8DLYg9GCy3NkWoq757g/L8MP4CAUl5xxbfOwDsD6xfdzRLOEixTcFPS8Eu3StH5PMAxPCY6zak8yNczfbfr7//e1c9+oGVu1JZu/JDAAiS1lHzBXcGiGMHz+e9957j8WLF7N//37GjRvnsB7Lvffeays/atQojh07xvjx49m/fz+LFy9m0aJFTJgwwVZm9OjRpKWl8eSTT3Lo0CF++OEHXn75Zf7973/X+vkhWRBt9FrLa3HojKy9IkRdcjozl+QMS0ruSaEJXFAUdhjtRwQcOHeAfWn72Je2j7PZZ93RTOECqgcOQbw03fRZyYQoxGWnZC9WRYYTr9qTzOiPd3I60/5vPjkjl1Ef72THMUvyjabhfjXb2DK4dQ6YK9ZjiYmJYc2aNYwbN4527dpRv359nnzySZ555plaPz9VkSGIVr5eWvKzzfwjAZgQdcrr6w4BoNcqXH1+Neu8jRSWmGt736rixYJHtx/NmKudp3AXlwfrb9cT1wELuGRC/ZmsXKLqlb7wuBDC0ZnMXM5klX7zItzfy66nuabqgiWQmvrdXtvP97+/nahAI1NvbsWANs7naxWaVZ7/bk+FbjcNaV+/AqVqhtuTcLhiPZauXbuydevWmmhetSgaSwCmAhSaQOv2l9ttQv28SM8u4FhadvmFhRC15sc9lnm1nWLqoZz+m00hgQA0CmjEq9e/6lA+zDusVtsnXMfzwi/QaDQoWM79bBlfBIUQzn3yWxKvr/+71Oef7NOUcX2b1Xhday9Wyc8tay+Wn5cOg85xYF9BoblCyyC9cEtrBrkw6UZJnhsR1AJrDxgAplzQ1l7XZl1TP8ibv89cICUz191NEUIUOXQ6i/TsAgAmXnWMgpRCNhXN/7qz+Z20CmnlkuOePn2aCRMmsH79es6cOePQE5Oenu6S4woLT+4BA9BoFArNKmkX893dFCEuO3d3iaVvqwhyCwoZtjABgC9HdcWot3znvXSYb03VLTSrTF+xr8ybRhfyTFCNeyoB3vqqV64CCcBcSL003XpBDnh5bgDWJMyPTQfPknYhjz1Fkx1LKq/rWQhRs6zDD731Wtqf/JStRi8ytVoUFP7V7F8uO+79999PUlISzz33HFFRUbL4ci2zrQPmgUk4AHQSgAlR5eGA4QFGwgOMZOcX9yq1ig7Ax1B+SFHVutsSz9nmKpeld/Mwhzafycxl48Hy5y6H+zue65h1Y7hYcJEnOj5BU++m5e6jMiQAcyXNJS+vybN7floVpaLPKTAz+P9+dlqmrK5nIUTNs16UejQNhaTtrAu0LBERGxCLQee6rLE///wzW7Zs4eqrr3b6vKwDVks8swMMvVZDnslMugRgwoNVZzhgbTuTVbHv0Ld2qM8tV9vP4yo0q1z36gZSMnKdfuQpQGSgkc5xwXbb8035/HLyF8yY+fXUrzRtIgHYZcM6BNEyB8yzP+g7xgbZ/v/+/Z14YMkOoOLd1kKImrVmbwrZ+YUATGx3EfORHDb4WP5OB8UNcumxY2JiPHb4W11Q3APmmb8Dg04DeXA+p8DdTRHCbaozlLC2/HniPB9tPcbpCvR+gfNeLK1GYerNrRj9sSV/hM7/T7wivyMvZQiFWe0AmHpzK7SaSxboMJsZ+eNI2yiBB1o/gDm3ZkcMSADmQsql64CZPHuyb8OQ4sWXL+QV2v5f0W5rIUTNeqPozme4vxeNDy7iTy8DZ3WWv8V7W91bVtVqmzdvHhMnTuTtt9+mUaNGLj2WcFT8NcMzAzDrF8zMovmP1c3MJoS7VOe9W52hhFD6osY1WffOd7babhSWpbReLKsBbaJYcE9Hnvs+gZzwr0GTizHqa3x1LZl+UxeHDIqv73ydfef2ARDrH4ufwY/M3JodmSHffF3p0nkNBTnua0cdoNFo8DFoyc4vZOexc+5ujhAe7UKuib2nLBeTO6+JgZ2bWV+UfCPSJxI/g2vnqw4fPpzs7GyaNGmCj48Per395OejR4+69PieztN7wLyLArCsPMsXz8tpKJYQl3LXe7cq6eDLqhtg1DO4XRRt6gfatp9Iz7YFX6G+BvJMZtvf7KWsn2cle7FK6t86kuXJG9hyMhdFAY02n66dNzGgza0OZVccWWH7//Ru08s8n6qSAKwWWIYgenYPGECon4GkcznsT85yd1OE8Givrz+ECmgUGNNei/nX86wNtVw0+8T2cfnx582b5/JjiArw0GGgvl5FPWBFQxAvh6FYQjjjjvduaengUzJyGf3xThbc07HUIKy0upm5BXy6LclpnSAfPTue62ur//x3eziTWTytJ7KCgd/qo6v5+dRmW9+IGTPrk9az6ugqBjQaYCuXfCGZszmW+dFzes6hU2SnMvdbVRKA1RYPH4II0CjUl6RzOSSdk7XAhHCnr3aeBKB9TD2MCa+xx2DgeFEv1CPtHnH58e+7774yn5ckHK5lS0Pv1la4T2ywD3+eyOBU0byS6g7FguoPY6yJYZDSBvctEuyuujXx3q2MvPxCJn/7l9PPDuu2Kd/+RavoAIfeqEKzWmrdS/egVRRbZlyNRmHsjcU9eAPaRNGxkS+9PrkNjeEc9zZ9mqe6DbIdy2w2czr7tMOe03PTmZYwDWyrAFooKEz/dTqxfrEEGS1zoN/Y9QYAeo2evrF9y3g1qkcCsFqgggRgQOuoQH46lCqpf4Vwoz9PnOdc0d/g+L7N4OuV/OhnmaMZ5RtFiHdIrbSjsLCQb7/9lv3796MoCq1atWLIkCFotdryK4sa4amJULo1CeX7P5M5n11z16LqDgWriaFk0gb3LRLsrro1oaLzsXLzTbSdtoYCc9mfG6kXCrh+1qYqtMRyzI8fvpauTUq/DmXmp6M1ngHgm6Q3eSi+v+26dcMXN5CWm1bhI6qoXCi4wPAfhjs8d3XY1Wg0jgs71xQJwFzIbm0bD8+CCNClcTALNv9Dnskz154RojaVdlf1+eV7AMswrB71zlGYc45VodEA3BR3U6207fDhwwwaNIiTJ0/SvHlzVFXl0KFDxMTE8MMPPxAWFlYr7fBUStH3J9VD+8D6tQ7n2W/ArMKB5ExaRAVUe5/VHQpWE0PJpA3uWSTYnXWrqzJzuRb/erTc4KsmlJVyXlVV5u6cbfs525TNi1tf5LXer/HLyV8qFXyVRUHhyY5P1si+SiMBWG2ROWB0axLq7iYI4THKu6vaKMQXfprNTqMXZ4qyHz7U9qFaadsTTzxBkyZN2Lp1K8HBlqxVaWlp3HPPPTzxxBN89tlntdIOT+epPWChfkb0WoWCQpU1+07bArDqZHWr7lCwmhhKJm1wzyLB7qxrVZX3bkXmcvVrFUGvOZs4kZ5DZWKvxfdd45CRcFviOR78YHu5dZ2lkrdafXQ1W05utv1sVgtZl7SOLp90Idtkmd5Sz6seq29fbSujqioTf57IlpNbKFQdMypqFA096/fklR6v2LYZdAZ0GteGSBKA1QJVUaQHDMvaK9ZMiEII13J2V/WW9lEs/yMZBZg3vD28v4ZVvkWLLxel2q0Nmzdvtgu+AEJCQnjllVfo3r17rbTBk3l6GnqAMH8jp87nsP2oJStvdbK6CeFOVXnvFppVpq/YV+pcLgWYvmIfuQWFJJ2zz+Id6K0nM6egzEWNezZ3DAB7Ng8jKtBY6QWRrdJy0piRMAMFxaH33hp8Adxy1S34GHzsnn+83SS2pewg23QR+889BW+tD/9uN9GhjqtJAOZCCpemoZceMIDoet4cPnPB3c0Q4orn7K7qhoOWzE4dYuvR1HyU/NwM1oTXB+C2prfVWtu8vLzIynLMhnrhwgUMBkOttcNTFaeh91zNIvw4dT6Hw2cuVCurmxDuVNp7Nzkjl1Ef78THoEWvdZzHZCo0c7GMm+Fq0T6e+uIPwJKJcPotrWkU4sup8zmVWtTYqrILIm9K2sS0hGkUmAtQVZUcUw4m1TENvZWPzoe3+rzlNGvhyt0XSDt6M94NSo6uUEk9OoSVuy/Q3HX5Npyq1OyyWbNmkZNTHAn/9NNP5OUVBxZZWVmMGTOm5lp3JZEeMABaR1d/rL0Qomqyci0Xr+cGt4LNs/nN28h5rRYFhXta3lNr7Rg8eDCPPvoov/32G6qqoqoqW7duZdSoUQwZMqTW2uHpPHUOGGC7y34mM7fMngCw9AQUVnD8VcmhYBWtV1P1pQ3Vr3+51C2rF8sqO7+QjJwCh0dZwZf9MSz/jugcy5D29WnXoJ5tUeOwenkYo75G0V7AGPU14UEF5d6sqEzdF397kbTcNDLzM8kqyCoz+AJLL1g9r3pOn7u7Syzf3DeGjqHFIyw0aLgmrCff3j+Gu7vEVuj1qEmKWolB4FqtluTkZMLDwwEICAhg9+7dNG7cGIDTp08THR1NYeHlPcQsMzOTwMBAMjIyCAioesAwYfMEVh9dTd+L2cxt/yR0f6IGW3l5+mbnScZ9vtv2874Z/V2WLlUIAVm5BbSdtsb2c4S/gd8m94WX6/NsgBcr/H1pWq8pX9/yda216fz589x3332sWLHCtgizyWRiyJAhLFmyBEVRauQz+EpTU9emu99pwZ9eeiZk5nBfYBvLxphroffEGmpp3ffP2Qv0+d/m8gsWiQ3y5raO9RnXt3mpZaxDwU5nFt+YrswwxurWlzZUv35dr2symXns4985dT6Hi3kmktJznO3OTr9WEUQG2s+rSsnIZc0+x3Ttzup2jA3i0evj7DICqqrKmHX/YctJy7paGjT0ju3NvN7zyt1nyboAvjo/6vtH25U7lH4IgBtibiDcO5yfT/3MyQsnnd44UlDoE9uH13q/VuaxT2SdYODXAwHw1/uz4rYVFc78W1Ofv1aV6gErGat56gTeKpEkHAAMbBPh7iYI4TFW7Unmxrn2XzJzCsxs2rSOCwUXWevrDcCIFiNqtV316tVj+fLlHDx4kC+//JIvvviCgwcP8s033xAYGFirbfFE5qIvPYF52XBko+WxeSac3ufehtWiJmF+6CqYYAMgKT2H19cf5nia83UsrUPBLv0CDcXDGFftSS5z/9WtL22ofv3Loe5r6/9m/YEz7E/JqlDwBXBTuyhm3NLG7rHgnnj8vU2UPg9Uxd/HxIJ74hnVq4lDOvayFjUuT8m6ABdNFziUfsjuAaDT6Hit12tM7jqZedcvwlvny6WzWC0UfHS+jGrzdLnHDjYWzy+b1GVSrS274ox0PbiQpii+VQEKy+469RRGg44Ao47MoqFQlc00JYSomNLmBmTlmvh97eekBfmQq9Gg1+gZ2nSoW9rYtGlTmjZt6pZje7JU70AozCbVpx54RUHaYcsTybsgopVb21abXh3alnd/OsKB0+XPS9YolrT1976/jTb17e9+q6rKuv1nyhzGOHbZbm7885T98jQ1VF/aUP36l0vdTUXzeAO9dUQGGDlYgfeus6yCWo3ClMEteOYLy9/+pfOxTFntAIUpN7Vw+t0sLSeNqb9OdXqsSVsm8eORH/HSOk+dn1eYx08nfnL6nFbR0jKkJXqNZVSERtEwosUIW/BX0/O4+sT2qXhhF5AArLbIHDDA8qUwt6B4iKpkmhKi5pWd4UplqHYLU/wtGQ+vjbrWpYtNWo0fP54XXngBX19fxo8fX2bZadOmubw9nixfb4TCbI61GwrXvQAvRUJBDpw76u6m1aqh8THc2qEB1726odzMbG2iA1i7/wyJqRdJTL1Y6WPlFpj5/s+UKre1uvWlDdWvX5fqTr6pFUM7Vuy9W1pWweHxzQn0CuC57xPICf8aNLkYo77GV9eS6Td1cfqdTFVVXtj6gl3WwUuZzCY2HN9Q2VO0ifSJLHUY4d1dYrmx5Rjm/JHEztRfAEtHR3xYD8bfOMal66W5QqUDsPfeew8/P8uF22QysWTJEkJDLes7OctqJaw9YBKASaYpIWrHtsRzJGc4X8yys3IAs1cafxijQYWxHcfWSpt27dpFQUGB7f/C/czWPIh6HyjIYVXqbl5Z1otJXSbRv1H/Su9v1dFVvPLbK26rX5V9XJqZTQG0pWRm69wohGPvJpCRXeCwj5yCQtuoDnDWm2ARYNThXbS4bk3WlzZUv/7lVLdZpD/DOtZHU8H3blkjjPq3jmR58ga2nMy1zOXS5tO18yYGtLnVrpzJbGLlkZX8c/4f1ietL3V/VkFeQQ7raJnMJtLz0kutU1i0ptfh9MNcFXSVw/PWzL4vBU6xzePy1fsyu/d0QrzLHrp+NvssZ3POkmsqvi4eOHcAo87SOxjmHUaYT1i551WTKhWAxcbG8u6779p+joyM5KOPPnIoIyzsupo9PACr6JoTfVtFynBEIarpTJbz4AtgmPYnvim6iearqU+z4Ga10qaNGzc6/b8zmZmZrm6OR7MukaKqKpw/DjojaRoNM7IPkqWoTP/leTphJMQQCD4hUC+m9J2dPw7ZaaTlZzDjl6lkmbJrt34N7GNAAxOf3OTF/23Zw97IL1E1+fhEfUkbjTePX9eGbg1M4GdgzbieTusn/JPGU+9+T5CSRaE2h1NRxfuIztGhLfQmXfXnfyMH07WJ45yT6taXNlS//uVYFyr43i3N+eOsPvKD87lcO99mQONBtr+bCZsmsP54+YEXqkITQ3u+vfMjx6dUlds+G8k/+X+C4uTbYFFdZ8GXtb1kpxFiyqV+gYkUnZZJze4iJP04pB8v8+/8x93v8v2eDwBoWbTtleXFc58Ht7mPe7s9W/751aBKBWBHjx51UTOubDIHrOw78lC85sS2xHOlfrgLISrG2Zh/AB9y6a/dyhB/y6iF66MG1GazbB588EFef/11/P397bZfvHiR//znP8ybN8+lx58/fz6zZ88mOTmZ1q1bM2/ePHr06FFq+c2bNzN+/Hj27t1LdHQ0//3vfxk1apRdma+++ornnnuOf/75hyZNmvDSSy9x2232a6tV9riups+7AG/Go5ryeCE8lGzMgEJ2wUVeXPkgr51JBZ0XPP678y8254/b1/fxBqUW69dgG7qa8vg8PJT9Gm8KFQWNJo+IwP+j2/pU2Fx2GzoHX2Sj8SkMFDAuPJSUS/bRMnohr51JJQ89uuDegOP1rbr1pQ3Vr3851i3tvato8ggK/D9abDxH2k8GMu/9GtXffnSRkpVM4cdDmR4RgqJRUC/pMFBUlem73+Cq1VPR3vMVqn8UW05tAUCPhsj8PE7qdZb+8xL1/M2FvJe4xtK2En8vSsYJ3ktcy5CoUC44OWZZda3niikPb8CW6uPo5OIypf2dnz/OyPXzuLeMZHjq6XnQamT5N3tqkOsH/nswrWLpLjYDmB2HLXiSsu7IV6WcEKJ0neOCifTXUzLD1SDtb/zuq+GcVotq8uX5no+4pX0ffPCB3ZqSVjk5OXz44YcuPfayZcsYO3YskydPZteuXfTo0YOBAweSlJTktHxiYiKDBg2iR48e7Nq1i2effZYnnniCr776ylYmISGB4cOHM3LkSP744w9GjhzJHXfcwW+//Vbl47qSdXSG1pQLpjxW+/qw3teHwqLthYrCOl8fVvn6gCkPstOc7yg7zb3160gbtDnn8KKgzH14UYA255xL6ksbql//cqxb2nvXrCj85OtDz4YN6NUgnCEbRnHL8lvsHkM2jOK26DAuaDV2gRCAqihc0Gq4LTrMVje/aBTXiu6zWHkymVfPptkFX9Z6z6WlE1qQ6/zvJTuN0IJcnk875/SY5dXFVE428dL+TrPTUMrJRK4UlvM54wKVCsB+++03fvzxR7ttH374IXFxcYSHh/Poo4/aLczs6axZYAoUxeOHIJZ2R76q5YQQpdNqFPr6HcEyuLc4CPuXdjNfFyXfiFHC8DP41mq7MjMzycjIQFVVsrKyyMzMtD3S09NZuXKlbZ1JV5k7dy4PPfQQDz/8MC1btmTevHnExMSwYMECp+UXLlxIbGws8+bNo2XLljz88MM8+OCDzJkzx1Zm3rx59O3bl0mTJtGiRQsmTZpEnz597HryKnvc2qCqqmXoYUgwSollZRRVZUZIEGnlJGhxd31pg7ShpupfSXVdoXlQc+r7WJYS6n8xmz4Xs9EWHVerqtx4MZsBF50n57hUdepeSSq1EPPAgQPp1asXzzzzDAB//fUXHTt25P7776dly5bMnj2bxx577LLPYFVTi63N/G0mnx74lGtyclkc2gOGLa7BVl5eCs1qhbL1/PzMDTIHTIhqMpvNtJ+2gqx8HTpNISazloZKCh/7Ps1NDaJRFYVXO01iUOu7arVdGo2m1BTWYOmZmT59Ov/5z39cshBzfn4+Pj4+fPHFF3bDA5988kl2797N5s2OC/Nef/31dOjQgddff9227ZtvvuGOO+4gOzsbvV5PbGws48aNY9y4cbYyr732GvPmzePYsWNVOi5AXl6e3U3NzMxMYmJiqv263PD5DZzNOYuigg6VAnC4mw2AqqIHfM3mUlcLytZo3Fpf2iBtqKn6V1pdrarSOzvHMgy3xNpZKirjwkPZ5ONt6zmrWN3iVqRpNNzcIJosjUKA2cx3J5IJMZsvKVtyvzVTt3w1WbfI4HlkNru9Rq9LlZoDtnv3bl544QXbz0uXLqVLly62xBwxMTFMnTr1sg/AaopRa+nNKVCU8rtOr3AlM005+3MoL1uPEKJi5q79G//888RqzmGs/xGY6nHfxRQ+9/VHVRR8zWZSM47Wers2btyIqqrccMMNfPXVVwQHF6dHNhgMNGzYkOjoaJcl4UhNTaWwsJCICPsF4SMiIkhJcZ4iOiUlxWl5k8lEamoqUVFRpZax7rMqxwWYOXMm06dPr/D5VVSjgEaczTmLqkBBaV84ABSFAuC81nnWu3K5u760QdpQU/Uvw7rWYYyH9XquKrCfBqMAz6WeY1uDaC5ocJiP5Ws2MyXVOuzReQATYjbzfNo5XgkOYlLauRIBVOn1qlu3fC6oa6r56TGVCsDS09PtLiCbN29mwIDiSdzXXHMNx48fr7nWXea8dNYhiEB+5dcNudIMaBPFgns6MvW7vQ4rvr91VwdJQS9EDTCZzHy/ZRsbvJ7CqBTAWVjl68PMiCByFMuQw1lnUulxYl6tTzru2dOSRS4xMZGYmJhaWX/MGWeLoZbXM1eyfMntFdlnZY87adIkuzXTrD1g1fVev/fYkbKD/NN/8sHWV9jmbcTspB0aVaVzTi4Pxd2CyT/S4XltZjKLj37ntvrSBmlDTdW/Eutae7Gu6j0dguPsnzyXSMiaZ3k+7RxPh4faPWWdjxViNkO/l+3rnkuENcXZAgeUNnSwZL0arluqmq5r1eAaKBkjVlOlArCIiAjbhTM/P5+dO3fa3Z3LyspCr9fXbAsvYwatASi6wygBGGAJwrpfFUrbaWvstp/P8ewskULUlOnf78O3MAOjznLH0zpHIEujgKIQVVBA95xcS79HdlqtBmBWDRs2BCA7O5ukpCTy8+3nyDZq1Mglxw0NDUWr1Tr0Op05c8ahd8oqMjLSaXmdTkdISEiZZaz7rMpxAby8vPDyqvnFRTUaDZ2jOwMGWp5N4+ZS7oL7mc28cjaNkFtGQvTVjjs6tZtm2993X31pg7ShpupfgXVtvViNujutC5b5WKsuZtuGIlqDNltgVLJuUb1ylXHMy6rupWp4ZEalbj8OGDCAiRMnsmXLFiZNmoSPj49dCt0///yTJk2a1GgDL2fWIYj5igIFnjW5sCyXDjNsEOQNwKKfj7irOUJcMbLzTXy2rTirngq8EBpMdlHwBeBnVqnGIKIacfbsWQYPHoy/vz+tW7emQ4cOdg9XMRgMxMfHs3btWrvta9eupVu3bk7rdO3a1aH8mjVr6NSpk+2GY2llrPusynFri3UoUGlZyRyHBtWt+tIGaUNN1fe0ulA8FNHHrILD0EPhSpUKwF588UW0Wi09e/bk3Xff5Z133sFgMNieX7x4Mf369avxRl6uvHWW4KJAUaDAMeWygLs7Wxbu/ufsRc5d8OxMkUJU16Sv/8JkVm3zsUumJwb428tgSa3tRmPHjiU9PZ2tW7fi7e3NqlWr+OCDD2jatCnfffedS489fvx43nvvPRYvXsz+/fsZN24cSUlJtnW9Jk2axL333msrP2rUKI4dO8b48ePZv38/ixcvZtGiRUyYMMFW5sknn2TNmjW8+uqrHDhwgFdffZV169YxduzYCh/XLXxCQOdVdlYynZelXF2sX4faYFIMZe7DpBjKbEO16ksbql//Mq1b5feuTwgmjWUQnDWICyk083xq8Xwsk0bnWPeSeqVxWq8G6qIrZzRAGeda5bouVKksiFYZGRn4+fmhLTEp8Ny5c/j7+1/2wxBrKgvi90e+Z9KWSYSbTKzPAMbvr7lGXsay8020en41AH8+35cOL66j0Kxyd5dYXrqtrZtbJ8Tl6Xh6Nte/uhEVeOSqDEad+nfR0BTHBS/9zGZWnEgm5OGNZQ+5cJGoqCiWL19O586dCQgIYMeOHTRr1ozvvvuOWbNmsXLlSpdkQbSaP38+s2bNIjk5mTZt2vDaa69x/fXXA3D//fdz9OhRNm3aZCu/efNmxo0bZ1uI+ZlnnnEInL788kumTJnCkSNHbAsx33777RU+bkXU1LXJzvnjkJ1GWn4GN//yX7JM2QTofPiu+yxCDIGWLyVlDVN1d/060obUE4c5n5bC+YJMHt8/jYvmHHy1PrzZYir19AHUC4kktMFVLqsvbah+/cuxbnXeu2kpf5CRXvroo8CgxoREtq+xetWtaz3XUpX1d1qdukVq+vO3UgHYgw8+WKFyixdf3unWa+pFXndsHeM2jSO4sJDNqbnwXxlmB/YB2L4Z/XlwyXa2HjmHv5eOv6b3d3PrhLg83fTGFvaeykSvVfjj0XCe/WF4+SmGb/3KLQFYQEAAf/75J40aNaJRo0Z88skndO/encTERFq3bk1KSopLA7DLlUsCsEusOrqKV357hUldJtG/UeU/i91dX9ogbaip+p5WV5TPrQGYRqOhYcOGdOjQgbKqffPNN9VumDvV1Iv8y8lfGLVuFP6FZn49nQmTJEMkOAZgf5/O4pa3fgVgzrB2DOtU+0kBhLicrd9/moc+2AHAhH7N6N/gALcnlJ/16ZuuM7mq2WBXN8/BNddcw4svvkj//v259dZbCQgIYObMmbzxxht8+eWX7Nq1SwIwJ1wdgAkhhHCupj9/K5UFcdSoUSxdupQjR47w4IMPcs8999it4yLs+egs8yzyFaBQ5jeVpn1MELFB3iSl5zB37SEJwISoBLPZzFNf/AFAqJ+Bx29oinryAn0uZrPRx7vs9MR+DWq7uYBlDlhycjIAU6dOpX///nzyyScYDAaWLFniljYJIYQQtaVSSTjmz59PcnIyzzzzDCtWrCAmJoY77riD1atXl9kj5qnsknCYC93cmrpt4sCWAJzKyOW3I2WM0xVC2Pnf2r85n21JOT9v+NUAKL6hPJd2Hq2Tj2VbeuLzF2p90rHV3Xffzf333w9Ahw4dOHr0KNu3b+f48eMMHz7cLW0SQgghakulV8H08vJixIgRrF27ln379tG6dWvGjBlDw4YNuXDhgivaeNmyBmBmRcGkSgBWlkHtoqjnY0neMvHrP93cGiEuD6czc1m46TAAV8fU47qmYZYnjIFoVBUFxwhMVRSeu/pJQsbscMsaYM74+PjQsWNHQkNDyy8shBBCXOYqHYBdSlEUFEVBVVXMFVgfwtN4671t/y9w8kVI2HtmQAsAElOzWbcvpZzSQoh7F22jULWsrffeffHFT6yayLIAX/I1Gi4dgKhFw43hnRgQ2QW07stWO2zYMF555RWH7bNnz+Zf//qXG1okhBBC1J5KB2B5eXl89tln9O3bl+bNm/PXX3/x5ptvkpSUhJ+fnyvaeNmyzgGDonlgEqSWaUTnWML9LWs1PPvNHqdlzmTmsudkRqmPM5m5tdlkIVyqrPf7rB/3c/B0FgBP92tOqJ9l4XcKTaTv+YIPiyYJ35RZPDLBp9DElO3fwjs9Ycf7tX06Nps3b+amm25y2D5gwAB++uknN7RICCGEqD2VSsIxZswYli5dSmxsLA888ABLly4lJMQ9cwguB5cGYCYUyDkHvjLEpiyvDG3Hg0u2cyYrj/mbDjOml/3aF5/8lsTr6/8utf6TfZoyrm8zVzdTiFpR3vsdoFGID6N6NSnesGYyH/t5k6XV4KVomTjkY77fVLTIcPvRhAy4zlLOP9JVzS7XhQsXMBgMDtv1ej2ZmZluaJEQQghReyoVgC1cuJDY2Fji4uLYvHkzmzdvdlru66+/rpHGXe40muIOxgJFgQupEoCV44YW4bSK8mdfchZz1xzi7s6xBPoUf1G7u0ssfVtFkFtQyLCFCQB8OaorRr1lUXBrD5oQV4LS3u+Tv93DwZQsFAU+erBLcQVTPmd3vs/H0Za5YP3iBvGPsXgodP0G3dins/w9hWm1hNXeqdhp06YNy5Yt4/nnn7fbvnTpUlq1auWmVgkhhBC1o1IB2L333oviJKWxKF+BAmSfBVq4uylucyYzlzNZeeQWFCck2Xcq0y54Cg8wsui+a+j+6gZMZpWHPtjBl6O72cqHBxgJDzCSnW+ybWsVHYCPoVJvZSEuC9b3e1ZugW3bV7+f4GCKZejhoz0aExNS3NPOj//lA39vsjUaDGhYcWQFK46ssD1936r7bP8f3X40Y64e4/qTcOK5555j6NCh/PPPP9xwww0ArF+/ns8++4wvvvjCLW0SQgghakulvrXK+ixVV6AokJXs7ma4lbPhVNa7+lA8fDCqnjcPXRfHu1sS2XEsnU9/S+KuLrG13Vwh6oRVe5KZ+t1e28+fbbcs6B7ub2DSoJbFBfOzSf7zE5ZGhwNwV6t7Gdh4YKn7DfN2V/8XDBkyhG+//ZaXX36ZL7/8Em9vb9q1a8e6devo2bOnDEMUQghxRZNuAxdTUFBRLQHY+ePubo5bWYdTlebS4YOTb2rFj3tSOJGew/PL99C7eRhR9bxLrSvElWjVnmRGf7zTaQ7VM1n5rNqTzIA2UZYN34/j3QBf8jQa/LXejIsfZzcMuq656aabnCbiEEIIIa50dffqfIXQKJaXuAAFMk+6uTXuFR5gpE39wFIf4QFGu/LLHrsWraJgMqsMeesXTCbJIik8R6FZZfqKfaUuYKEA01fso9CsQm4m+w5+w1f+lky0/+74ZJ0OvoQQQghPJldoF1OKVuEpUIAsWduqMurX82Hm0LYAnM3KY8R7W93cIiFqz7bEcyRnlL6sggokZ+SyLfEc5i/uZ25QIGZFIdIYyt2t7q69hlZQcHAwqampAAQFBREcHFzqQwghhLiSyRBEF1MUBdSiOWDZqe5uzmXnjk4x7E5K59Ntx9l+NJ1p3+1l2pDW7m6WEC53Jqtia9plnNjPhuRf+S0iDI2qMrPnbBe3rGpee+01/P39AZg3b557GyOEEEK4kQRgLmYdgpivKHDhDJzaXXph/0i3rs1TV718ezv+OpnBXyczWfLrUUL9DNzXrZHt+W2J5+jRNAytRjJ0iitHuL+x/EJAq9//y+jgegC0DWtPp8hOLmxV1f3xxx8MGzYMLy8v4uLi6NatGzqd80uQJOEQQghxJVNUVS1tioHHyszMJDAwkIyMDAICAqq1ry6fdCHblM0bp8/Su0ADBRdLL9xzIvSeVK3jXanyTWaun7WBlMw8APy9dGTlFaeijwo0MvXmVsUJCYS4zBWYCmn5/GpMZucf0QrQ3fcEfXxfYG5IEHoVVt2xnnCf8NptaAXp9XpOnDhBREQEWq2W5ORkwsOdt7UmP4OvJPK6CCGEe9T056/0gLlYcRIOLN+YHt0MphxYPMBS4MFVoCvK7ie9X6Uy6DSsG9+LrjPXk5Vnsgu+AFIychn98U4W3NNRgjBxRXjs49/LDL5A5Sn9XB4LCgTgpqturbPBF0CjRo1444036NevH6qqkpCQQFBQkNOyV199de02TgghhKhFEoC5mC0AUxQw5UP01ZB/SS9YZDsw+LqncZcZb4MWHy+tQ/AFloQE1qxwfVtFljoc0boYdGmsi0EL4U4zVuxlw4GzALSrH8DprDxOZxa/byMDjbxVfy3vXCwkR2MgSGNkerfp7mpuhcyePZtRo0Yxc+ZMFEXhtttuc1pOURTS09NruXVCCCFE7ZEAzMW0ihYomgNmLnBzay5v2xLP2X0JLenSrHBdm4Q4LeNsMehLWReDFsJdFv+cyOJfjgLQKtzMyzemkmsy8+TniRSi59+9Yunof4rDmz/k5/AQNKrKi73/V+fTzt96663ceuutXLhwgYCAAA4ePFjqEEQhhBDiSiYBmItpNZYArEAp6pE5fxx8JM1yVVQ0K1xZ5ayLQecWFDJsYQIAX47qilFv+T1duhi0ELXtk9+OMeP7fYDlvTiw0WJGbPvL8uRVln9mnYAJaem8X5R4o1N4R65vcL0bWls1fn5+bNy4kbi4OEnCIYQQwiO5/Zbp/PnziYuLw2g0Eh8fz5YtW8osv3nzZuLj4zEajTRu3JiFCxeWWnbp0qUoisKtt95aw62uOJ1i+YJx0RqAnfzdbW253FU0K1xZ5ayLQbeKLp5A2So6oNTFoIWorjOZuew5mVHq40ym5YbBlzuOM/mbPQDU89Gzdtz1DGp6L29H3s3rQbdSv8CEVlV5ojCCvw160nRaAszw+o3z3Xl6FXZpUNWhQweys7PJzMx0+hBCCCGuZG7tAVu2bBljx45l/vz5dO/enbfffpuBAweyb98+YmNjHconJiYyaNAgHnnkET7++GN++eUXxowZQ1hYGEOHDrUre+zYMSZMmECPHj1q63Sc8tH7AJCm01s2nN4DTfu6sUWXr85xwUQFGknJyKWs1J3/t/4QHWKuwWiQDl7hfhUZ9tooxJcJX/4JgL9Rx4bxvQjMP43v1yNpouaTptGQ2SCaQuBdJZkcfz8UVeW102n4ZaeDwa+WzqbqgoKCbJkP69WrZ1kjsQRVVWUOmBBCiCueW7+hzp07l4ceeoiHH34YsCzOuXr1ahYsWMDMmTMdyi9cuJDY2FjbIp4tW7Zkx44dzJkzxy4AKyws5O6772b69Ols2bKF8+fP18bpOOWnt3wxStUXBWCppX8RE2XTahSm3tyK0R/vRIFSg7Bfj5yjwwvr+N+/2jOonWREFO5V3rDXdftPM+7z3QD4emlZP74nwX4GOJWGTs1HBV4IDSZbo4CikFO03wczMumcmw3ZaVAvpvZPrJI2bNhAcLBl+PXGjRvd3BohhBDCfdwWgOXn5/P7778zceJEu+39+vXj119/dVonISGBfv362W3r378/ixYtoqCgAH1RkDNjxgzCwsJ46KGHyh3SCJCXl0deXnFyh5ocAhPoZUkRnWrtATt/rMb27YkGtIliwT0dmfrdXruEHFGBRp67qSW//pPGx78lkVNQyJhPd9JsvR9v3xNPXFjd7yEQV6bwACPhAUaycouT8FzIM9EhNoi5aw/x1sbDAAQYdawee73DMNjVvj6s9/Up3lDUc9Q0//JK6tOzZ0+n/3dGhiEKIYS4krktAEtNTaWwsJCIiAi77REREaSkpDitk5KS4rS8yWQiNTWVqKgofvnlFxYtWsTu3bsr3JaZM2cyfbprUjjX86oHQLrWcrebrGSXHMeTDGgTRferQmk7bQ0ASx64hh5Nw9BqFAa1i+aOa2J47KPfSc7I5dDpC9zwv830bx3Bq0PbEehjAKDwkvWVtiWes9WvCEllLypr1Z5kpn631/bz/e9vx6jXkFtgBiDUz4v146+3vT+t0jQaZoQEo6gq6iVD9hRV5aWQIK7NycV5vs+6bdWqVfj5+XHdddcB8NZbb/Huu+/SqlUr3nrrLbTWz0shhBDiCuT2STIl5wFY5wBUprx1e1ZWFvfccw/vvvsuoaGhFW7DpEmTGD9+vO3nzMxMYmJqZkhPsLdlyE26tijfSfa5Gtmvp7s0WOocF2z3c7sG9UiY1IdFW44wa/VB8kxmVu09zZp9axncLprezcN5ZdV+W/n7399OVKCRqTe3qtAizpLKXlTGqj3JjP54p8OQ2eLgy8DP/+3lMGdRVVXb0EO15OeeopCt0fBiaDCvubLxLvL000/z6quvAvDXX38xfvx4nnrqKTZs2MD48eN5/fXX3dxCIYQQwnXcFoCFhoai1WodervOnDnj0MtlFRkZ6bS8TqcjJCSEvXv3cvToUW6++Wbb82az5UuOTqfj4MGDNGnSxGG/Xl5eeHm5Jv14uLdlnZss6/enwnzIz3bJsYS9h3o05u4usTzz9V+s+OMUZhW+++MU3/1xyqFsSkYuoz/eyYJ7OpYbhEkqe1FRhWaV6Sv2lZk0Rq/VoNc59vgcvnjCfuhhyX0rCut8fTh84QRXcXX1G1uLEhMTadWqFQBfffUVN998My+//DI7d+5k0KBBEoAJIYS4orktDb3BYCA+Pp61a9fabV+7di3dunVzWqdr164O5desWUOnTp3Q6/W0aNGCv/76i927d9seQ4YMoXfv3uzevbvGerUqI8wnDIA81Vy88dgvtd4OT2U06Hj9zg7seq4vA9s4D+yhOKHH9BX77IYnOiOp7EVFbUs8R3JG2evXWRcPL+kq3wbccDEbVOfvR62qcuPFbK7ya1Ajba1NBoOB7GzLjah169bZ5vYGBwe7dP5Xeno6I0eOJDAwkMDAQEaOHFlukiZVVZk2bRrR0dF4e3vTq1cv9u7da1cmLy+P//znP4SGhuLr68uQIUM4ceKEXZlGjRqhKIrdo+QcaCGEEJ7BrUMQx48fz8iRI+nUqRNdu3blnXfeISkpiVGjRgGWoYEnT57kww8/BGDUqFG8+eabjB8/nkceeYSEhAQWLVrEZ599BoDRaKRNmzZ2x6hXrx6Aw/baEu0bDYCKilnvg6YgG5IS3NIWTxboY+DernH8uOd0qWVUir8Md23iupk1MofMc1Rn8XDlYioNCwpA8bEEYSXmgPmazUxJvTyHNF933XWMHz+e7t27s23bNpYtWwbAoUOHaNDAdQHlXXfdxYkTJ1i1ahUAjz76KCNHjmTFihWl1pk1axZz585lyZIlNGvWjBdffJG+ffty8OBB/P39ARg7diwrVqxg6dKlhISE8NRTTzF48GB+//13u/lsM2bM4JFHHrH97OcnyYGEEMITuTUAGz58OGlpacyYMYPk5GTatGnDypUradiwIQDJyckkJSXZysfFxbFy5UrGjRvHW2+9RXR0NG+88YbDGmB1SZRf8XC2i/4R+J9LlMWY3aSiX4afWLqLB7o34v5ujfApYy2xqibykDlkniMn31Shcg6Lh5vNfP7jaN6vZ8miipM5YM+lpROi0YPP5ZeG480332TMmDF8+eWXLFiwgPr16wPw448/MmDAAJccc//+/axatYqtW7fSpUsXAN599126du3KwYMHad68uUMdVVWZN28ekydP5vbbbwfggw8+ICIigk8//ZTHHnuMjIwMFi1axEcffcSNN94IwMcff0xMTAzr1q2jf//+tv35+/sTGRnpkvMTQghx+VBUtZTxLR4sMzOTwMBAMjIyCAgIKL9CGcxmM+0/ag/At8ZWNNm/Coz1IPe8pcCzp8DgW70GexBr71FZ869K6z1K+CeNEe9urfCxFKBhiA+3XB3Nwz0a42/U256zZrUrmQq/Iok8qnMOJfdRGulFq1mVfb3NZjPTVuzjw4Syl51QgMhAIz8/c4Nd8L75ve48pT1PnkZDN8UPQ3ATNqXuBkVBi4be4R15rf2TluDrMlgDrLJq8jPYavHixYwfP95hyGG9evV47bXXeOCBBxzqHDlyhCZNmrBz5046dOhg237LLbdQr149PvjgAzZs2ECfPn04d+4cQUFBtjLt27fn1ltvtWXYbdSoEXl5eeTn5xMTE8O//vUvnn76aQwGg8NxS+OK10UIIUT5avrz1+1ZEK90Go0GBQUVlaPhLSwBmDX4EpXmrPfIGsRA2b1HneOCiQo0kpKR6zQpggL4G3WE+nmRmHoRFTials3r6w/z+vrDhPt70bVJCFeF+TF37SGHfVQ0kYd1XajsS3pHWkUHlNnbVpL0otWuyrzeW/4+yxOf7SI927JOl16rUFCoOiwebg23pt7cyi74OvD1A0xT0snTaGlSqPB//f6PMxeT2ZT2BwA+Oi+mtLzfUlir53K0c+dO9Ho9bdu2BWD58uW8//77tGrVimnTprnkmCkpKYSHhztsDw8PL3PpE8Dp8ifHjh2zlTEYDHbBl7XMpft98skn6dixI0FBQWzbto1JkyaRmJjIe++9V2qbXblGpRBCCPeRAKwWeGm9yC3MZb9/IH3c3ZjLnDUDYWnKykCo1ShMvbkVoz/eWeqX4VnD2jGgTRT5JjMfJRzli99PcOh0FmYVzmTlsXy3YwZFK7VoP9NX7KNvq8hyhyNWZy2y6mZilB60yrG+3hfzTAx/x9KLOu3mVnSIDUKrUQj39+LchXwe/WgHO46l2+pd2ziYxfdfw0+Hzjr0mEY66TE99tMrPJ2WQKpBT0ShmZnRgzj84QByFQWiLUPX7j1zitOfDuU0ENZ5DGE3zqidF6EGPfbYY0ycOJG2bdty5MgR7rzzTm677Ta++OILsrOzmTGj4uc0bdq0ctdx3L59O+C4jAmUv/SJs3oVqVOyzLhx42z/b9euHUFBQQwbNoxXX32VkBDnw0hduUalEEII95EArBb4G/zJzcnl5Lm/weAH+ReKn0z5E3Telv9focOJapK196iqBrSJYsE9Hcv9MmzQaXioR2Me6tEYk8nMt7tP8s2uk/yelG5bv8kZayKP346k0u2qsFLLOVuYtzJrkVW3F0160ConPMDIzqR0u9/ZtBX7iAo0MmlgCxb/nMi3u09ijamDfPS8MaIDPZqGQVYKA4JPc91d9Ri8cDcmtLx0S3Oui/FGqzltWaPCP5Izf3zKUwfe56iXgXqFhfxv0EdsPLaeBfXt3w9vBQfxVtH/R/v7MaaWXoOadOjQIa6++moAvvjiC66//no+/fRTfvnlF+68885KBWCPP/44d955Z5llGjVqxJ9//snp045JeM6ePVvm0idg6eWKiir+PVy6XEpkZCT5+fmkp6fb9YKdOXOm1Iy+ANdeey0Ahw8fLjUAc+UalUIIIdxHArBaEGIM4WzOWUIP/GgffAEsvmTCuc4LHv/deRCWlWJ5lMY/0vIQ5RrQJoruV4XSdtoaAJY8cE2ZvU86nYZhnWIY1imG5btP8uTS3eUe4673thHoraNRqC+dGgbTt2UEneOC0Gg0pS7MW5m1yKyq2otWE2uZeVIvWmm/s+SMXJ645P2gVRQevb4xT/dvhkZjWeXj2E9vcPGPdwCYbx02vhEOFv3Xt/2jBDXvxYStUzloNOJfaObl61+lfWQ80QGx9LpqcKntCvMuPcivy1RVta3RuG7dOgYPtpxjTEwMqampldpXaGgooaGh5Zbr2rUrGRkZbNu2jc6dOwPw22+/kZGRUWqgFBcXR2RkJGvXrrXNAcvPz2fz5s22haTj4+PR6/WsXbuWO+64A7AkkNqzZw+zZs0qtT27du0CsAvsSnLlGpVCCCHcRwKwWhDpG8mB9AOkaMtZds2UB9lpzgOwHe/D5ldKr9tzIvSeVL2GepBLg5TOccEVHvrnkK2uDBk5Jv44nsEfxzNY9HMiCuDrpSW3wOx0DlplhzBWpxfNE+ehVTVgrMhiygB9W4Yz51/tCfSxT6ow46LKtvql/z5uTEvm/Lp/s8vbiLfZzOT48fS4yrKYfJhPmG0twStJp06dePHFF7nxxhvZvHkzCxYsACwLNJfWG1VdLVu2ZMCAATzyyCO8/fbbgCUN/eDBg+0yILZo0YKZM2dy2223oSgKY8eO5eWXX6Zp06Y0bdqUl19+GR8fH+666y4AAgMDeeihh3jqqacICQkhODiYCRMm0LZtW1tWxISEBLZu3Urv3r0JDAxk+/btjBs3jiFDhhAbG+uS8xVCCFF3SQBWCxoGNoQTkKyrxsvd6QFoPhBMOcW9Zg+uKh6+KL1ftaIiiTwiAoxMHNicnw6l8ufJDE6m55BTUIgKXMgrLHP/1iGMCzcf5p4uDR2+zFvVVC+aO+ahuavnrKoBY0UWUwZ48LrGTn9fE3r9m8Npt5Gfn8G03ZbhZP9qdC8dwntiTv2LL/e9ym5vI0azmada3sdN7R+qxFldnubNm8fdd9/Nt99+y+TJk7nqqqsA+PLLL8sctlddn3zyCU888YRt4echQ4bw5ptv2pU5ePAgGRkZtp//+9//kpOTw5gxY0hPT6dLly6sWbPGtgYYwGuvvYZOp+OOO+4gJyeHPn36sGTJEtsaYF5eXixbtozp06eTl5dHw4YNeeSRR/jvf//rsnMVQghRd0kaeidqOtXk8sPLmfLLFMJMJjYcLz2JAwCPboboq0t/Pv8ivGxZ3FlS2Fdddr6JVs+vBmDfjP6V6vmxBj/gPJGHs+AnK7eA9fvPsGx7EglHKr54rlZR8PXSEubvRYMgHxqG+NA4zJf/W3+YtIv5TuuUltrc2XlUNZV+yXOr6HBOgNfWHqpyz1l1gjdr3bISaZSsm51v4j+f7mL9gTOlHtPq9Tuv5par6zs+cf44ZKdxMvM4g7ZOwgz46334KG4EU/9awB9GL7zNZv7bdATDrptS7nGuZLm5uWi1WnJyciTduhOShl4IIdxD0tBfhjpHWuYbnNXpyNQoBJjdEPPKHDLAfg0uq32nMiu1BldFE3lcyt+o59YO9YkIMJJwpOJrkRWqKpm5JjJzTfxz9mKF6lzaizakXX3qBxlt85GsaqoHrSrDIKsz/6w6wx7LSqQx9eZWtKlvWfQ4O9/E+78c5eudJzly9kK5Qw9t+3c2PPX8cXgzHtWUx+zwUBQfb1AULhZkM/LAu2QZvfAzm3k25mZu9vDgC8BotLyGOTk5bm6JEEII4ToSgNWCKL8ovFDIQ+UfvZ4Oec57LlxK5pAB1VtH7FKVTeRhVZEhjJGBRn56uheJadnsOJrOnpPnOXz2IsnnczifU8DFPBMVieFnrz7E7NWHANAqYNBp8TVo8TfqOHE+p9rz0KoaxFnnn2XlFti2Xcgz2XqiylKRdPBVae+oj3dyXZMQjp3L5kS642tjXcvLGevvrHNcsOOT2WlgymO1rw/rfX1sm82KQpZWi6/ZzLvJZ2jTv+wsfleawsJCXnvtNT7//HOSkpLIz7f/TDx69Kh7GiaEEELUAgnAakmIzpdTpgv8bTC4JwCTOWRA9dYRK6kqiTwqshbZ1JtboddpaRbhT7MIf8B+kn7CP2mMeLfivWgAhSrkFBSSU1BIailDF62sPWjNpqzEqNPirdfi46XF36gn2NdAsK+BED8DX+w4UeUgrqoJRCrai+Vw/mUk0rBu+/mfNNs2BWgS7sftHetzf7dG/HTobJnDTksupnypNI2GGSHBKKqKeunaUUWjv6NMJqf1rmTTp0/nvffeY/z48Tz33HNMnjyZo0eP8u233/L888+7u3lCCCGES0kAVktijGGcunCBQwa9expgHWKYf8kwtsh2lZtDdgUMY6zuOmI1oSpDGC9V0V60n5+5gQu5BRw+e5F/zl4gKS2bk+dz2Hsyg0NnLjipaa/QDBfzC7mYXwgXASo+LMwaxC3bnsTtHepjvGSOXXWGP1a27vG0bH7+J5XVe5IrlEijabgvIzo35J5rG2LQFQ/brOrvTFVVXggNJluj2AdfAIpCLvBiaDCvlduyK8snn3zCu+++y0033cT06dMZMWIETZo0oV27dmzdupX777/f3U0UQgghXEYCsFrSIrQNv11I5C8v51ntAMs6YD7OF+SsE2QYI1Bz88iqMoQRKt6LptUoBPoYiG9oIL5h8QKxFe1Bu/fahngbNJzNyiftYj7ns/PJzC0gK7eQrNx88kzlj4N89ps9PPvNHjQKeOm0+Bg0nM8xldkTNfmbPYT7e+Fn1OPrZXlN8wvMZOcXMvnbPWXWffzTXQR67+Fivom8UtL9l+XxG5o6TaSReuoYVxUe571+ev7z5VFMaHm8Z0M6ROrRFv5D6ql8QqMbOtQ7fPGE3dDDkgoVhXW+Phy+cIKruLqSrb18paSk0LZtWwD8/PxsWQcHDx7Mc889586mCSGEEC4nAVgt6d9qBB8cXcE+g4HUu5YS+mmJOR/t74LezzpfA6yuqO4wxiugBw1qbh5ZVdcig+r1olW0B23qkNaltqmywyDNlwyBLE/axXxuX5BQbjlnTGbVITukRgFvvdbSk1eO0tZ5+/vH/6Pr8XcB2GQt8lvx8wkxjxD60ByHetnJfxBQWEhmUTrykrSqSu/sHK7ya1Bu264kDRo0IDk5mdjYWK666irWrFlDx44d2b59uyw8LIQQ4oonAVgtaRvWFo2iwYyZd1K3cWvRUMSwkBaEJf9F/l/LyOjxBHV6ydXqDmOsiR60OhDE1eQ8suqoai9aZXrQSlPRIG7DU71ITL3AodMXOHL2Ar/+k8aOY+mVOc1Ki29Yj+5NQmkS7se1jUOICDBSaFa57tUN5bbXaSINIPiGu1md0hGlMI8GPz0NwInrZ6NqLb/rRpEt7cqbzWbe/PwmPslJIlurtc334pJhiIqq4ms2MyW14ssSXCluu+021q9fT5cuXXjyyScZMWIEixYtIikpiXHjxrm7eUIIIYRLSQBWi/z0fmTmZ/LZ31/yWX1LD4XenM4KnZb6pkJSP7qFsHGH3NxKF6qJRCB1YBhkdeeR1cQQRquq9qJVdx5aRYM4b4OWVtGBtIq2JMfoWsGes88euZYucUFczC9EQcGo07D9WHqF6k7o14KuTeyH8lY36FyfkcCCQwssPxT97fLPXNvzo71H05wOAJxM2c3U70bwm5cONBo65ObS/0I2r4TaB3eqovBcWjohZnO553SleeWV4r/hYcOG0aBBA3799VeuuuoqhgwZQmZmphtbJ4QQQriWBGC1qF/Dfnz595dosPSEAegNfpzt/AjRvy6kZcZp+HYM3DrfzS11kZpIBFIXhkFWcx81MYTR3fPQrPUrG8RVtOesc1wwGo2Cv1FTpbo11V6rfzX7F71iepGbe5771j0GwL/bPsr1DfsAEOZt6bt+78t/8XHGXtK8dGhVldvyFJ49k4GuMI/t3kY2+XhTqCi2oYcDLmbX/bmfteDaa6/l2muvdXczhBBCiFohAVgtGtV+FF/+/aUt+ALIK8zlA30+r8X1hMTNsPsTqNcQej3jxpbWYXVhGGQ19zGytRc3hztPlw5QL6z8IYzVDeKcBXB+Xjr2J1t6HiraC1fZIK46PVHV7cWqaiINgLD8XMLy8jmZlYZGVTEDH+7/iH8FtyPEEMiBpO+ZsP1VdnrpQaelfoGJ/9S/kZsG/p9lMebsNJ7OPM7GrZNAVfHR+zBl0FtgCLQEX3V57mcN+e677ypctlevXq5riBBCCOFmEoDVogjfCPz1/mQVZNm2Fapm1iWtY1WPVxlw7h/IOAGbXrY8eWkQVvQlDtMlqcBT/izu+fGQL3HVVhPDIKu5j9CDnxJaXgAXXXYQWN0g7tstO1n+s2Vdq9ZF8crUtxNtz99yXUcevalbmfuoahBXnZ6oAQ0VFgwKZtrm86RcLL6REemnZer1gQxoWHrvXVUTaXD+OLwZj2rKY3Z4KIqPNygK2QXZzFj5IB1z81gQFMhFLz06VWVgvsJ/h66gXshVAJzNz+Rs5lFy87MwF80Bu7dBX07npnM6N50wnb5uz/2sIbfeemuFyimKQnq6a+cJCiGEEO4kAVgtSstJI68wz2G7gsKM317kmodWE7KwF2SftQRhqQfh9vcg8yS8GQ+mEnWtX/7BMozp8d8lCCtPTQyDrO4+aiAIrG4Qd7d2PY96OQk2ilzUTgDKDsCqGsRVpyfq4i/vMmDrHPqqCtv0LThDPcI5T+eCA2g3qFzMnoDvAOdpzJsO/A+Hz96KYsqlyYqhAPxz81eoOks01jSslL+d7DQw5bHa18cupXyhorDB14cNRdta5OXzeLMR9Ow1za76F7+8yILMPXbb3jr6HW8dtfQIjQ5ow5jbPnN+7CuIuRJz3WQOmBBCiCuZBGC1RFVVXtj6AoWqYypsFZXsgmxe/P1/vPbkbljQDc4fgz1fwdFf4Kb/OQZfJZnyLF8UJQCr++rAXDjf7o9Au5tLre9bgSCwqkFclXuigE8K+7A8Lwgj+XzlNR2AoXlT+f/27jy+qSrvH/jnJk2Tpku6t1RaCrIUZJMCpTgIONAWRcQREcECgggCIjCK7KsDgv6QZ2RTEHAcnAcRmEFmplCQFh/ZV1HLJmUTQkvpXtqmyf39ERoIWXrbpkmXz/v1yovm3HvOPTm55OSbc+65xTDeX+8FfSe8aaM+gWFNjIFdaSHwnTHt8Xaxkto9SybDwgB/CKJocUNlQRTxXlY2XnnpG7hHPmWR9+WnZqPX3YsW6eWC/FtUeHwiIiKqPxiAOcmlnEvYd22fze16UY+91/biUscJaD7pNPCPV4CLe4ACLbDltepXgFMY65fqBnEOCAKrGsRVeSQKwMAendC94xMQdEXAJmMAtnDsqxAVxlEoe7cAyCzKROa9TEB3D7h/GwjcPQcojPUN8ghCkNpyMqAoilgU6I9CmWARfAHG689OeqiQaKPtgoKfQFDwEzbr1ZB8//33mDhxIg4fPgwfHx+zbbm5uejevTvWrFmDjh07uqaCRERETsAAzEma+zbHHyP+iJTrKVZHweSCHL3De6O5n/G6EQzbCpz5X+BfEwGDrnoHv38NC6cwkkNVMYirzkiU6RYApQ8+up4I85GU97+n12HXz18CAPKCg6B1kyN01zD4GIzLefRvOwLDu880y1N89wpWJo01m3r4KIMgYK+nGpcKbqA5OlZYj4ZsxYoVGDNmjEXwBQAajQZjx47FJ598gi+//NIFtSMiInIOBmBOIggC5nSbg6O3jqJAVwDxkYW0PRWemN1ttnmmDkOAlgnAtjeAS8kVH+RvLwABzYFGHYCmTwPN+wBKL9M1LHZJmcLIUTSqq3KuI3HfCgzXlyBLJsPzjcOgB5Ank2HzzVsIMBgg3l4BtEkEfB7D7yfW4+8n/gfJCgNuu93/mBRFsxsplytfUr65V2OnvqS66MyZM1i6dKnN7XFxcfj4Y9vTWomIiOoDBmBOFOARgLmxc/Hegfcstg1tPRQBHlbuBeThCzwzW1oAVpwD/H7c+Dj+hTFNpqjctUW2OGIUjQEcVVNVpxGiKAuCvgQigEWB/iiSCcaVDGUyfBDoj08y7sCgL8G/v3kRe/Q5OOihgs5DBkAGjV6PgfmF2ObjhULAbBqiIIrwNBgw+87dGn/t9cHt27ehUChsbndzc0NmZqYTa0REROR8DMCcLD4yHv/+bSdSfv/BLH39T+vx9GNPo11QuwrLSPJU40N/P8zIuov4ooeCmcefAfJuGldNLLm/1L1BZwzMpDj4KdCsFxD2JBAUBcgfOj2qO4rmqGmQDOIatKpMI3yYtZUM93qq8U5wIH5RuuO2WwkA47kUoStDT99WeCv6XXh/+RzalpbiveBAs/JEQcCcrGwEVGKFv4bssccew9mzZ9G8eXOr23/66Sc0amT7NgRERET1AQMwJyr/9f6lxweYAjCVXIUSfQnKxDIM+88wLHpqEV5o/oLNMspXY8uXCVgQ6I/ON249+PL3x3lAWEfj3wYDcPMUcPVH4HIK8JvtBUBMfv7W+CgncwPcvQAPP9MIQ5U5YhpkbRmFYxBYLVUexarMNEIr7W9vJcPypeS9DAa0FzwwMHoS+rUbbjou3JSILyxCUmERUtQe0AuCaephQmGR8bxTWxnBJjPPPvss5s6di379+kGlMr9H3L179zBv3jz079/fRbUjIiJyDgZgTrT1wlasObPGLK1YX2z6W4SI2T/Oxr9++xdW9F4BH/f7F6qrAyDKlYC+xOb0KVGuhPDwF0CZDGgcbXw0fdosALM5guamBvQlQPkiIYYy4+iZ1BE0APhmBODTCFAHAd4hgHcj45fhsmouJALUjlG4+hIEurAOVR7FkjCNUNA/9P7ry4DLqcDFJNy8mISpIUEosLGSIUQR7UpKsa7fJnhGPLJ0vm84st5IRm72ZQwp1GL/z6sAUYTKTYXEP7yPywovaPyaIYBBd4Vmz56N7du3o2XLlpg4cSJatWoFQRCQlpaGVatWQa/XY9asWa6uJhERUY0SRFEUK96tYcnLy4NGo0Fubq7V1bqqyuyX/w3xxsRRuwGFB+4U3cG8g/Nwp/gOAEAmyNC/WX/M6jYLajc1kHMdSZf/jffOrrIo96N2E5HQ7FnbX5hvngY+7wkAppGDfJkAb4MB3z08gvZmqnEErbQIuHUauPUTkHnOeE+y7KvA3d9MRdoM4iSymb/JU4BfpPHLu2cg4BVifPg0Bu5lPWg3e8pfh512qFJ+R5SRcx3ip9HGQMEGUa6E8Lb9AK5aZdRAHay9n1bLeCifrXPR5rHvt32Sp9piKiAAfJRxxzgaFdAc2qIM/CTo8JNKiaMqFdKU7jZf68N2xC5B85aWIzCrT6+2+PHkYW91eAvjO46XdIyG7urVq3jrrbewe/dulHc/giAgPj4eq1evRmRkZI19Btd1bBciItdw9OcvR8CcKEh9f2pVaSFQen9EyD/KuEhGALDv5X348OiH2HJhCwyiATt/24nvfvsO7YPaIyY0Bl+lfQUBgtkKigIELDz/Jbq0HoSKJkDZGzkw464GmnQ3Pso9EsTZnAYZ2g7Q64CSPOPrLCsB9KWA+OAaGbv5r/5ofEhgM4jbNhrwDAIUakDp/eBRUiAtfyVUuoz7ozj28puN4tREGQ6ug63302oZlR3FAozTaXOvA+kHbE8jFEXMCfRHstoDvyoLccPH16LKXnqDzXt5VbSS4cthPdFLFQboi423hgCAF1YCcuM0Ot5MWbomTZrgP//5D7Kzs3Hp0iWIoogWLVrAz8/P1VUjIiJyCgZgzpSvNT5sTNmSeYdiZreZmNhpImb9MAupN1IhQsSZzDM4k3nGapEiRBSUFmDEf0fg9bavI1ITiRZ+LR5MXwSMo0luSuxWyq0uQJDkqUZCiV7SNSwVBnEDVlof+fn9FLCuV8X5vUIAgx4oKzYGcgadWfBWzm4Ql3XJ+LDDbn7TCJcACDLjQyYHBLn0Mra/Cag0gFxhfMgUgNzdOLooJf+pzUD6gft53YzHLv8773dpZeRcBxSexumoEO7/CyD/lrT8JflAcZ7x9QMP2kKQGd8jSDgf7lwAykqNU1vLioFbxvPY1mIYSZ5q4yjWl/2N779eZ5oSWygImBYSaD2IEgQUCwL2eN1f8VMU4SW4IcQnAp1CO2Oof0f4bRmB5xuHoUBW+ZUMg37ZiaDUD80Tt4598HfP6QBvtlwpfn5+6NKli6urQURE5HScgmhFjU3z2L8EePRL3MN6Tgd6zzA9LSorwtrTa7Hj0g7klORU+nAKmQIqNxW8FF7wExS4UHADZbAMZrzcPLDzj+sQFNrBdmFSp39VMP2vyvl1xcClvcCWYRABTAkOtFgMwfSlP7gNAMEY6JYV3x+F0xlH4sqKK84vQXXLqC91qPD9tKJ86uGj12MJogi1KGJR5h3ck8mhlcvxu8INV93ccE3hhkw3ab8XjWrcF0Ni3kMjr4dW07t/7V6SUm67viV629fulf94Ykv5TanJITjVzjq2CxGRa3AKYl3W+XWgVT/b2x/5Aqd2U2Nq56mYEj0FU1KmIOV6CvTlC2Q8QgYZBEEw264z6KAr1SG/NB+3rOYyKii7h2d2vwYBAtxkbnCTuUEhU0ApV0LlpoJKroIf5Gjn54e/+3ha3JBWEEXMD/RHE70Ib0MpvIpz4OPuA1n5iMt9tqaPCaKIhQF+6HKv2PY0SoUK0Binh1U4ejJwjd0gsOL8a43vRUk+UJoPlBTe/7vAOBXu7NaKywhsZRyxMpQ99NAbr/8rulNxfrk7AAGAaGxv07+4PyIoVlxGBaqbv6L3s0VJKdwhIl8mQ55Mhiy5HFlyOb719rS6GIYoCCgUBEwNCbZ5TIUoQgfYvSHylNbDgYeDL8AYVE08gfjCO/j3qf+HlDunAUGAHDL0Du6EhL7v2F94hAEWEREROQgDMGeq4pc4QRAwp9scHL11FAW6AotrwLzdvbFz4E4EeATAYDDgVtEtXMq+hKt5V3Ej/wYu5lzE8dvHKzyOCNEYtBl0uAfL65mO+npbz3f/i/PgRkHAvtHmdYcAQRDgJgKK8DAUCoLFl2dREFAgk2FoWCh8D82CQaE2CwQVcgUUMgU0xQVo7uON1X4aiyAQooi5gf7QAci89C10mUcgl8nhJrhBLpNDLsjhmXcLfh4qzAv0txlEBpeVoUQohk4pB5S+AHwh4MF+iqzLUF74F+YHWg885gf6o5GuDLqnJ6I00PxeR3qDHvI7l+D+3/cxz0p+iCLmBfrDw2DA3Z7vosgnDCJEGEQD9KIeelEPg2iAIucGgo98jgVWXkd5O+QJAq61H4R8tS8MhjKUGXQoM+hRJuqgLLyDxpf/D5s0PlbzzwoMwI8eKlwObIoCNwX0ogGiaIDh/kMvGqDQlaBIV4B8mfX3M18ux8DwMKvnixQBMhU8PYMRrA5GM99maBvQFl0FNVRfD6nyNMJMdxUy9Uq89MRrSDlgnAqpdFNhcIc38avSHUHuKlhZ/J6IiIjIoTgF0YraOs0jKT0J7x14zyL9o54fISEywUoOI1EU7Y6gyQQZooOj8UqrV5BxLwM5JTnIK8lDXqnxUagrRHZxNq7kXXHgq6GGQBABCMaRVaVcCX9RhrJ7Wbjl5mZ3MYxPBm6zHMWs5jRCrmRYd9TWz2BXY7sQEbkGpyA2YPGR8fj3bztNN3GWCzL0Dn/GbvAFVDyC5qXwwkc9P0KAh+1FOCoK4uSCHL3De2NJjyXIK81DTkkO8kvzUVBagLzSPBSUFmDLhS24nHPZ7PgP81P64XHfx42jcHod9KIeZYYylBnKoBf1uFd2D1nFWRW2k7vMHYJwf7VI0TiyJ0KEYDBAZ+UaOIv2AmzU0P42KfvJRMBg5TZUj1JCgF4mN42+CRCMa4JAgJtej0KUVViGv0wFg0IFmSB78IAMbroi3NDlVZi/s3dTuHmFQC7IjSOS90cT3WRu8CrMwdnr+3He3R0GK4GUTBTRq+ge/ufRQOrmaWSt7121UaxqTiN8ueXL6BXey+ptIADjDaCJiIiIahoDsDqi/B5iLz0+wBSAKeUqDG41GL9m/Yogj/tL3NsQ4BGAubFzLUbQRIiYEzvHbvAFVBzEeSo8MbvbbOM1Y24qBKstr+OJi4zD8zuetzmNcscLOxwSBH7S+xPb+ZPHIeXWYeitBGJyQYbeod3wSdxn9utQjTLE7GuY8r9/RIqHEnpbI0D3SvDJq9/bDCSkl/Efq2VIzv/CKrv3Acs69U88HxpoNZDyMhgwN6fAcmVNdQACZArMzbprMYolCgLmZGUjQKawuSJndaYRBun1CCopNa7KWH4biNJSwHB/dUt369dXUv2RnZ2NSZMmYefOnQCAAQMG4NNPP4Wvr6/NPKIoYsGCBfj888+RnZ2NmJgYrFq1Ck888WDVy88//xxff/01Tp48ifz8fGRnZ1uUWZVjExFR/SSreBeqDbZe2IpXdr2Ctx8KoIrKivBm8pt4Zdcr2Hphq938mUWZCPcOR3TQk6Y0GWToGtoVEd4RyCzKrLAO5UHcoyNYUoO46uYvDwLVbmqz67IA8yDQbv4ei6FWeNrI74XZPRZXXIdqlCH4RWDOn7bfz//INsD4Gv603Xbg44AyHFEH+IYjYPxxzO34ttXFNOZ0fAcB449bllE+ijX0P+gV0MG0sIgcMvQJ7oyEYf+1vRIhqvn/4PhG4y0GNjw0YrwhwZj2eU/jdqrXhg4ditOnTyMpKQlJSUk4ffo0EhMT7eZZtmwZli9fjpUrV+LYsWMIDQ1F3759kZ+fb9qnqKgICQkJmDlzpkOPTURE9ROvAbOiNs6zLx8Bszd9yt4ImCOuf8ksykRGUQY+OroUJzJPATAGcZ1DO+PPnf9cYR0A46/Jk/ZNsJhGaWvUypqqXgvnqPyswwNVeT/Lz+WMvGumQErtpsaK3iugUWrsnkfV+n/ApeTrjJr4DE5LS0ObNm1w+PBhxMTEAAAOHz6M2NhYnDt3Dq1atbLII4oiwsLCMHnyZLz//vsAgJKSEoSEhGDp0qUYO3as2f4pKSno3bu3xQhYVY5tTW3sm4iIGgJeA9ZABanvf7EsLXwwfco/CnD3lJT/4etfDv3jBWzU+OD1Tm8jNtx402Ep179svbDVIogzwICj2qN4ZdcrFQZx1Z1GWV5G+Ujeo0Fg+UievTKqm591MC+jKu+ntfOofBQLsP9jQLX+HzDAatAOHToEjUZjCoAAoFu3btBoNDh48KDVICg9PR1arRZxcXGmNKVSiZ49e+LgwYMWAZgjjw0Yg72SkhLT87y8iq/dJCKi2o8BWF1R/ut92UPLw2t/AtyMv/xX9OXy4S+ubXLzMTo3H3htuOQADqh+EFedL972yqhMEFjd/KyD/TKkvJ/VWgyjmv8PqOHSarUIDra8NjU4OBharfWR0fL0kJAQs/SQkBBcvXq1Ro8NAEuWLMGCBQskH4eIiOoGBmB1xfGNQOqH5mkPX8vSczrQe0aNVqG6QZwjVqGrbhDoiJFA1sGyjMq8n9UaxaoF/w+odpk/f36FQcqxY8cAGK/hfJQoilbTH/bodil5KipDSjkzZszA1KlTTc/z8vIQHm7n2kwiIqoTGIDVFZ1fB1r1s729ol/9a8HIQXWnUT5aRlWCQEeMBLIOlmVU6v2szrko8f+BXq+HTqeT+ErI2RQKBeRyuUPKmjhxIoYMGWJ3n8jISPz000+4ffu2xbbMzEyLEa5yoaHG80mr1aJRo0am9IyMDJt5bJVT2WMDxumOSqVS8nGIiKhuYABWV1Q3QHLEyEEtCOKoHqjOuVjBOSaKIrS3biEnJ6f69aQa5evri9DQ0EqPJD0qMDAQgYGWN+Z+VGxsLHJzc3H06FF07doVAHDkyBHk5uaie/fuVvM0bdoUoaGhSE5OxpNPGleQLS0tRWpqKpYuXSq5jlU5NhER1V8MwBqK6o6gAdUP4hwRwFW3DNbB9XVwxLlog1arRU5ODoKDg6FWq6v95Z4cTxRFFBUVISMjAwDMRpZqUuvWrZGQkIAxY8bgs8+M9+l788030b9/f7NFMKKiorBkyRK8+OKLEAQBkydPxuLFi9GiRQu0aNECixcvhlqtxtChQ015tFottFotLl26BAA4e/YsvL29ERERAX9/f8nHJiKihoHL0FvBpX5tqO4y3vuXWAZwD5MyClfdMliH2lMHB9Pr9bhw4QKCg4MREGD/nnLkellZWcjIyEDLli0tpiPW1Gfw3bt3LW6GvHLlSrMl4wVBwMaNGzFy5EgAD27E/Nlnn5ndiLlt27amPLauQ3u4HCnHrgj7JiIi13D05y8DMCvYydUQR9yHqbplsA61pw4OVlxcjPT0dERGRsLDw8Npx6WquXfvHq5cuYKmTZtCpVKZbeNnsHVsFyIi1+B9wKjucsQX8uqWwTrUnjrUkKpOO8zIK0ZGfonN7cHeSgT7qGxup8rh9FAiImqoGIAREQHYfOQa/mffRZvb3/ljC0zp29KJNSIiIqL6iAEYERGAYTER6NsmBMU6PQatPQQA+HZcLFQK4/VJwd5cDpyIiIiqT+bqChAR1QbBPiq0fUyDVqHeprSCkjK0buSDto9panT6oVarxdtvv41mzZpBqVQiPDwczz//PPbt24fS0lIEBgbigw8+sJp3yZIlCAwMRGlpqaRjjRw5EgMHDjRL+/bbb6FSqbBs2bJK112n0+H9999Hu3bt4OnpibCwMAwfPhw3b96sdFlEREQNgcsDsNWrV5suwo6OjsYPP/xgd//U1FRER0dDpVKhWbNmWLt2rdn2devWoUePHvDz84Ofnx/69OmDo0eP1uRLIKJ6IunnW+izPNX0fOTGY/jD0u+R9POtGjvmlStXEB0dje+//x7Lli3D2bNnkZSUhN69e2PChAlwd3fHa6+9hk2bNsHamkkbN25EYmIi3N3dq3T89evXY9iwYVi5ciWmTZtW6fxFRUU4efIk5syZg5MnT2L79u24cOECBgwYUKX6EBER1XcuDcC2bNmCyZMnY9asWTh16hR69OiBfv364dq1a1b3T09Px7PPPosePXrg1KlTmDlzJiZNmoRt27aZ9klJScGrr76K/fv349ChQ4iIiEBcXBx+//13Z70sIqqDkn6+hbf+fhK388wX4tDmFuOtv5+ssSBs/PjxEAQBR48exaBBg9CyZUs88cQTmDp1Kg4fPgwAGD16NH777TccOHDALO8PP/yAixcvYvTo0VU69rJlyzBx4kR8/fXXeOONN6pUhkajQXJyMgYPHoxWrVqhW7du+PTTT3HixAmbn+VEREQNmUsDsOXLl2P06NF444030Lp1a6xYsQLh4eFYs2aN1f3Xrl2LiIgIrFixAq1bt8Ybb7yBUaNG4eOPPzbts3nzZowfPx4dO3ZEVFQU1q1bB4PBgH379jnrZRFRLSCKIopKyyQ98ot1mLfzF1i7J0d52vydvyK/WCepPKl397h79y6SkpIwYcIEeHp6Wmwvv0dUu3bt0KVLF2zcuNFs+4YNG9C1a1eze1JJNX36dCxatAi7du3CSy+9ZLZt8+bN8PLysvvYvHmzzbJzc3MhCEKl7nFFRETUULhsEY7S0lKcOHEC06dPN0uPi4vDwYMHreY5dOgQ4uLizNLi4+PxxRdfQKfTQaFQWOQpKiqCTqeDv7+/4ypPRLXePZ0ebebudkhZIgBtXjHazd8jaf9fF8ZD7V7xx+ulS5cgiiKioqIq3HfUqFF49913sXLlSnh5eaGgoABbt27F8uXLJdXpYf/973/xr3/9C/v27cMzzzxjsX3AgAGIiYmxW0ZISIjV9OLiYkyfPh1Dhw7lvaqIiIiscFkAdufOHej1eotOPCQkBFqt9Zu7arVaq/uXlZXhzp07aNSokUWe6dOn47HHHkOfPn1s1qWkpAQlJQ+mHeXl5VXmpRARVUn5SJmUe2K9+uqrmDp1KrZs2YLRo0djy5YtEEURQ4YMqfRx27dvjzt37mDu3Lno0qULvL29zbZ7e3tbpEmh0+kwZMgQGAwGrF69utL5iYiIGgKXL0P/6BcPURTtfhmxtr+1dMB4fcM//vEPpKSkQKWyvYLZkiVLsGDBgspUm4hqOQ+FHL8ujJe079H0uxi58ViF+216vQu6Nq14NN3j/tL1FWnRogUEQUBaWprFyoSP0mg0GDRoEDZu3IjRo0dj48aNGDRoUJVGmR577DFs27YNvXv3RkJCApKSkswCrs2bN2Ps2LF2y/jss88wbNgw03OdTofBgwcjPT0d33//PUe/iIiIbHBZABYYGAi5XG4x2pWRkWFzaktoaKjV/d3c3BAQEGCW/vHHH2Px4sXYu3cv2rdvb7cuM2bMwNSpU03P8/LyEB4eXpmXQ0S1jCAIkqYBAkCPFkFopFFBm1ts9TowAUCoRoUeLYIgl1U8WiWVv78/4uPjsWrVKkyaNMniOrCcnByz66hGjx6NXr16YdeuXfjxxx+xePHiKh87IiICqamp6N27N+Li4rB7925T0FTZKYjlwdfFixexf/9+i89jIiIiesBli3C4u7sjOjoaycnJZunJycno3r271TyxsbEW++/ZswedO3c2u/7ro48+wqJFi5CUlITOnTtXWBelUgkfHx+zBxE1HHKZgHnPtwFgDLYeVv583vNtHBp8lVu9ejX0ej26du2Kbdu24eLFi0hLS8Nf//pXxMbGmu3bs2dPNG/eHMOHD0fz5s3x9NNPV+vYjRs3RkpKCrKyshAXF4fc3FwAximIzZs3t/soHzErKyvDoEGDcPz4cWzevBl6vR5arRZarVbyvcmIiIgaEpeugjh16lSsX78eGzZsQFpaGqZMmYJr165h3LhxAIwjU8OHDzftP27cOFy9ehVTp05FWloaNmzYgC+++ALvvvuuaZ9ly5Zh9uzZ2LBhAyIjI01fBAoKCpz++oio7kho2whrXuuEYB+lWXqoRoU1r3VCQlvLa0wdoWnTpjh58iR69+6NP//5z2jbti369u2Lffv2WV0RdtSoUcjOzsaoUaMstl25cgWCICAlJUXy8R977DGkpqYiJycHffv2RU5OTqXqf+PGDezcuRM3btxAx44d0ahRI9PD1oJKREREDZkgSl0vuYasXr0ay5Ytw61bt9C2bVt88sknpl91R44ciStXrph9mUhNTcWUKVPwyy+/ICwsDO+//74pYAOAyMhIXL161eI48+bNw/z58yXVKS8vDxqNBrm5uRwNI6ojiouLkZ6ebrqxe1XlF+tMqx1uer2Lw6cd1qSUlBS8+OKLuHz5Mvz8/FxdHbvsvV/8DLaO7UJE5BqO/vx1+SIc48ePx/jx461u27Rpk0Vaz549cfLkSZvlXblyxUE1I6KGJCOvGBn5JSjW6U1pXko3pN0yrooa7K1EsE/VAztnSEpKwsyZM2t98EVERNSQuTwAIyKqDTYfuYb/2XfRLG3Q2kOmv9/5YwtM6dvS2dWqlA8//NDVVSAiIqIKMAAjIgIwLCYCfdtYX4EVMI6AEREREVUXAzAiIgDBPqpaP8WQiIiI6j6XroJIRERERETUkDAAIyIiIiIichIGYERERERERE7Ca8CIiAAgX2t82OIdanwQERERVQMDMCIiADi+EUi1s4x7z+lA7xnOqw8RERHVSwzAiIgAoPPrQKt+QNk9YEOCMW1UEuDmYfybo19ERETkALwGjIgIAPS66m2vAkEQ7D5GjhxZ5bIjIyOxYsUKh9WViIiIHIMjYEREOdeBldFAWYl5evlIGAC4KYGJJwDfcIcd9tatW6a/t2zZgrlz5+L8+fOmNA8PD4cdi4iIiGoHjoARERVlWQZfjyorMe7nQKGhoaaHRqOBIAhmaQcOHEB0dDRUKhWaNWuGBQsWoKyszJR//vz5iIiIgFKpRFhYGCZNmgQA6NWrF65evYopU6aYRtOIiIiodmAARkT1kygCpYXSHmX3pJVZdk9aeaJY7erv3r0br732GiZNmoRff/0Vn332GTZt2oS//OUvAIBvv/0Wn3zyCT777DNcvHgR//znP9GuXTsAwPbt29G4cWMsXLgQt27dMhtpa8iys7ORmJgIjUYDjUaDxMRE5OTk2M0jiiLmz5+PsLAweHh4oFevXvjll1/M9vn888/Rq1cv+Pj4QBAEq2VGRkZaTDGdPn26A18dERHVFZyCSET1k64IWBzm2DIfnpJoz8ybgLtntQ71l7/8BdOnT8eIESMAAM2aNcOiRYswbdo0zJs3D9euXUNoaCj69OkDhUKBiIgIdO3aFQDg7+8PuVwOb29vhIZy8ZByQ4cOxY0bN5CUlAQAePPNN5GYmIjvvvvOZp5ly5Zh+fLl2LRpE1q2bIkPPvgAffv2xfnz5+Ht7Q0AKCoqQkJCAhISEjBjhu2VMhcuXIgxY8aYnnt5eTnolRERUV3CAIyIqBY6ceIEjh07ZhrxAgC9Xo/i4mIUFRXh5ZdfxooVK9CsWTMkJCTg2WefxfPPPw83N36sW5OWloakpCQcPnwYMTExAIB169YhNjYW58+fR6tWrSzyiKKIFStWYNasWfjTn/4EAPjyyy8REhKCr7/+GmPHjgUATJ48GQCQkpJitw4MiImICGAARkT1lUJtHImSQvuTtNGtUUlAaHtpx64mg8GABQsWmL74P0ylUiE8PBznz59HcnIy9u7di/Hjx+Ojjz5CamoqFApFtY9f3xw6dAgajcYUfAFAt27doNFocPDgQasBWHp6OrRaLeLi4kxpSqUSPXv2xMGDB00BmFRLly7FokWLEB4ejpdffhnvvfce3N3dq/6iiIioTmIARkT1kyBInwboJnG1QTePak8tlKpTp044f/48mjdvbnMfDw8PDBgwAAMGDMCECRMQFRWFs2fPolOnTnB3d4der3dKXesCrVaL4OBgi/Tg4GBotVqbeQAgJCTELD0kJARXr16t1PHfeecddOrUCX5+fjh69ChmzJiB9PR0rF+/3maekpISlJQ8WBwmLy+vUsckIqLaiQEYEVEtNHfuXPTv3980WiKTyfDTTz/h7Nmz+OCDD7Bp0ybo9XrExMRArVbjq6++goeHB5o0aQLAuOjDgQMHMGTIECiVSgQGBrr4FdWM+fPnY8GCBXb3OXbsGABYXQ1SFMUKV4l8dLuUPI+aMmWK6e/27dvDz88PgwYNwtKlSxEQEGA1z5IlSyp8bUREVPdwFUQiInWA8T5f9rgpjfs5SXx8PHbt2oXk5GR06dIF3bp1w/Lly00Blq+vL9atW4ennnoK7du3x759+/Ddd9+ZvswvXLgQV65cweOPP46goCCn1dvZJk6ciLS0NLuPtm3bIjQ0FLdv37bIn5mZaTHCVa78eq1HR8gyMjJs5pGqW7duAIBLly7Z3GfGjBnIzc01Pa5fv16tYxIRUe3AETAiIt9w402Wi7KMS82XXw82KunB9ER1gENvwvyokSNHYuTIkWZp8fHxiI+Pt7r/wIEDMXDgQJvldevWDWfOnHFgDWunwMBASaN7sbGxyM3NxdGjR02rRR45cgS5ubno3r271TxNmzZFaGgokpOT8eSTTwIASktLkZqaiqVLl1ar3qdOnQIANGrUyOY+SqUSSmUFPwwQEVGdwwCMiAgA5BUsXFHRdqrVWrdujYSEBIwZMwafffYZAOMy9P379zdbgCMqKgpLlizBiy++CEEQMHnyZCxevBgtWrRAixYtsHjxYqjVagwdOtSUR6vVQqvVmkazzp49C29vb0RERMDf3x+HDh3C4cOH0bt3b2g0Ghw7dgxTpkzBgAEDEBER4dyGICIil2MARkQEAMc3Aqkfmqc9vDJiz+lAb9v3eKLab/PmzZg0aZJpVcMBAwZg5cqVZvucP38eubm5pufTpk3DvXv3MH78eGRnZyMmJgZ79uwx3QMMANauXWt2rdbTTz8NANi4cSNGjhwJpVKJLVu2YMGCBSgpKUGTJk0wZswYTJs2rSZfLhER1VKCKIqiqytR2+Tl5UGj0SA3Nxc+Pj6urg4RSVBcXIz09HQ0bdoUKpWq8gXka40PW7xDjQ9yCHvvFz+DrWO7EBG5hqM/fzkCRkQEMMAiIiIip+AqiERERERERE7CAIyIiIiIiMhJGIARERERERE5Ca8BIyICkFmUicx7mTa3B3kEIUhdf29oTERERM7BAIyICMDWC1ux5swam9vf6vAWxncc78QaERERUX3EAIyICMDLLV9Gr/BeKC4rxoikEQCACR0n4OnGxns6BXlw9IuIiIiqj9eAEREBCFIHoU1AGwSrg01pf/vlbwhRh6BNQJsan36o1WrxzjvvoHnz5lCpVAgJCcEf/vAHrF27FkVFRab9Tp06hZdffhkhISFQqVRo2bIlxowZgwsXLuDEiRMQBAH/93//Z/UY8fHxGDBgQI2+DiIiIrKPARgR0X2iKGLp0aWm50VlRfjg8Ac1ftzLly/jySefxJ49e7B48WKcOnUKe/fuxZQpU/Ddd99h7969AIBdu3ahW7duKCkpwebNm5GWloavvvoKGo0Gc+bMQXR0NDp06ICNGzdaHOP69evYu3cvRo8eXeOvh4iIiGzjFEQiovt2X9mNlBsppud6UY+91/Yi6UoSEiITauy448ePh5ubG44fPw5PT09Tert27fDSSy9BFEUUFRXh9ddfx7PPPosdO3aY9mnatCliYmKQk5MDABg9ejRmzpyJv/71r2Zlbdq0CUFBQXjuuedq7HUQERFRxTgCRkT1kiiKKNIVSX7cyL+BBYcWQIBgVo4AAQsPLsSN/BuSyxJFUXI9s7KysGfPHkyYMMEsYDKrgyBg9+7duHPnDqZNm2Z1H19fXwDAsGHDoNPpsHXrVrO22LRpE0aMGAE3N/7uRkRE5ErsiYmoXrpXdg8xX8dUuxwRIvJ1+ei3vZ/kPEeGHoFaoZa076VLlyCKIlq1amWWHhgYiOLiYgDAhAkTEBAQAACIioqyW56/vz8GDhyIjRs3YuTIkQCAlJQUXL58GaNGjZL8GoiIiKhmcASMiKgWEATzkbejR4/i9OnTeOKJJ1BSUlKpUbXRo0fjwIEDuHTpEgBgw4YNeOqppyyCPCIiInI+joARUb3k4eaBI0OPSNpXFEVM/2E6fvj9B+hFvcV2mSBDz8Y98WGPDyUfW6rmzZtDEAScO3fOLL1Zs2bGsjyMZbVs2RIAcO7cOcTGxtots0+fPmjSpAk2bdqEadOmYfv27Vi5cqXkOhEREVHN4QgYEdVLgiBArVBLeni6e2J+9/lQu6mtXgPmpfDCvNh5kst7dDTLnoCAAPTt2xcrV65EYWGhzf3i4uIQGBiIZcuWWd1evghH+Wt//fXX8eWXX+Lrr7+GTCbD4MGDJdeJiIiIag4DMCIiAAEeAZgbOxcizKf6iRAxJ3YOAjwCauzYq1evRllZGTp37owtW7YgLS0N58+fx9///necO3cOcrkcnp6eWL9+Pf79739jwIAB2Lt3L65cuYLjx49j2rRpGDdunFmZr7/+Om7evImZM2diyJAhNhf4ICIiIudiAEZEdF98ZDx6Ne5lei4X5OgT0adGl6AHgMcffxynTp1Cnz59MGPGDHTo0AGdO3fGp59+infffReLFi0CALzwwgs4ePAgFAoFhg4diqioKLz66qvIzc3FBx+Y368sIiICffr0QXZ2NhffICIiqkUEsTJXdjcQeXl50Gg0yM3NhY+Pj6urQ0QSFBcXIz09HU2bNoVKpapyOTfyb5hWPPRWeOO7F7+r0dGvhsre+8XPYOvYLkREruHoz18uwkFEBCCzKBOZ9zJRXFZsShv+xHDcLrqN20W3EeQRhCB1kAtrSERERPUBAzAiIgBbL2zFmjNrzNJWnV6FVadXAQDe6vAWxncc74qqERERUT3CAIyICMDLLV9Gr/BeNrcHeXD0i4iIiKqPARgREYAgNacYEhERUc3jKohEVK9wXaG6ge8TERE1VAzAiKheUCgUAICioiIX14SkKH+fyt83IiKihoJTEImoXpDL5fD19UVGRgYAQK1WQxAEF9eKHiWKIoqKipCRkQFfX1/I5XJXV4mIiMipGIARUb0RGhoKAKYgjGovX19f0/tFRETUkDAAI6J6QxAENGrUCMHBwdDpdK6uDtmgUCg48kVERA0WAzAiqnfkcjm/4BMREVGt5PJFOFavXo2mTZtCpVIhOjoaP/zwg939U1NTER0dDZVKhWbNmmHt2rUW+2zbtg1t2rSBUqlEmzZtsGPHjpqqPhER1RHZ2dlITEyERqOBRqNBYmIicnJy7OYRRRHz589HWFgYPDw80KtXL/zyyy+m7Xfv3sXbb7+NVq1aQa1WIyIiApMmTUJubm61j01ERPWTSwOwLVu2YPLkyZg1axZOnTqFHj16oF+/frh27ZrV/dPT0/Hss8+iR48eOHXqFGbOnIlJkyZh27Ztpn0OHTqEV155BYmJiThz5gwSExMxePBgHDlyxFkvi4iIaqGhQ4fi9OnTSEpKQlJSEk6fPo3ExES7eZYtW4bly5dj5cqVOHbsGEJDQ9G3b1/k5+cDAG7evImbN2/i448/xtmzZ7Fp0yYkJSVh9OjR1T42ERHVU6ILde3aVRw3bpxZWlRUlDh9+nSr+0+bNk2MiooySxs7dqzYrVs30/PBgweLCQkJZvvEx8eLQ4YMkVyv3NxcEYCYm5srOQ8RETlGTXwG//rrryIA8fDhw6a0Q4cOiQDEc+fOWc1jMBjE0NBQ8cMPPzSlFRcXixqNRly7dq3NY33zzTeiu7u7qNPpqnxsa9g3ERG5hqM/f112DVhpaSlOnDiB6dOnm6XHxcXh4MGDVvMcOnQIcXFxZmnx8fH44osvoNPpoFAocOjQIUyZMsVinxUrVtisS0lJCUpKSkzPy6eO5OXlVeYlERGRA5R/9ooOvFnzoUOHoNFoEBMTY0rr1q0bNBoNDh48iFatWlnkSU9Ph1arNet3lEolevbsiYMHD2Ls2LFWj5WbmwsfHx+4ublV+dgA+yYiotrC0f2SywKwO3fuQK/XIyQkxCw9JCQEWq3Wah6tVmt1/7KyMty5cweNGjWyuY+tMgFgyZIlWLBggUV6eHi41JdDREQOlp+fD41G45CytFotgoODLdKDg4Pt9jkArPYpV69etZonKysLixYtMgvOqnJsgH0TEVFt46h+yeWrID56o1RRFO3ePNXa/o+mV7bMGTNmYOrUqabnBoMBd+/eRUBAQK24kWteXh7Cw8Nx/fp1+Pj4uLo6dRbb0XHYltKwnaR5tJ1EUUR+fj7CwsIqzDt//nyrQcrDjh07BsCybwAq7h+s5bOVJy8vD8899xzatGmDefPm2S1DyrFrc9/Ec9tx2JaOw7aUhu0kzcPt5O3tLblfksJlAVhgYCDkcrnFr38ZGRkWvzaWCw0Ntbq/m5sbAgIC7O5jq0zAOKVEqVSapfn6+kp9KU7j4+PD/ygOwHZ0HLalNGwnaR5uJ6m/ME6cOBFDhgyxu09kZCR++ukn3L5922JbZmam3T4HMI5gNWrUyJRurU/Jz89HQkICvLy8sGPHDigUCrNyKntsoG70TTy3HYdt6ThsS2nYTtKUt5OjZmQALlwF0d3dHdHR0UhOTjZLT05ORvfu3a3miY2Ntdh/z5496Ny5s6mzs7WPrTKJiKjuCgwMRFRUlN2HSqVCbGwscnNzcfToUVPeI0eOIDc312b/0LRpU4SGhpr1KaWlpUhNTTXLk5eXh7i4OLi7u2Pnzp1QqVRm5VTl2EREVH+5dBn6qVOnYv369diwYQPS0tIwZcoUXLt2DePGjQNgnH4xfPhw0/7jxo3D1atXMXXqVKSlpWHDhg344osv8O6775r2eeedd7Bnzx4sXboU586dw9KlS7F3715MnjzZ2S+PiIhqidatWyMhIQFjxozB4cOHcfjwYYwZMwb9+/c3WwQjKirKdO9IQRAwefJkLF68GDt27MDPP/+MkSNHQq1WY+jQoQCMI19xcXEoLCzEF198gby8PGi1Wmi1Wuj1+kodm4iIGgiHrKVYDatWrRKbNGkiuru7i506dRJTU1NN20aMGCH27NnTbP+UlBTxySefFN3d3cXIyEhxzZo1FmVu3bpVbNWqlahQKMSoqChx27ZtNf0yalRxcbE4b948sbi42NVVqdPYjo7DtpSG7SSNs9opKytLHDZsmOjt7S16e3uLw4YNE7Ozs832ASBu3LjR9NxgMIjz5s0TQ0NDRaVSKT799NPi2bNnTdv3798vArD6SE9Pr9Sx6xKe247DtnQctqU0bCdparKdBFF04Dq/REREREREZJNLpyASERERERE1JAzAiIiIiIiInIQBGBERERERkZMwACMiIiIiInISBmDUIKxevRpNmzaFSqVCdHQ0fvjhB1dXiYiIGjj2TUQNEwOweurFF1+En58fBg0a5OqquNyWLVswefJkzJo1C6dOnUKPHj3Qr18/XLt2zdVVq5d47klz/fp19OrVC23atEH79u2xdetWV1epVsrPz0eXLl3QsWNHtGvXDuvWrXN1laga+PnwAPsm5+K5VzH2S9JVt2/iMvT11P79+1FQUIAvv/wS3377raur41IxMTHo1KkT1qxZY0pr3bo1Bg4ciCVLlriwZvUTzz1pbt26hdu3b6Njx47IyMhAp06dcP78eXh6erq6arWKXq9HSUkJ1Go1ioqK0LZtWxw7dgwBAQGurhpVAT8fHmDf5Fw89yrGfkm66vZNHAGrp3r37g1vb29XV8PlSktLceLECcTFxZmlx8XF4eDBgy6qVf3Gc0+aRo0aoWPHjgCA4OBg+Pv74+7du66tVC0kl8uhVqsBAMXFxdDr9eDvhnUXPx+M2Dc5H8+9irFfkq66fRMDsFrmwIEDeP755xEWFgZBEPDPf/7T6n6cNy7NnTt3oNfrERISYpYeEhICrVbrolrVXlLOP557Ro5sq+PHj8NgMCA8PLyGa+18jminnJwcdOjQAY0bN8a0adMQGBjopNpTOfZNjsW+qXLYN0nDfkk6V/dNDMBqmcLCQnTo0AErV660uQ/njVeeIAhmz0VRtEijis8/nnsPOKqtsrKyMHz4cHz++efOqLbTOaKdfH19cebMGaSnp+Prr7/G7du3nVV9uo99U81g3yQN+yZp2C9J5/K+SaRaC4C4Y8cOi/SuXbuK48aNM0uLiooSp0+fbpa2f/9+8aWXXqrJKtZ6JSUlolwuF7dv326WPmnSJPHpp592Ua3qBmvnH88966raVsXFxWKPHj3Ev/3tb86opstV55wqN27cOPGbb76pqSqSBOybqo99U9Wxb5KG/ZJ0ruibOAJWx3DeeOW4u7sjOjoaycnJZunJycno3r27i2pVN/Hck05KW4miiJEjR+KZZ55BYmKiK6rpclLa6fbt28jLywMA5OXl4cCBA2jVqpXT60r28fOhctg3OQ7PPWnYL0nnjL7JzXHVJWeQOm88Pj4eJ0+eRGFhIRo3bowdO3agS5cuzq5urTB16lQkJiaic+fOiI2Nxeeff45r165h3Lhxrq5ancJzTzopbfXjjz9iy5YtaN++vWnu+VdffYV27do5u7ouI6Wdbty4gdGjR0MURYiiiIkTJ6J9+/auqC7Zwc+HymPf5Bg896RhvySdM/omBmBOMH/+fCxYsMDuPseOHUPnzp0ll1nRvPHdu3dXrpL12CuvvIKsrCwsXLgQt27dQtu2bfGf//wHTZo0cXXV6iSee9LZa6s//OEPMBgMrqhWrWOvnaKjo3H69GkX1Kr+Y9/kWuybHIvnnjTsl6Sryb6JAZgTTJw4EUOGDLG7T2RkpKSyAgMDIZfLLVZJysjIsIjU6YHx48dj/Pjxrq5GncZzTzq2lTRsJ9di3+R67Juqj+eeNGwn6ZzRVrwGzAkCAwMRFRVl96FSqSSVxXnj5Co896RjW0nDdnIt9k1UH/Dck4btJJ0z2oojYLVMQUEBLl26ZHqenp6O06dPw9/fHxEREQA4b5xqTkXnH8+9B9hW0rCd6gf2TeRK/ByRhu0kncvbSvJ6ieQU+/fvFwFYPEaMGGG236pVq8QmTZqI7u7uYqdOncTU1FTXVJjqFSnnH889I7aVNGyn+oF9E7kSP0ekYTtJ5+q2EkRRFB0TyhEREREREZE9vAaMiIiIiIjISRiAEREREREROQkDMCIiIiIiIidhAEZEREREROQkDMCIiIiIiIichAEYERERERGRkzAAIyIiIiIichIGYERERERERE7CAIyIiIiIiMhJGIAR1TOzZ8+GUqnE0KFDXV0VIiIi9ktEjxBEURRdXQkicpy8vDx89dVXmDhxIi5evIjmzZu7ukpERNSAsV8iMscRMKJ6xsfHB6NGjYJMJsPZs2ddXR0iImrg2C8RmWMARlQPlZWVQa1W4+eff3Z1VYiIiNgvET2EARhRPTR79mwUFBSwoyMiolqB/RLRA7wGjKieOXHiBLp3746+ffsiPT0dv/zyi6urREREDRj7JSJzDMCI6hGDwYCuXbuiZ8+eiImJwbBhw1BYWAh3d3dXV42IiBog9ktEljgFkage+fTTT5GZmYmFCxeiXbt2KCsrw/nz511dLSIiaqDYLxFZYgBGVE/8/vvvmDNnDlavXg1PT0+0aNECSqWS8+2JiMgl2C8RWccAjKiemDRpEvr164fnnnsOAODm5obWrVuzoyMiIpdgv0RknZurK0BE1bdr1y58//33SEtLM0tv164dOzoiInI69ktEtnERDiIiIiIiIifhFEQiIiIiIiInYQBGRERERETkJAzAiIiIiIiInIQBGBERERERkZMwACMiIiIiInISBmBEREREREROwgCMiIiIiIjISRiAEREREREROQkDMCIiIiIiIidhAEZEREREROQkDMCIiIiIiIichAEYERERERGRk/x/f9krAkUXmskAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "color_cycle = plt.rcParams['axes.prop_cycle'].by_key()['color']\n",
    "marker_cycle = ['o', 's', 'd', 'v', '^', '<', '>', 'p', 'h', '*']\n",
    "\n",
    "fig, axes = plt.subplots(1, 2, figsize=(10, 4))\n",
    "\n",
    "sketch_i = 0\n",
    "\n",
    "for k, K in enumerate(Ks):\n",
    "    markevery = (5, 20)\n",
    "    axes[0].plot(lamdas, np.mean(cv_reg_errs[:, sketch_i, k, :], axis=0), color=color_cycle[k])\n",
    "    axes[0].errorbar(lamdas, np.mean(cv_reg_errs[:, sketch_i, k, :], axis=0), yerr=np.std(cv_reg_errs[:, sketch_i, k, :], axis=0), errorevery=markevery, markevery=markevery, marker=marker_cycle[k], color=color_cycle[k], capsize=3, label=f'CV, K={K}')\n",
    "\n",
    "    axes[1].plot(lamdas, np.mean(cv_cls_errs[:, sketch_i, k, :], axis=0), color=color_cycle[k])\n",
    "    axes[1].errorbar(lamdas, np.mean(cv_cls_errs[:, sketch_i, k, :], axis=0), yerr=np.std(cv_cls_errs[:, sketch_i, k, :], axis=0), errorevery=markevery, markevery=markevery, marker=marker_cycle[k], color=color_cycle[k], capsize=3, label=f'CV, K={K}')\n",
    "\n",
    "markevery = (5, 20)\n",
    "axes[0].plot(lamdas, np.mean(test_reg_errs[:, sketch_i, :], axis=0), color=color_cycle[len(Ks)])\n",
    "axes[0].errorbar(lamdas, np.mean(test_reg_errs[:, sketch_i, :], axis=0), yerr=np.std(test_reg_errs[:, sketch_i, :], axis=0), errorevery=markevery, markevery=markevery, marker=marker_cycle[len(Ks)], color=color_cycle[len(Ks)], capsize=3, label='Test')\n",
    "# axes[0].plot(lamdas, gcv_reg_errs[:, sketch_i, :].T, color=color_cycle[len(Ks) + 1], alpha=0.5, label=['GCV'] + [None] * (n_trials - 1))\n",
    "axes[0].plot(lamdas, np.mean(gcv_reg_errs[:, sketch_i, :], axis=0), color=color_cycle[len(Ks) + 1])\n",
    "axes[0].errorbar(lamdas, np.mean(gcv_reg_errs[:, sketch_i, :], axis=0), yerr=np.std(gcv_reg_errs[:, sketch_i, :], axis=0), errorevery=markevery, markevery=markevery, marker=marker_cycle[len(Ks) + 1], color=color_cycle[len(Ks) + 1], capsize=3, label='GCV')\n",
    "\n",
    "axes[1].plot(lamdas, np.mean(test_cls_errs[:, sketch_i, :], axis=0), color=color_cycle[len(Ks)])\n",
    "axes[1].errorbar(lamdas, np.mean(test_cls_errs[:, sketch_i, :], axis=0), yerr=np.std(test_cls_errs[:, sketch_i, :], axis=0), errorevery=markevery, markevery=markevery, marker=marker_cycle[len(Ks)], color=color_cycle[len(Ks)], capsize=3)\n",
    "# axes[1].plot(lamdas, gcv_cls_errs[:, sketch_i, :].T, color=color_cycle[len(Ks) + 1], alpha=0.5, zorder=10)\n",
    "axes[1].plot(lamdas, np.mean(gcv_cls_errs[:, sketch_i, :], axis=0), color=color_cycle[len(Ks) + 1])\n",
    "axes[1].errorbar(lamdas, np.mean(gcv_cls_errs[:, sketch_i, :], axis=0), yerr=np.std(gcv_cls_errs[:, sketch_i, :], axis=0), errorevery=markevery, markevery=markevery, marker=marker_cycle[len(Ks) + 1], color=color_cycle[len(Ks) + 1], capsize=3, label='GCV')\n",
    "\n",
    "axes[0].legend()\n",
    "axes[0].set_xlabel(r'$\\lambda$')\n",
    "axes[0].set_title('RNA-Seq Ridge Regression')\n",
    "axes[0].set_ylabel('MSE')\n",
    "axes[0].set_xscale('symlog', linthresh=10)\n",
    "axes[0].set_xlim(-30, 1.1e3)\n",
    "axes[0].set_ylim(0, 0.1)\n",
    "\n",
    "axes[1].set_title('RNA-Seq Ridge Classification')\n",
    "axes[1].set_xlabel(r'$\\lambda$')\n",
    "axes[1].set_ylabel('Classification Error')\n",
    "axes[1].set_xscale('symlog', linthresh=10)\n",
    "axes[1].set_xlim(-30, 1.1e3)\n",
    "axes[1].set_ylim(-0.02, 0.02)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Clasification Test Error\n",
      "K=2:      0 +-      0\n",
      "K=5:      0 +-      0\n",
      "GCV:      0 +-      0\n",
      "Regression Test Error\n",
      "K=2: 0.034819 +- 0.00490708\n",
      "K=5: 0.034819 +- 0.00490708\n",
      "GCV: 0.034819 +- 0.00490708\n"
     ]
    }
   ],
   "source": [
    "print('Clasification Test Error')\n",
    "\n",
    "cv_min_idx = np.argmin(cv_cls_errs, axis=3)[:, sketch_i]\n",
    "gcv_min_idx = np.argmin(gcv_cls_errs, axis=2)[:, sketch_i]\n",
    "\n",
    "for k, K in enumerate(Ks):\n",
    "\n",
    "    test_mean = np.mean([test_cls_errs[i, sketch_i, cv_min_idx[i, k]] for i in range(n_trials)])\n",
    "    test_std = np.std([test_cls_errs[i, sketch_i, cv_min_idx[i, k]] for i in range(n_trials)])\n",
    "\n",
    "    print(f'K={K}: {test_mean:6g} +- {test_std:6g}')\n",
    "\n",
    "test_mean = np.mean([test_cls_errs[i, sketch_i, gcv_min_idx[i]] for i in range(n_trials)])\n",
    "test_std = np.std([test_cls_errs[i, sketch_i, gcv_min_idx[i]] for i in range(n_trials)])\n",
    "\n",
    "print(f'GCV: {test_mean:6g} +- {test_std:6g}')\n",
    "\n",
    "print('Regression Test Error')\n",
    "\n",
    "cv_min_idx = np.argmin(cv_reg_errs, axis=3)[:, sketch_i]\n",
    "gcv_min_idx = np.argmin(gcv_reg_errs, axis=2)[:, sketch_i]\n",
    "\n",
    "for k, K in enumerate(Ks):\n",
    "\n",
    "    test_mean = np.mean([test_reg_errs[:, sketch_i, cv_min_idx[i, k]] for i in range(n_trials)])\n",
    "    test_std = np.std([test_reg_errs[:, sketch_i, cv_min_idx[i, k]] for i in range(n_trials)])\n",
    "\n",
    "    print(f'K={K}: {test_mean:6g} +- {test_std:6g}')\n",
    "\n",
    "test_mean = np.mean([test_reg_errs[:, sketch_i, gcv_min_idx[i]] for i in range(n_trials)])\n",
    "test_std = np.std([test_reg_errs[:, sketch_i, gcv_min_idx[i]] for i in range(n_trials)])\n",
    "\n",
    "print(f'GCV: {test_mean:6g} +- {test_std:6g}')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.03176873438820051 0.001223979893198133 0.7406361314290153\n"
     ]
    }
   ],
   "source": [
    "ens = PreSketchedEnsembleRidge(n_ens=n_ens, lamda=0.0001, which_ridge='svd')\n",
    "ens.fit(XS_train, y_train)\n",
    "print(mean_squared_error(y_test, ens.predict(XS_test)), mean_squared_error(y_train, ens.predict(XS_train)), ens.gcv_div())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(297, 401, 297, (401, 297))"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "q = XS_train.shape[1] // n_ens\n",
    "\n",
    "XS_this = XS_train[:, :q]\n",
    "beta = np.linalg.solve(XS_this.T @ XS_this, XS_this.T @ y_train)\n",
    "np.linalg.matrix_rank(XS_this), len(y_train), q, XS_this.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Unsketched test error: 0.13256741307150258\n"
     ]
    }
   ],
   "source": [
    "lamda = 1\n",
    "ridge = Ridge(n_train * lamda, fit_intercept=False).fit(X_train_norm, y_train)\n",
    "\n",
    "y_hat = X_test_norm @ ridge.coef_\n",
    "print(f'Unsketched test error: {np.mean((y_hat - y_test)**2)}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([ 1,  1, -1,  1,  1,  1,  1,  1,  1, -1, -1, -1,  1,  1,  1,  1,  1,\n",
       "       -1, -1,  1,  1,  1, -1,  1,  1,  1,  1,  1, -1, -1,  1, -1,  1, -1,\n",
       "        1, -1,  1,  1, -1, -1,  1, -1,  1,  1,  1])"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y_test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.9.17"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
