{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:526: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_qint8 = np.dtype([(\"qint8\", np.int8, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:527: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_quint8 = np.dtype([(\"quint8\", np.uint8, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:528: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_qint16 = np.dtype([(\"qint16\", np.int16, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:529: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_quint16 = np.dtype([(\"quint16\", np.uint16, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:530: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_qint32 = np.dtype([(\"qint32\", np.int32, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:535: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  np_resource = np.dtype([(\"resource\", np.ubyte, 1)])\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import argparse\n",
    "import pickle\n",
    "from tqdm import tqdm\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from tensorflow.python.keras.layers import Dense, Input, Flatten, Add, Multiply, Lambda\n",
    "from tensorflow.python.keras.layers.normalization import BatchNormalization\n",
    "from tensorflow.python.keras import regularizers\n",
    "from tensorflow.python.keras.models import Model, Sequential\n",
    "from tensorflow.python.keras import optimizers\n",
    "from tensorflow.python.keras.callbacks import ModelCheckpoint\n",
    "from scipy.spatial.distance import pdist\n",
    "\n",
    "from utils.explanations import calculate_stability, calculate_robust_astute_sampled"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'orange_skin'\n",
    "run_times = 1\n",
    "prop_points = 0.05\n",
    "calculate = True\n",
    "epsilon_range = np.arange(0.01, 1.1, 0.05)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_dict = pickle.load(open('data/' + datatype + '.pk', 'rb'))\n",
    "\n",
    "x_train, _, x_val, _, _, input_shape = data_dict['x_train'], data_dict['y_train'], \\\n",
    "                                       data_dict['x_val'], data_dict['y_val'], \\\n",
    "                                       data_dict['datatype_val'], data_dict['input_shape']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "median_rad = 0.5 * np.median(pdist(x_val))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "save_astuteness_file = 'plots/rise_' + datatype + '_astuteness_classifiers.pk'\n",
    "classifiers = ['2layer', '4layer', 'linear', 'svm']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 1 of 1\n",
      "WARNING:tensorflow:From /home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/ops/resource_variable_ops.py:435: colocate_with (from tensorflow.python.framework.ops) is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "Colocations handled automatically by placer.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2021-09-28 22:23:37.981715: I tensorflow/core/platform/cpu_feature_guard.cc:141] Your CPU supports instructions that this TensorFlow binary was not compiled to use: AVX2 FMA\n",
      "2021-09-28 22:23:38.005911: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-28 22:23:38.006750: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x55ecdf1838c0 executing computations on platform Host. Devices:\n",
      "2021-09-28 22:23:38.006774: I tensorflow/compiler/xla/service/service.cc:158]   StreamExecutor device (0): <undefined>, <undefined>\n",
      "100%|████████████████████████████████████████| 22/22 [34:11<00:00, 93.24s/it]\n",
      "100%|███████████████████████████████████████| 22/22 [41:55<00:00, 114.35s/it]\n",
      "100%|████████████████████████████████████████| 22/22 [22:49<00:00, 62.23s/it]\n",
      "100%|████████████████████████████████████████| 22/22 [24:09<00:00, 65.89s/it]\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXQAAAD4CAYAAAD8Zh1EAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8QVMy6AAAACXBIWXMAAAsTAAALEwEAmpwYAABIEElEQVR4nO3dd3xU1bbA8d9OI72HXoKhSA8QaRFpUlREEUGKXkEUUbFcGz69duwNFAvYUFSKikgTG00CSO86QwIhCaGk9zKT2e+Pk4QACZkk05LZ38+H90zm5Ow1l7Cys8/eawkpJYqiKEr952LvABRFURTLUAldURSlgVAJXVEUpYFQCV1RFKWBUAldURSlgXCz18ChoaEyPDzcXsMriqLUS3v27EmVUoZV9prdEnp4eDi7d++21/CKoij1khDiZFWvqSUXRVGUBkIldEVRlAZCJXRFUZQGQiV0RVGUBkIldEVRlAai2oQuhPhCCHFOCHG4iteFEOJ9IUSsEOKgEKKX5cNUFEVRqmPODH0RMOoyr18HtC/9MwP4uO5hKYqiKDVV7T50KeUWIUT4ZS65CfhaanV4dwghAoUQzaSUpy0VpDM6l3+OlbErKS4ptncoSj1UZDSRlJFPak4RNS2Q7SJLCM4/QZGbLzmNmtboa32zC+h8KIGk1qGcbhGEdLHuqq5PXi5X6vQ0PZ2KqEeVwL17RDHmkXkWv68lDha1ABIrfJxU+rlLEroQYgbaLJ7WrVtbYOiG6WDKQR7Z+AgpBSkIhL3DUeoRWfp/apvbBBJR9tXFQLH2kTTz+/CxlUb66bSvz/WEQ20EB9q6cKCtIDWg7t/LLiZJ+2RJjxMmusdL2ieDS2m4pjrf3XYOmXZa5b6WSOiV/S1V+v0kpVwILASIioqqRz9PbWd13Gpe2PYCYd5hrBizgvZB7e0dkuKgEtPz2XIshb/0qcTEpZJTaEQI6N4igIHtwxjYPpSerYPwcDNjlnx8M/z+HJzeD026wbBn4ewR2PoeFOdCz9th8NPg36zKWxQcOkz86+MJnjoVz25dydu2jcCtMfRffxYAj/BwfAYMwOfqaLz79MHV19es91l8MIa8tUvI/Xs3+XGZmAwChMSzqQe+o67E59ob8Bo2HuHpbdb9HEEXK93XEgk9CWhV4eOWQLIF7utUSkwlzN07l0VHFtGnaR/eGfQOgZ6B9g5LcSDZhQa2x6Wx9Vgqfx1LIT4tH4DmAZ7c0K0ZV7cPJToilCAfD/NvevYI/P48xP4OAa1g7ELoNh5cXKDDSOh1J/z1Nuz8FA5+D/0fgOiHwdP/klulzJuHa0AAobMewNXXl4AbbkBKSfHx4+TFxJAbE0PmTz+R8d134OaGV2QPfKOj8RkwAM+uXRGurgCUnEskb81i8v7aRN6RRAzZ2v3dfcG/Z0t8rhmEzw134NosvI7/izY8wpwWdKVr6GuklF0ree0GYBZwPdAXeF9K2ae6e0ZFRUlVy0WTXZzNk1ueJOZUDBM7TuTJPk/i7uJu77AUB7F0ZwI/7EliX2ImJSaJt4cr/a8IYWD7UAZ2COOKUB+EqOFyRlYSbHwV9n+nJeeBj0OfGeDuWfn16Sdgwxw4/AN4h8Cgp6D3VHDTfnjk7dxJwn/upPETTxAy/a4qhzUVF1Owdx95MTHkxcRQePQoAC5+PvhEhGA8d4aC00UgBS5uEu+IAHz6RuEzagIekQMRVl6Trw+EEHuklFGVvlZdQhdCLAEGA6HAWeB5wB1ASvmJ0L6T5qPthMkHpkkpq83UKqFrTmSd4KEND5GUk8TT/Z5mfIfx9g5JcSC/HDrNfd/u5cqmfgzr1JiB7cPoZe4ySmUKMiFmLuz4GKQJ+t4LVz8K3sHmff2pvdrSTPxfENQWrn0e2ekmTt5+B4akJCJ++xUXzyp+KFSUfhziNmA8+Bt5f+8iL9FE/jkPXH298OnVEd9hN+B17YR6tYxiK3VK6NaiEjr8lfQXs7fMxt3VnXcHv0vvJr3tHZLiQJIy8rl+3l+0DfXh+5kDap/EAYxFsOtz2PKmltS73wZDn4HAWmxOkBJi/9AS+7mj5BZ1IfGnDJq+8DxBEydW/jUFmdoPgbgN2p+MeO3zAa0gYqj2p+015v9gcWKXS+h2K5/rzKSUfHXkK97b+x4dgjowb8g8mvs2t3dYigMxlJh4aMk+pIQPJvWqfTI3meDICvjzJcg8CVcMgeEvQrMetQ9OCGg/HCKGIvd9x7mHXsXdx0igYQWci4TGV0KJEU7tgeMbtQSetBtkCXj4QvhA6PeAlsRDIrT7KRahErqNFZUU8cK2F1hzfA3D2wxnTvQcvN3Vr5XKheb+oWdvQiYfTOpJ65Bafn+kxcEPd53fuXL7Cmg3zHJBuriScy6UojRoPv1aRNIa+Lg/tB4AZw5BURYgoEUvGPiolsBbXgWu6vmQtaiEbkPn8s/xyMZHOJR6iAciH+De7vfW/GGW0uDFxKby0aY4botqxY09avmbm8kEK+/XljYq7lyxIGk0kjLvfRq1b4f/o/OhcI62IyZuI3S5qXQZZZBaRrEhldBtpOywUK4hl7mD5zKsjQVnSkqDkZpbxCPL9hMR5svzYzrX/kYHvoPEHXDTh9DjNssFWEHWzz9THB9Py/kfaFsOfUJg1GtWGUsxj9oDZAOr4lYxbf00PFw9+Ob6b1QyVyplMkkeW36ArAID8yf3xNujlvOt/HT47Vlo1Q96TLZskKVMxcWkzP8Qz+7d8R2mvp8dhZqhW9nCgwv5YN8HXNX0Kt4Z9A5BnkH2DklxUJ9tPc5mfQov39yVK5teenDHbH+8AIVZMPpdiy+zlMlcugzj6dM0f/UVtWzoQFRCtyIpJYsOL2Jgi4HMGzpPHRZSqrQ/MZM31+sY1aUpt/etQ52jxJ2w9yvoPwuaWOeAuSkvj9QFC/Du1w+f/v2tMoZSOyqhW1FyXjI5hhwGtxqskrlSpexCAw8t2UcTf0/eGNe99jPeEiOseRT8msPgpywbZAXpi7+hJC2Nxh/Ot9oYSu2ohG5FunQdAB2COtg5EsVRSSl5esUhTmUWsPzefgR41+EH/65P4ewhmPA1NPKzXJAVlGRlkfb55/gOHYpXZKRVxlBqTz0UtSJdhg6BUAldqdLy3YmsOXiaR4d3oHebOmzvyz4NG16BdtdCpzGWC/AiaZ9/gSk3l7CHH7LaGErtqYRuRfp0Pa38WqmDQ0qljp3N4flVR4huF8LMQRF1u9mvT0NJMVz/ltVOXhpTUkhfvBj/G27As2NHq4yh1I1K6Fakz9DTMVh94yuXKjSU8OCSffh4uPHehEhcXeqQhOM2aMf7Bz4GwVdYLsiLpC5YiCwuJuzBWVYbQ6kbldCtJN+QT2JOolpuUSo1Z+1R/j2TwzsTetDY34zqhFUxFMLaxyA4QqtTbiWGU6fIWLaMwHHj8GjTxmrjKHWjHopaiT5Dj0TSMUjN0JUL/XLoNN/sSGDGNVcwuGPjut0sZp5Wivb2FVXXMreAlA8/QghB6P33WW0Mpe7UDN1K9Bl6ALXkolwgKSOf2T8epEfLAB4fUcfvjfTj8Nc70GWsZYtuXaQoLo6slSsJmjwZ96Y1axqt2JZK6FaiS9fh5+FHM5+qezAqzsViJXFBq0m+7glw9YCR1q2fkvL+B7h4ehIy4x6rjqPUnUroVqLL0NEhqIM6Fq2UKyuJ+8ot3WpfErfMP6u0JhNDLt+4ua4KDh8h59dfCZ46FbdgVTXR0amEbgUmadJ2uKj1c6XUrvj08pK4Y2pbErdMUQ788pRW47zPDMsEWIWyxs/B06ZadRzFMlRCt4KknCQKjAVq/Vwpt2DzcUJ8POpWErfMptchJ1krvuVqvX0N+bt2kffXX4TMmIGrn3VOniqWpRK6Fegy1JF/5bzkzAI2/HuWCVGtal8St8zZI1qD5153Qqs+lgmwElJKzs2dh1vjxgRNsU4JXsXyVEK3Al26DhfhQrvAdvYORXEAS3cmIIFJfepQRRG0LkRrHgXPALj2BUuEVqW8v/6iYM8eQu+/DxdP622HVCxL7UO3Al2Gjjb+bfB0U/8QHFLiLtj0Goz/UkuOVmQoMbF0VyKDO4TRKriOD0LLuhCNmW+Vtm7SaKTg4CHyYmLIXLEC91atCLzlFouPo1iPSuhWoE/X0z2su73DUKry54sQ/xds/1DbJWLNof45y7mcIl7tW8fTlRW7EEVOsUxwQHFCAnkxMeTGxJC/429Mubng4oJnt640efxxhIeHxcZSrE8ldAvLLs4mOS+Z8R3H2zsUpTLJ+7Rk7hmgJfQ+M8An1GrDfbMjgeYBngy5so4nQsu6EN3wTp26EJVkZ5O3Ywd527aRF7MNQ2IiAO7Nm+N//fX4REfj068vrgHW/c1FsQ6V0C3sWMYxQD0QdVgx70Mjf+2o/OfDYet7MPIVqwx1IjWPrbGpPDq8Q92KbyXuOt+FqGnXGn1pxWWUvJgYCg4eBJMJFx8fvPv1I3jqnfhGR+Pepo06M9EAqIRuYWVNLdQedAeUfgKOroQBD0LLKOgxCXZ+Cv3uh4AWFh9uyc4EXF0EE69qVfubHN8Eqx+pVRei/D17SLzvfkzZ2eXLKKEz78UnOhqv7t0R7qqLVkOjErqF6TP0BDYKpLF3HX/FVixvx0cgXKFvaYGpQbPh4HLY8ibcOM+iQxUaSvh+dyIjOjepXTXFM4fhj+e106ABrWDcpzXuQpS+6CuEuzst5s1TyyhOQm1btDBduo6OQR3Vr6+OJi8N9i6G7hPOH5UPagNRd2mfT4uz6HC/HD5NRr6BKTV9GJqVBCvvh0+uhqRdMGIOzNoN4VfX6DamvDxyt2zBf9Qo/EeOUMncSaiEbkElphJiM2PpEKzWzx3Ors/AWKAtt1Q08DFwawQbX7XocN/uSKBtqA8DIkLM+4KCTPj9efigNxz6HgbMgof2a/HWoixu7ubNyKIi/K8bVeOvVeovldAt6GTOSQpLCtX6uaMxFMDOhdB+JDTudOFrfk2g70w4/AOcOWSR4f49k83ukxlM7tMal+oehhqLtN0270dqtc073wwP7tFm5nXYa579y3rcwsLw6tWr1vdQ6h+V0C1In65qoDuk/d9BfipEV9HYOPohaBSgNVm2gO/+TsDDzYVxvVtWfZHJBId+gPlRWj/Q5j3h3i1wywIIrNuJ0rLlFr8RIxB12OKo1D9m/W0LIUYJIXRCiFghxCWP2oUQAUKI1UKIA0KII0KIaZYP1fHpMnS4CTeuCLBeX0elhkwlsO0DaN4L2kRXfo1XkJbU9b9A4s46DZdXZGTF3lPc0K0ZwT5VHMo5vhk+HQI/Ttf2w9/xk/anmWUOo6nlFudVbUIXQrgCHwLXAZ2BSUKIi0vGPQAclVL2AAYD7wghnO6ImS5dR3hAOB6uTvfWHde/ayDjhNZv83IPqvvOBJ8w+PMlrXlELa06kExukZHb+1Uyyz57BL65Fb4eA/lpMHYhzNgCEUNrPV5l1HKL8zJnht4HiJVSHpdSFgNLgZsuukYCfkLb2uELpANGi0ZaD+gydGq5xZFIqa1LB7WFTjde/tpGvjDwce0U6fFNtRxO8s2Ok1zZ1I9erYMufHHnp/BxNCTtPL9zpcdtdTr1WRm13OLczPkbbwEkVvg4qfRzFc0HOgHJwCHgYSml6eIbCSFmCCF2CyF2p6Sk1DJkx5RZmMm5/HPqgagjObkNTu2B/g+Ai2v110dNA/+WtZ6lH0jK4khyNlP6tr5w2+rB72Hd49DxujrtXDGHWm5xbuYk9Mp+T734u30ksB9oDkQC84UQ/pd8kZQLpZRRUsqosLCwGobq2MpqoKuE7kC2vQ/eIeYXs3JrpJ3GTN4L/66t8XDf7jiJt4crN/esMN859gesnAnhA+HWL61SJbEitdzi3MxJ6ElAxbPLLdFm4hVNA1ZITSxwArjSMiHWD/oMbYeL2oPuIM79C/r1WvEtjxqUre0xCULawYY52gNVM2XlG1h9MJmbIlvg51l6pD5xJyy/Axp3honfWW1WXkYttyjm/K3vAtoLIdqWPuicCKy66JoEYBiAEKIJ0BE4bslAHZ0uXUeIZwihXtar3KfUwLYPwM0Lrqphp3pXNxjyDKT8o20rNNOKfUkUGkxM6Vv6MPTcP/DtePBrCrf/CJ6X/MJqceXLLaNGWn0sxTFVm9CllEZgFvAr8A+wXEp5RAgxUwgxs/Syl4EBQohDwJ/AbCllqrWCdkT6DL16IOoosk/DwWXQcwr4mHlSs6LON0PTbrDpVTAWV3u5lJJv/06gR6tAurYIgMwEWHwLuHlq2xF9bVPXRy23KGb9XialXCel7CCljJBSvlL6uU+klJ+U/neylHKElLKblLKrlPIbawbtaAwmA7GZsWr93FH8/QnIEu1haG24uMDQ5yAjHvYtrn64E+nEnsvVZud5qbB4LBjy4I4VEBReuxhq6ILlFlczHgArDZJaaLOA+Kx4DCaDWj93BIXZsPtL6DQGgutwwKv9cGjdHza/qZUOuIxv/07A39ONGzv6wbe3agW2Ji2DJl1qP34NqeUWBVRCtwi1w8WB7P0KirKqPuZvLiFg2HOQe0bbQ16FlJwi1h8+zYSejfFa8R84fRAmfA1t+tdt/BpSyy0KqIRuEfp0Pe4u7oQHhNs7FOdWYoAdH2tbBFv0rvv92gyAdtfC1ne1mX8lvt+TSElJCY9kvwUnNsNNH0IH286S1XKLUkYldAvQZeiICIzA3UV1gLGrwz9C9ikYUMfZeUVD/wcFGVpFxIuYTJLvdpxkQfB3+MathRGvQOQky41tJrXcopRRCd0CdOk61UPU3qTU+oWGddLWvy2leU/ofBNsn6898Kxgy7EUxucuZnj+Orj6v1oNczvI/mU9rmGharlFUQm9rlILUkkrTFPr5/YW+yecO6KtnVu6W9SQZ8CQrzWUruD0r3N52O0nSiLvgGHPW3ZMM5V3JhoxUi23KCqh15Wqge4gts3TGil3vdXy9w7reL6hdNYpADJ2fMuk9A/RBw3C9ca5lv8hYia13KJUpBJ6HakdLg4geR+c2AL9ZoKblUoXD5oN0qQ1lD72B/6/PsgOUye8Ji3STpfaiVpuUSpSCb2OdBk6Gns3JtAz0N6hOK+Y98HDD3pPtd4YQW20aox7FyOX30GsbM1XbV6jVWPrFtu6HLXcolxMJfQ60mfo1ezcnjLi4ehKLdl6Wrmz/cDHwa0R+Y3CmFL4JLf0v7jPi22p5RblYiqh10FxSTEnMk+o9XN72v4hCFfod5/1x/JrAjM287j/O7gHNGFIR/uWgFbLLcrFVEKvg+NZxzFKo5qh20t+Ouz7BrqNB//mNhkyXrTgl+PFTOrTGjdX+/3zUcstSmVUQq8DXbr2QFTVcLGTvxdo2wkHPGizIb/bmYCri+C2q1pVf7EVqeUWpTL2ezzfAOgydHi6etLGr429Q3EOBZnabpbjGyFug7Z+3mEUNLHNWvbag6f5ens8Izo3oYm/dZtVVEcttyiVUQm9DvTpetoFtsPVnH6VSs2VGLWeoHEbtD+ndmtbBz18oe010O8B6D7B6mGYTJK5f+h5f0MsvVoH8tJNXa0+5mXjKV1uCRw3Ti23KBdQCb2WpJToMnQMbT3U3qE0LOnHSxP4Rm02XpQNwkU7gj/wMYgYCi2vAlfb1M3JLTLy6LL9/Hb0LBOiWvLyzV1p5GbfJKoaQStVUQm9ls7lnyOzKFPVcKmrEoPW+zP2T20pJSNe+3xAa+gyVkvgba+xenPlyiSk5XPP17uJTcnl+Rs7M3VAOMJOJ0IrUsstSlVUQq8ldULUAvLS4Ps7If6v88so/WdpSTz4CrsdpwfYFpfK/d/uRUr4alofrm7vGL1i1XKLcjkqodeSPkOr4aJ2uNTSmcOwdBLknIUx86HHRJsto1yOlJLFO07y4uqjtA314bP/RBEe6mPvsMqp5RblclRCryVduo7mPs3x97B+N/cG5+gq+GkmePrDtF+gpQWaUVhAsdHE86uOsGRnAsOubMzciZH4edr/h0xFarlFuRyV0GtJl6FTs/OaMplgy1uw6VWto9Bt34J/M3tHBUBqbhH3f7OXnfHp3D84gsdGdMTVxf7r5RWp5RalOiqh10KhsZCT2ScZ0WaEvUOpP4rztFn5P6u0UrSj54K7ffdylzmSnMWMr/eQmlvEvImR3BTZwt4hVapsucVPHSZSqqASei3EZcZhkiZVw8VcmQmwZLLWgGLEK9D/Abs+8Kxo7cHTPP79AQK83Plh5gC6tbRyga86KFtu8e7tGEtUiuNRCb0W1A6XGoiPgeV3aIeEJn8P7a+1d0TApYeFPrmjN439HOM3hsqo5RbFHCqh14IuXYe3mzct/VraOxTHtvsLWPcEBLWFSUsgtL29IwKgoLiEh5fu47ejZxnfuyVzxtr/sFB11HKLYg6V0GtBl6GjfVB7XISqbVapEgP8Mht2fw7troVxn4NXoL2jKvft3yf57ehZnh3dmbuiHeOwUHXUcotiDpWRakhKiT5dNbWoUl4aLB6rJfMBD8Hk5Q6VzAE26s7RsYkf069uWy+SeXmp3OEj1HKLcllqhl5Dp/NOk2PIUUf+K3P2CCyZqB0WGrsQetxm74gukVdkZNeJDKZFh9s7FLOp5RbFXCqh11BZDXS1w+Ui/66DH+92uMNCF9sel0ZxiYlBHezbbagm1HKLYi615FJDZTtc2gc5xgM+h5B+An6YBmEd4J6NDpvMATbrU/D2cKV3eJC9QzFLUVwcOZs24X/ddWq5RamWWQldCDFKCKETQsQKIZ6q4prBQoj9QogjQojNlg3Tcegz9LTya4WPu+PU97ArKeGXJ8HFDSZ+5zAnPysjpWST/hwDIkIcflcLaPGeefElXLy9CZ05097hKPVAtQldCOEKfAhcB3QGJgkhOl90TSDwETBGStkFGG/5UB2DLl2nHohW9O8aOPYbDHnaZn09ays+LZ/E9IJ6s9ySvWYN+Tt30vi//8Ut2Pblg5X6x5wZeh8gVkp5XEpZDCwFbrromsnACillAoCU8pxlw3QM+YZ8EnMSVQ2XMkW58MtT0KQr9LnX3tFUa7NO+7Yc1KGxnSOpXkl2NmdffwPP7t0JnNBg50eKhZmT0FsAiRU+Tir9XEUdgCAhxCYhxB4hxH8qu5EQYoYQYrcQYndKSkrtIrYjfYYeiVQz9DKb34DsJLjhXXB1/Ofrm/QpXBHqQ+sQb3uHUq2UufMoycig6fPPIVzUoy7FPOZ8p1S2UVde9LEb0Bu4ARgJPCuEuGQaK6VcKKWMklJGhYXVj197Kyqrga52uABnj8KOj6DnHdC6r72jqVahoYQdx9O4ph4stxQcPkLGkiUETZ6MV5cu9g5HqUfMmVYlAa0qfNwSSK7kmlQpZR6QJ4TYAvQA9BaJ0kHoM/T4ufvR3Mex14qtTkpY+xg08ofhL9k7GrPsPJFOocHEoI6OndBlSQlnXngB19AQwh5+yN7hKPWMOTP0XUB7IURbIYQHMBFYddE1PwMDhRBuQghvoC/wj2VDtT9dunbkvz6cLrSqA0sgYRsMf9EuvT5rY7M+BQ83F/q1DbF3KJeVuXw5hYcP02T2U7j6+dk7HKWeqTahSymNwCzgV7QkvVxKeUQIMVMIMbP0mn+A9cBBYCfwmZTysPXCtj2TNKHP0Kvllvx0+O1ZaNkHIm+3dzRm26Q7R9+2wXh5OO52RWNqKufefQ/v/v3wv+F6e4ej1ENmPcmSUq4D1l30uU8u+vgt4C3LheZYTuWcIt+Yrx6I/vkSFGTA6PegnjysS0zPJy4lj8l929g7lMs699ZbmAoLafrsc+q3QKVW6se/SAdQXgPdmWfoSbthzyLoOxOadrV3NGbbckzbUeXI+8/z/t5J1s+rCJl+F42uaGvvcJR6SiV0M+kydLgIFyICI+wdin2UGGHNf8GvKQz5P3tHUyObdSm0CPQiIswxT/fK4mLOvPQS7i1bqhOhSp04/uZhB6FL19HarzVebl72DsU+dn8OZw7C+EXQqP48rCs2mtgWl8aYyOYOu4yRtugriuPiaPnxR7h4Om7XJMXxqRm6mZz6gWjOGdgwByKGQeeb7R1Njew5mUFukdFhl1sMp06R+tFH+F47DL8hQ+wdjlLPqYRuhpziHE7lnnLeB6K/PgPGIrj+LYdp7myuzfoU3FwE0e1C7R1Kpc68+hoIQdOnn7Z3KEoDoBK6GZz6hOjxTXD4B7j6vxBS/54fbNanEBUehG8jx1tdzNmwkdw//yTsgftxb+7kh9UUi1AJ3QxlTS2crkuRsUg7ERrUVkvo9czZ7EL+OZ3tkMW4TAUFnJ0zB492EQTfeae9w1EaCMebtjggfYaegEYBNPFuYu9QbGvb+5AWC7f/CO7172HdFr3jbldM/fgTDMnJtFn8NcLd3d7hKA2EmqGbQZ+hNYV21F0SVpF+Ara8rT0EbXetvaOplU36FBr7NaJTM8falVMUF0fal18ScNNNeF91lb3DURoQldCrUWIq4VjGMedabqnYhWjUa/aOplaMJSa2HktlUIcwh/pBXN6FyMuLxk8+Ye9wlAZGJfRqJOQkUFhS6FwPRP9dW2+6EFXlQFIWWQUGh6uuWN6F6NH/4hbi2IXClPpHJfRqlB/5d5Yti0W58MvsetOFqCqb9Sm4CLjagbYrXtCFaLzqQqRYnnooWg19uh434eY8R/63vKl1Ibr183rRhagqm/UpRLYKJNDbw96hlCvrQtRqwQKEq+NWfVTqLzVDr4YuQ0d4QDgero6TGKzm3L+w/cPSLkT97B1NraXlFnEwKdOhtisWHCntQjRpEl5dVRcixTpUQq+GLl3nPA9E/3wJ3H3g2hftHUmdbI1NRUoY7EDr52mffIKLvz9hjzxs71CUBkwl9MvIKsribP5Z53ggmrQbdGsh+kHwqd8P6zbrUgj28aBbiwB7hwJA0YkT5PzxJ0GTJqouRIpVqYR+GWUnRJ3igeifL4F3KPS9z96R1InJJNlyLIWB7UNxcXGM7YrpXy5CuLsTfHv96fCk1E8qoV+GXZtapMbCuiegMNv6Yx3fBCc2wzWPQyNf649nRUdPZ5OaW+wwp0ONqalkrVxJwM034xbqODtulIap/m5jsAFduo5gz2BCvWz8D9Fkgp/vh8S/wcUdRr1qvbGk1Gbn/i0h6i7rjWMjm3TnABjY3jESevq33yINBoKnTbV3KIoTUDP0yyg78m9z+7/RknnYlfD3J3DmkPXG0q2DU3tg8FPg1sh649jIZn0K3VoEEOZn//diyssj47sl+F07jEZtVVs5xfpUQq+CwWQgNjPW9ssteWnw+3PQegBM+wW8AmHNo9qs3dJMJVrjipB20GOS5e9vY1kFBvYmZDrMckvmjyswZWURfFf9/81HqR9UQq/CyayTGEwG229Z/PMFKMqBG94B72AY/jIk7dRm7ZZ2+Ec4dxSGPFOvDxGV2RabSolJOsRxf2k0kr5oEV69euHds6e9w1GchEroVbDLA9GEv2Hv19DvfmjSWftc5GRttv77c9rs3VJKDLDxFWjard61lavKZn0Kfp5u9GwVaO9QyF7/K4bkZELunm7vUBQnohJ6FXQZOtxd3GkbYKO1zxIjrH1Uezg5aPb5zwuhzdaLcuCP5y033r7FkBEPQ58Dl/r/bSClZJMuhavbheLmat/3I6Uk7YvP8bjiCnwHD7ZrLIpzqf//kq1En64nIjACdxcbNR/YuQDOHobrXr9062CTztqsfd9iSNhR97EMBbD5TWjVD9oPr/v9HID+bC5nsgsd4nRo/o4dFB39h+BpUxEN4IelUn+o77Yq6DJseOQ/Oxk2vgrtR8CVoyu/ZtBsbfa+5lFtNl8Xuz6DnNMw7Ll61/S5Kpv12nbFaxzggWjaZ5/jGhZKwJgx9g5FcTIqoVcirSCN1IJU221ZXP9/YDLCdW9WnWAb+Wqz93NHtNl8bRVmw1/vQsQwCI+u/X0czGZ9Ch2b+NEswMuucRT++y95MTEE334HLo3sv3VScS4qoVfCpg9EY/+Aoyth4OMQXM16/ZWjtVn8xlch61TtxtvxERSkw7Bna/f1DiivyMiuExkOsbsl7YsvEN7eBE28zd6hKE5IJfRK6NP1ANZfcjEUasf7Q9pB9EPVXy+ENos3GeHXp2s+Xl4abJsPncZA84azlW57XBrFJSa77z83JCeTvXYdQePH4xrgGIXBFOeiEnoldBk6Gns1JsgzyLoDxcyF9ONw/dvmn9IMbqvN5o+u1Gb3NRrvPTDkafvOG5DN+hS8PVyJCrfy31c10r/6GoDgO/9j1zgU56USeiV0GTo6BFt5dp4Wp61ldx0HEUNq9rXRD2mz+rWPa7N8c2Qnw85PoftEaHxlzeN1UFJKNunPMSAihEZu9usCVJKVReb33+N//fW4N6+ffViV+s+shC6EGCWE0AkhYoUQT13muquEECVCiFstF6JtFZcUcyLzhHUfiEqpLbW4esCIV2r+9W6NtFl9xgnY+p55X7P5Te2o/+DZ1V9bj8Sn5ZOYXmD35ZaMpcsw5ecTMl0d81fsp9qELoRwBT4ErgM6A5OEEJ2ruO4N4FdLB2lLx7OOY5RG6z4QPfozxP0JQ/8H/s1qd4+IIdrsfut72mz/ctLitD3svadCUHjtxnNQm0urK9qz3ZypuJj0bxbjEx2N55UN57cfpf4xZ4beB4iVUh6XUhYDS4GbKrnuQeBH4JwF47M5qze1KMrRtik27QZX3V23e418VZvlr3tCm/VXZdPrWhneax6v23gOaJM+hbahPrQO8bZbDNmrVlGSkqpm54rdmZPQWwCJFT5OKv1cOSFEC2As8MnlbiSEmCGE2C2E2J2SklLTWG1Cn6GnkWsjWvu3ts4Am17XDvWMnlv3glh+TbVZftyf2kPSypw9Aoe+h773atc3IIWGEnYcT7Prcos0mUj7/Asade6Ed//+dotDUcC8hF7ZSZeLp4NzgdlSypLL3UhKuVBKGSWljAoLs/+e4croMnS0C2yHm4sVqg+eOQw7Pobed0LLKMvc86q7oWl3bdZflHPp6xtegUb+EN3wmhPvPJFOocFk1/3nuZs2UXziBCF3TUc0kFO3Sv1lTkJPAlpV+LglkHzRNVHAUiFEPHAr8JEQ4mZLBGhLUkr06XrrrJ+bTFrxLa9AGGbBIluubjD6Pcg5Axtfu/C1io2fvYMtN6aD2KxPwcPNhX5t7dfUOu3zL3Bv3hz/USPtFoOilDEnoe8C2gsh2gohPICJwKqKF0gp20opw6WU4cAPwP1SypWWDtbaUgpSyCjKsM6Bov3fal2Ihr9s+eTaMkp74Hlxd6MG0vi5MlJKNv57jr5tg/HysM92xfx9+yjYs4fgqVMRbvW/nrxS/1Wb0KWURmAW2u6Vf4DlUsojQoiZQoiZ1g7Qlqz2QDQ/vbQLUX/rdQYa9hx4BcHax7TfBhpQ4+fKHD6VzfHUPEZ1td9zgfQvvsAlIIDAcbfYLQZFqcisaYWUch2w7qLPVfoAVEo5te5h2UdZDReLHyr643kozIIb3rVe7XHvYBjxMqy8T+tutOcrrTpj72nWGc/OftybhIerC6O72ecQT9GJE+T88Sch987AxcfHLjEoysXU74kV6NP1NPdpjr+Hv+VuWtaFaMCD57sQWUuPSbB3sTZLLymGMR+Au6d1x7QDQ4mJ1QeSubZzYwK8bVSv/iLpXy5CuLsTfPvtdhlfUSqjjv5XYPEj/yYTrHsc/FvAoCoP2FpOWXcjaYLgCOgx2fpj2sEWfQppecXc0rOlXcY3pqaStXIlATffjFtoqF1iUJTKqBl6qUJjIfHZ8QxvY8EOPkd/gjMH4ZZPbbeO3aQzTPlB+yHSABo/V2bF3lME+3jYbbti+rffIg0GgqdNtcv4ilKVhvkvvhbiMuMwSZPltiyWGLU94I27QFcbl7apabGveiQr38Dv/5xlcp/WuNuhd6gpL4+M75bgd+0wGrW1Ub9ZRTGTSuilyh+IWmrL4oHvID0OJi5pEE2YHcXaQ6cpNpoY18s+yy2ZP67AlJVF8F3qmL/ieFSmKaVL1+Hl5kUrv1bVX1wdQyFsegNaREHH6+p+P6Xcir1JtGvsS9cWFnxwbaaS7GzSvvgCr1698O7ZcBqEKA2HSuildBk62ge1x0VY4H+SPV9CdlKDasLsCE6m5bH7ZAa39Gph82P2UkpO/+9ZjKmpNJn9pE3HVhRzqYRO6ZH/DL1lDhQV5cCWt6HtILhiUN3vp5RbsfcUQsDNkS2qv9jCMpctJ+e33wh7+CG8evSw+fiKYg6V0IEzeWfIKc6xTELf8Qnkp2qzc8VipJT8tO8UAyJCaB7oZdOxC/V6zr72Gj4DBhAyfbpNx1aUmlAJnfMPROu8wyU/Hba9Dx1vsFw1RQWAPSczSEjPt/nec1NBAacefRQXPz+av/kGQj3gVhyY2uXC+Rou7YPa1+1GMfO0JZehDasJsyP4ce8pvNxdbV675eyrr1EcG0erzz9Th4gUh6emG2gz9FZ+rfBxr0NNjpwz8PcC6DYemnSxXHAKhYYS1hxMZlTXpvg0st0cJHvdOjK//56Qe+7GNzraZuMqSm2phA6WeSC65W0wGWCwDY74O5k//zlHTqGRW3rZ7mFocVISp597Hs8e3Ql76CGbjasodeH0CT3fkE9CdkLdarhkxMOeRdDzDgiJsFRoSqkVe5No4t+IARG2WfKQBgOnHnsMhKDFO+8g3O1TAExRasrpE/qxzGNIZN1m6JveAOECg9T+ZEtLzS1isz6Fm3u2wNXFNnvPU95/n8IDB2n28kt4tLTPiVRFqQ2nfyha3tSitjtczv0LB5dCv/vB3z61uRuy1QeSMZqkzXa35G6NIe3TzwicMAH/UaNsMqZyIYPBQFJSEoWFhfYOxa48PT1p2bIl7jX4DdHpE7o+Q4+fux/NfWqZjDe+Au4+cPWjlg1MAbTDRF1b+NOxqZ/VxzKmpJA8ezaN2rejyf+pZyH2kpSUhJ+fH+Hh4U7beFtKSVpaGklJSbStQRE4p19y0aVrR/5r9Y1zai/8swoGzAIf+zUqbqj0Z3M4dCrLJrNzaTKRPPspTHl5tHj3XVy8bHt4STmvsLCQkJAQp03mAEIIQkJCavxbilMndJM0aTtcarvcsmEOeAVryy2Kxa3YewpXF8GYSOsvZaV9/jl527bR5On/o1H7Op5HUOqspsn8tgXbuW3BditFYx+1+YHm1An9VM4p8o35tSuZG78V4v6EgY+Cp+0r/zV0JSbJyn2nGNQhjFDfRlYdq2D/flLmzsNv1CgCx4+36liKYk1OndDLj/zXdIeLlPDny+DXDK662wqRKdvj0jiTXWj1vecl2dmceuxx3Js2pdlLLzr1r/mKJjExkSFDhtCpUye6dOnCvHnzAJg6dSo//PCDnaO7PKd+KKrP0OMiXGgX1K5mX3jsd0jcAaPfA3e11moNK/Yl4efpxrWdmlhtDCklp597HsPZs4R/sxhXf/WblgJubm6888479OrVi5ycHHr37s3w4RZsTVkFo9GIm1vdUrJTJ3Rduo7Wfq3xcqtBUjaZYMNLEBSuHSRSLC6vyMj6w2e4KbI5nu6uVhsnc/n35KxfT+PHH8MrMtJq4yi19+LqIxxNzq72uqOntWvMWUfv3Nyf52+sujxHs2bNaNasGQB+fn506tSJU6dOXXDNSy+9xOrVqykoKGDAgAEsWLCA48ePM378ePbu3QvAsWPHmDhxInv27GHPnj08+uij5ObmEhoayqJFi2jWrBmDBw9mwIABxMTEMGbMGB577LFq478cp19yqfED0aMr4cwhGPIMuKoThNbw65Ez5BeXcIsV28wVHTvG2VdfxSc6WrWTU6oUHx/Pvn376Nu37wWfnzVrFrt27eLw4cMUFBSwZs0aIiIiCAgIYP/+/QB8+eWXTJ06FYPBwIMPPsgPP/zAnj17uOuuu3jmmfMF/DIzM9m8eXOdkzk48Qw9tziXU7mnGNd+nPlfVGLU9p037gxda/B1So2s2HuKVsFeRLUJssr9LyiJ+8brqiSuA7vcTLqispn5snv7W2zs3Nxcxo0bx9y5c/G/aDlu48aNvPnmm+Tn55Oenk6XLl248cYbufvuu/nyyy959913WbZsGTt37kSn03H48OHyZZuSkpLy3wAAbrvtNovF7LQJXZ+hB2p4QvTAEkiLhYnfgYv1lgKc2emsAmLiUnlwaC3PBpjh7GuvU3QsVpXEVapkMBgYN24cU6ZM4ZZbbrngtcLCQu6//352795Nq1ateOGFF8r3i48bN44XX3yRoUOH0rt3b0JCQkhOTqZLly5s3175cpCPTx2qvF7EaacmZTtczN6yaCyCTa9Di97Q8XorRubcVu5LRkq4pad1drdkr19P5vLlhNxzjyqJq1RKSsn06dPp1KkTjz566QnwsuQdGhpKbm7uBTtfPD09GTlyJPfddx/Tpk0DoGPHjqSkpJQndIPBwJEjR6wSu/Mm9HQdAY0CaOJt5i6K3arxs7VJKVmxN4nebYIID7XcrKVMcVISp599Dq8ePQh76EGL319pGGJiYli8eDEbNmwgMjKSyMhI1q1bV/56YGAg99xzD926dePmm2/mqquuuuDrp0yZghCCESNGAODh4cEPP/zA7Nmz6dGjB5GRkWzbts0qsTv1kkvHoI7m/VpflAtb3oK218AVg60em7M6kpzNsXO5vDK2q8XvXV4SF2j+ztuqJG4DY8m186uvvhop5SWfv/7687+Zz5kzhzlz5lT69Vu3buWuu+7C1fX8smxkZCRbtmy55NpNmzbVPeAKnDKhl5hKOJZxjFs73GreF/z9sdb4eahq/GxNP+5NwsPVhdHdLH/Uv6wkbou576mSuIrVjB07lri4ODZs2GCX8c1achFCjBJC6IQQsUKIS8rQCSGmCCEOlv7ZJoToYflQLSchJ4HCkkLzHojmp0PMB9q6eaurqr9eqRVDiYlV+5O5tnNjArwtO3tWJXEVW/npp584ePAgoXZ62F5tQhdCuAIfAtcBnYFJQojOF112AhgkpewOvAwstHSgllSjI//b3oeibBj6PytH5dy26FNIyyu2eGVFVRJXcSbmLLn0AWKllMcBhBBLgZuAo2UXSCkrrvDvABz6d1p9uh434UZEYDXt4nLOwo5PVONnG1ix9xTBPh4M6hhmsXteUBJ30ZeqJK7S4Jmz5NICSKzwcVLp56oyHfilsheEEDOEELuFELtTUlLMj9LCdBk6wgPC8XD1uPyFf6nGz7aQlW/g93/OMqZHc9xdLbfxqrwk7v+pkrgN3pc3aH+cnDn/eirbBnLpI2BACDEELaHPrux1KeVCKWWUlDIqLMxyM7Ga0mfoq99/nnFS26qoGj9b3dpDpyk2mixaWbFg/35S5r2vlcSdoEriKs7BnISeBLSq8HFLIPnii4QQ3YHPgJuklGmWCc/ysoqyOJN3pvoHoptV42db+WlfEu0a+9KtRYBF7ldeErdJE1USV6m1kpISevbsyejRo4H6UT7XnIS+C2gvhGgrhPAAJgKrKl4ghGgNrADukFLqLR+m5ZQf+b/cA9EUnXbMv889qvGzlcWey2FXfAa39GphkcRbXhL3zBlavPO2Komr1Nq8efPo1KmTzcYzGo11vke1D0WllEYhxCzgV8AV+EJKeUQIMbP09U+A54AQ4KPSf5RGKWVUnaOzAl166Q6Xy83QN74C7t6q8bMV5RYZWbjlOJ9uOY6XuytjLXTUv6wkbthjj6qSuA3BL09p1U2rc+ag9v/NWUdv2g2ue/2ylyQlJbF27VqeeeYZ3n333Uted9TyuWYdLJJSrgPWXfS5Tyr8991AvWjdo8vQEewZTKhXFftEk/fB0Z9h0FOq8bMVGEpMLN2ZwLw/j5GaW8wN3ZvxxIiONAuo+w6U8pK4AwYQMn26BaJVnNUjjzzCm2++SU5OTqWvz5o1i+ee0w4a3nHHHaxZs4Ybb7yxvHxuZGTkJeVzf/75Z8LCwli2bBnPPPMMX3zxBXC+fK4lON1JUV267vLLLRvmgFcQ9H/AdkE5ASkl6w+f4c1fdZxIzaNv22A+u7MTka0CLXL/8pK4vr6qJG5DUs1MulzZzHza2joPuWbNGho3bkzv3r2rPJqvyuc6AKPJSFxmHJM7Ta78gvgYiP0Dhr+sGj9b0K74dF5d9w/7EjJp39iXz++MYuiVjS36sLK8JO5nn+Fmxx1USv0XExPDqlWrWLduHYWFhWRnZ3P77beXt4dT5XMdRHxWPMWm4sq3LEoJf76kNX7uc4/tg2uAYs/lcM/Xuxn/yXaSMwt4Y1w3fnl4IMM6NbFoMi8viXv3dHyvViVxlbp57bXXSEpKIj4+nqVLlzJ06FC++eab8tcduXyuU83Qy4/8V/ZAVDV+tphz2YW898cxlu1KwNvDjSdGduSu6LZ4eVi+KUhZSVzPHt0Je/hhi99fUS5WsXxueHh4peVzV6xYcUn53IceeoisrCyMRiOPPPIIXbpY/vS5qKxMpC1ERUXJ3bt323TMd/e8yzdHv+HvKX/j7lKhAJTJBAuvgaIcmLVb9QqtpYo7VwwlJm7v14YHh7YjxLeRVcaTBgPxt99Ocdxx2q78SVVRbCD++ecfm24XtLS3336brKwsXn755Trfq7L/LYQQe6raRehUM3R9up6IwIgLkzmcb/x8y6cqmdeClJLluxN561fdBTtXrNGkoqKUefO0krjvvauSueIQ7F0+12kS+pm8M+w6s4ux7cde+IJq/FwnRcYSnlt5hGW7E7kqPIhP/xNFz9bWae5cUcbSZaR99rlWEve666w+nqKY46effrLr+E6T0BceXIgJE9O6TrvwBdX4udZScoq475s97D6ZwYND2/Hfazvg4mL9Y/bZ63/lzIsv4jPoGpo+q8oaK0oZp0joCdkJ/HTsJyZ0nEAL3wonElXj51o7fCqLGV/vJj2/mPmTezK6u21KJORt307yE0/gFRlJy7lzVSs5RanAKbYtfnTgI9xd3bmn+0XbEVXj5wsYzpwhdeGnGKspbbz6QDK3fqKVwP9h5gCbJfOCQ4dJemAWHuHhtPrkY1XfXCk3bf00pq2fVv2FDVyDT+j6DD3rjq9jSqcpFx73L8rV6p2rxs8AGDMySJh2FynvvkvsyFGkfDCfkty8C64xmSRv/6rjwSX76No8gJ9nXU1XC1VIrE7R8RMkzpiBa2AgrT77FNcA24yrKPVJg0/o8/fNx9fdl6ldpl74wt+fQF4KDHveLnE5kpLcPBJn3IshOZnmb72F76BrSP3wQ+JGjiRjyRKkwUBOoYEZi/cwf2Mst0W14tt7+hLmZ53tiBcznD1Lwt3TQQhaf/E57k2a2GRcxXn5+voCkJyczK23mtlM3gE06DX0AykH2Ji4kYd6PkRAowozuvx0iHkfOt4ALR2yKKTNmIqLOfXQgxQePUrLDz7Ab+gQAm4cTcHUqZx7623OvPgSZ79YxGdXXsdG3/a8cGNn7hwQbrMa4yWZmSRMn44pK5vWX3+FR3i4TcZVFIDmzZtbvQa60WgsLytQVw06oX+w9wOCPYOZ0mnKhS+UN35+xj6BOQhZUkLyk7PJ27adZq+9ht/QIeWvefXoQevFX7Pru1VkzXuP6b8v4D9XdqXd+P9DiLY2ic+Un0/ivTMxnEyg1aef4mWFk3WKY3tj5xv8m/5vtdeVXWPOOvqVwVcyu0+lTdUuER8fz+jRozl8+DCLFi1i1apV5OfnExcXx9ixY3nzzTcB+O2333j++ecpKioiIiKCL7/8El9f30rL7AohLF42t0yDXXLZcXoHf5/5mxndZ+Dt7n3+BdX4GdAOA52ZM4ec9etp/MQTBI69+ZLXv9oWz6Qj7rw94Tk8Zv8Pr7SznJw8hcRZsyg6fty68RkMJD38CAWHDtH8nbfx6dfXquMpijn279/PsmXLOHToEMuWLSMxMZHU1FTmzJnDH3/8wd69e4mKiiqvoT5r1ix27drF4cOHKSgoYM2aNeX3Kiuba6lkDg10hi6l5P2979PUpynjO1zUT1I1fgYgdf6HZC5ZSsjd0wmZftcFrxUbTTz382GW7krk2k6Nee+2SPw83THdNpb0r78m7dPPOH7jGAJvvZWwWQ9YvLqhNJlI/r+nyfvrL5q+9CL+pTUxFOdj7ky6bGb+5agvrRkOw4YNI6D0gXznzp05efIkmZmZHD16lOhorTBccXEx/fv3B6ouswuWLZtbpkEm9I2JGzmUeoiXBryEh6vH+RdU42cA0r/5ltQPPyRg3C2EXTQ7qHhYaNaQdjw6/PxhIRdvb0JnziRwwgRSP/qYjKVLyVq9mpCpUwm+6y5cfet+1F9KydnXXid7zRrCHnmEoAkT6nxPRbGURo3ObwRwdXXFaDQipWT48OEsWbLkgmsvV2YXLFs2t0yDW3IpMZXwwb4PCPcP58aIG8+/ICVseNnpGz9nrVnL2VdewXfYMJq9qDVQllJy7GwOn289wU3zt3I4OYsPJvXk8ZEdKz356RYcTNP/PUPE2jX4XnMNqR99RNzIkZybN4/8PXuQBkOt40tbsICMxYsJvvM/hNw7oy5vVVFsol+/fsTExBAbGwtAfn4+er3+smV2raXBzdB/if+F2MxY3hr0Fm4uFd7e1nfh0PdwzZNO2/g596+tJD/1FN69e+P14iusOnyWrcdS+etYKmeytW++jk38WPifKLP2l3u0aUPLue9RcGAqKfPmkbZgIWkff4KLjw/e/frhEz0A3+ho3Fu3NmtXTMbSZaTMnYf/mBtpPHu2zXbSKEpdhIWFsWjRIiZNmkRRUREAc+bMoUOHDpcts2sNDap8rsFkYMxPY/D18GXZ6GW4iNJfQPYsgtUPQ7cJMHYBOGF7sqy9+zg17S6yQ5ry9vWPsDtV6zAe4OXO1e1CGdg+lKvbh9IyyLuaO1WtJCuLvL//Ji9mG3lbt2I4dQoA9xYt8ImO1v7061vpoaDs9b9y6r//xeeagbSaP18d6Xdi9b18riU5dfncn479RFJuEh8O+/B8Mj+6Ctb8F9oNh5s/cppkLqUk9lwuW46lcnT7AW5b/DI57j7MjryTK/z8ebxXKAPbh9G1RQCuFiqo5RoQgP+IEfiPGIGUEkNCArkxMeTFbCN77Voyly8HFxe8unXDJ3oAPtHReHXvTv6ePao+i6JYQINJ6IXGQhYcWEDPxj0Z2GKg9skTW+DH6dAiCiZ85RS1zouNJt75XcfP+5I5k11IWH4G82I+wq2RBy5vfMCm6G74NrL+X7sQAo82bQhu04bgyZORBgMFBw9qs/eYGFI/WUDqRx/j4uODNJlUfRZFsYAGk9CX6ZZxruAcb1zzhrb2mrwflkyG4AiYvAw8rNtswRGk5mo7VHbFZzCicxMea9GY7m98AC5G2nyzGM+OlbTesxHh7o5379549+5N2EMPasszO/4mLyYGQ3IyzV6Zo+qzKEodNYiEnlucy2eHPiO6eTRRTaMgLQ6+GQdegXDHCvAOtneIVnckOYsZX+8hNbeIeRMjubF9ICenTqPo7Blaf/6ZXZN5ZVwDAvAfOQL/kWqPuaJYSoNI6IuPLiazKJMHez0I2adh8c2AhDtWOsWOlrUHT/PY9/sJ8vbgh5kD6NLYi6SZ95XXZ/GOcu56NUrDd/KO/wDQZvHXdo7Evup9Qs8ozOCro18xvM1wung3hy+v14pv3bkaQtvZOzyrMpkkc//Q8/6GWHq3CWL+sKZ4bv+NxJ9Xkb979yX1WRRFadjqfUL//NDnFBgLmNVlOnx3m9ZObsr30KKXvUOzqtwiI7O/3k7q1u28I5LpcUBP5rwTALg1aULTF56/pD6LoigNW71O6GfyzrDk3yXc2PYGrvjtBUjcCeMXNdiGFbKkhMKjRzn1+yb+Wf07M8/E4SZNCC8vPK6KInjibfhER+MREaEO5SiKE6rXCb2s8fN9Z5Pg2G8w+j3ocrO9w7IoQ3Jy+V7uvO3bMWVlAeAS1BLDuElcMXo4Xr164uLhUc2dFKX+OfPqqxT9U3353MJ/tWvK1tIvp1GnK2n69NNVvp6Xl8eECRNISkqipKSEJ554grVr17J8+XIANm3axDvvvMPq1avx9fXlgQce4I8//iAoKIhXX32VJ598koSEBObOncuYMWPMfKeWUW8Telnj5/GeLWlx+GcY8j+Iuqv6L7QBKSXFcXHkxcSQt30HxvT0Wt2nJCsTw8kEQFtGOdutD18WNSbzykjeu3cI4aENfyumotja+vXrad68OWvXrgUgKyuLZ599lry8PHx8fFi2bFl5pcS8vDwGDx7MG2+8wdixY/nf//7H77//ztGjR7nzzjtVQjfXRwc+wg3JDN1W6DsTrnncrvEYMzLI27at/OCM8exZQKt34t66da3u6d6kMcGTJ+PRrz+vHC5kSWk5269Ly9kqSkN3uZl0RZbc5dKtWzcef/xxZs+ezejRoxk4cCCjRo1i9erV3Hrrraxdu7a8sYWHhwejRo0q/7pGjRrh7u5Ot27diI+Pr3MsNWVWQhdCjALmAa7AZ1LK1y96XZS+fj2QD0yVUu61cKzltMbPa7krM4vQzuNg5Gtg4zVjU3ExBXv3abPwbdsoPHoUpMTF3x+f/v21o+0DovFo2aJO46TmFnHXN3vZGZ/OA0MieGx45RUQFUWxjA4dOrBnzx7WrVvH//3f/zFixAhuu+02PvzwQ4KDg7nqqqvw8/MDwN3dvfx5lYuLS3l5XRcXF4xGo81jrzahCyFcgQ+B4UASsEsIsUpKebTCZdcB7Uv/9AU+Lv3/VjF/y//wNZmYFtwLbrJNfRYpJcXHj5MXE0NuTAz5O3chCwrAzQ2vyB6EPfQgPtHReHbpgnB1tciYFx8Wuimybj8cFEWpXnJyMsHBwdx+++34+vqyaNEinnnmGaZPn86nn35qlcYUlmLODL0PECulPA4ghFgK3ARUTOg3AV9LrXTjDiFEoBCimZTytKUD/uX92YxefogpJYK9Mgv50ShLD1Ep76J8AvK1B5JnA5rw7xV90bXoRGzzDhR6eEEB8EcG/LHVYmMmpOcT7KMdFurWUh2LVxRbOHToEE888QQuLi64u7vz8ccf4+rqyujRo1m0aBFfffWVvUOskjkJvQWQWOHjJC6dfVd2TQvggoQuhJgBzABoXct1ZREQQk6QK8WuXTGJRtV/gYVkuXmwp3VHEq7oSk7g+ZZrraw4ZlR4MP8d3p7Gfp5WHEVR6j9LnhAdOXIkI0eOvOTz8+fPZ/78+Rd8Ljc3t/y/X3jhhSpfsxVzEnplC7YXF1E35xqklAuBhaDVQzdj7EuMuvNJuNN5Ow4piqJUxZzF5yQunIi2BJJrcY2iKIpiReYk9F1AeyFEWyGEBzARWHXRNauA/whNPyDLGuvniqI4B3t1UnMktfnfoNolFymlUQgxC/gVbdviF1LKI0KImaWvfwKsQ9uyGIu2bXFajSNRFEUBPD09SUtLIyQkxGlLWEgpSUtLw9OzZs/PGlRPUUVR6j+DwUBSUhKFhYX2DsWuPD09admyJe4XtWR0mp6iiqLUf+7u7rRt29beYdRLztExWVEUxQmohK4oitJAqISuKIrSQNjtoagQIgU4WcMvCwVSrRCOI3GG9wjqfTY06n3aThspZVhlL9gtodeGEGJ3VU93GwpneI+g3mdDo96nY1BLLoqiKA2ESuiKoigNRH1L6AvtHYANOMN7BPU+Gxr1Ph1AvVpDVxRFUapW32boiqIoShVUQlcURWkgHC6hCyFGCSF0QohYIcRTlbwuhBDvl75+UAjRyx5x1pUZ73NK6fs7KITYJoToYY8466q691nhuquEECVCiFttGZ+lmPM+hRCDhRD7hRBHhBCbbR2jJZjxfRsghFgthDhQ+j7rZeVVIcQXQohzQojDVbzumHlISukwf9DK88YBVwAewAGg80XXXA/8gtYlqR/wt73jttL7HAAElf73dQ31fVa4bgNaGeZb7R23lf4+A9H68LYu/bixveO20vt8Gnij9L/DgHTAw96x1+K9XgP0Ag5X8bpD5iFHm6GXN6SWUhYDZQ2pKypvSC2l3AEECiGa2TrQOqr2fUopt0kpM0o/3IHWBaq+MefvE+BB4EfgnC2DsyBz3udkYIWUMgFASlkf36s571MCfkIrZO6LltCNtg2z7qSUW9Bir4pD5iFHS+hVNZuu6TWOrqbvYTrabKC+qfZ9CiFaAGOBT2wYl6WZ8/fZAQgSQmwSQuwRQvzHZtFZjjnvcz7QCa0F5SHgYSmlyTbh2ZRD5iFHq4dusYbUDs7s9yCEGIKW0K+2akTWYc77nAvMllKW1OPuNOa8TzegNzAM8AK2CyF2SCn11g7Ogsx5nyOB/cBQIAL4XQjxl5Qy28qx2ZpD5iFHS+jO0pDarPcghOgOfAZcJ6VMs1FslmTO+4wClpYm81DgeiGEUUq50iYRWoa537epUso8IE8IsQXoAdSnhG7O+5wGvC61heZYIcQJ4Epgp21CtBmHzEOOtuTiLA2pq32fQojWwArgjno2i6uo2vcppWwrpQyXUoYDPwD317NkDuZ93/4MDBRCuAkhvIG+wD82jrOuzHmfCWi/hSCEaAJ0BI7bNErbcMg85FAzdOkkDanNfJ/PASHAR6WzV6N04CpvlTHzfdZ75rxPKeU/Qoj1wEHABHwmpax0S5yjMvPv82VgkRDiENqyxGwppb3LzdaYEGIJMBgIFUIkAc8D7uDYeUgd/VcURWkgHG3JRVEURaklldAVRVEaCJXQFUVRGgiV0BVFURoIldAVRVEaCJXQFUVRGgiV0BVFURqI/wfuG9h9yb5PEwAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "if calculate:\n",
    "    total_astuteness = np.zeros(shape=(run_times, len(classifiers), len(epsilon_range)))\n",
    "    for i in range(run_times):\n",
    "        print('Completing Run ' + str(i + 1) + ' of ' + str(run_times))\n",
    "        for j in range(len(classifiers)):\n",
    "            if classifiers[j] == '2layer':\n",
    "                activation = 'relu' if datatype in ['orange_skin', 'XOR'] else 'selu'\n",
    "                model_input = Input(shape=(input_shape,), dtype='float32')\n",
    "                net = Dense(200, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "                net = BatchNormalization()(net)  # Add batchnorm for stability.\n",
    "                net = Dense(200, activation=activation, name='dense2',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = BatchNormalization()(net)\n",
    "                preds = Dense(2, activation='softmax', name='dense4',\n",
    "                              kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                bbox_model = Model(model_input, preds)\n",
    "                bbox_model.load_weights('models/' + datatype + '_blackbox.hdf5',\n",
    "                                        by_name=True)\n",
    "                pred_model = Model(model_input, preds)\n",
    "\n",
    "            elif classifiers[j] == '4layer':\n",
    "                activation = 'relu' if datatype in ['orange_skin', 'XOR'] else 'selu'\n",
    "\n",
    "                model_input = Input(shape=(input_shape,), dtype='float32')\n",
    "                net = Dense(50, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "                net = BatchNormalization()(net)  # Add batchnorm for stability.\n",
    "                net = Dense(50, activation=activation, name='dense2',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = BatchNormalization()(net)\n",
    "                net = Dense(50, activation=activation, name='dense3',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = BatchNormalization()(net)\n",
    "                net = Dense(50, activation=activation, name='dense4',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = BatchNormalization()(net)\n",
    "                preds = Dense(2, activation='softmax', name='dense5',\n",
    "                              kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                bbox_model = Model(model_input, preds)\n",
    "                bbox_model.load_weights('models/' + datatype + '_blackbox_extra.hdf5',\n",
    "                                        by_name=True)\n",
    "                pred_model = Model(model_input, preds)\n",
    "\n",
    "\n",
    "            elif classifiers[j] == 'linear':\n",
    "                activation = None\n",
    "\n",
    "                model_input = Input(shape=(input_shape,), dtype='float32')\n",
    "\n",
    "                net = Dense(200, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "                net = BatchNormalization()(net)  # Add batchnorm for stability.\n",
    "\n",
    "                preds = Dense(2, activation='softmax', name='dense4',\n",
    "                              kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                bbox_model = Model(model_input, preds)\n",
    "                bbox_model.load_weights('models/' + datatype + '_blackbox_linear.hdf5',\n",
    "                                        by_name=True)\n",
    "                pred_model = Model(model_input, preds)\n",
    "            elif classifiers[j] == 'svm':\n",
    "                pred_model = pickle.load(open('models/' + datatype + '_svm.pk', 'rb'))\n",
    "            fname = 'explained_weights/rise/' + 'rise_' + datatype + '_' + classifiers[j] + '_' + str(\n",
    "                i) + '.gz'\n",
    "            explanations = np.loadtxt(fname, delimiter=',')\n",
    "            if classifiers[j] == 'svm':\n",
    "                for k in tqdm(range(len(epsilon_range))):\n",
    "                    _, total_astuteness[i, j, k], _ = calculate_robust_astute_sampled(data=x_val,\n",
    "                                                                                      explainer=pred_model,\n",
    "                                                                                      explainer_type='rise',\n",
    "                                                                                      explanation_type='attribution',\n",
    "                                                                                      ball_r=median_rad,\n",
    "                                                                                      epsilon=epsilon_range[k],\n",
    "                                                                                      num_points=int(\n",
    "                                                                                          prop_points * len(\n",
    "                                                                                              x_val)),\n",
    "                                                                                      NN=False,\n",
    "                                                                                      data_explanation=explanations)\n",
    "            else:\n",
    "                for k in tqdm(range(len(epsilon_range))):\n",
    "                    _, total_astuteness[i, j, k], _ = calculate_robust_astute_sampled(data=x_val,\n",
    "                                                                                      explainer=pred_model,\n",
    "                                                                                      explainer_type='rise',\n",
    "                                                                                      explanation_type='attribution',\n",
    "                                                                                      ball_r=median_rad,\n",
    "                                                                                      epsilon=epsilon_range[k],\n",
    "                                                                                      num_points=int(\n",
    "                                                                                          prop_points * len(\n",
    "                                                                                              x_val)),\n",
    "                                                                                      NN=True,\n",
    "                                                                                      data_explanation=explanations)\n",
    "    pickle.dump(total_astuteness, open(save_astuteness_file, 'wb'))\n",
    "else:\n",
    "    total_astuteness = pickle.load(open(save_astuteness_file, 'rb'))\n",
    "astuteness_mean = total_astuteness.mean(axis=0)\n",
    "astuteness_std = total_astuteness.std(axis=0)\n",
    "image_name = 'plots/rise_' + datatype + '_astuteness_classifiers.PNG'\n",
    "fig, ax = plt.subplots()\n",
    "for i in range(len(classifiers)):\n",
    "    ax.errorbar(x=epsilon_range, y=astuteness_mean[i, :], yerr=astuteness_std[i, :],\n",
    "                label=classifiers[i])\n",
    "plt.legend()\n",
    "plt.savefig(image_name)"
   ]
  }
 ],
 "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.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
