{
 "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",
    "import cxplain\n",
    "from cxplain import MLPModelBuilder, ZeroMasking, CXPlain\n",
    "from tensorflow.python.keras.losses import categorical_crossentropy\n",
    "\n",
    "from scipy.spatial.distance import pdist\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",
    "\n",
    "from utils.explanations import calculate_stability, calculate_robust_astute_sampled\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'XOR'\n",
    "run_times = 1\n",
    "prop_points = 0.05\n",
    "calculate = True\n",
    "epsilon_range = np.arange(0.01, 1.1, 0.05)\n",
    "masking_operation = ZeroMasking()\n",
    "loss = categorical_crossentropy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_dict = pickle.load(open('data/' + datatype + '.pk', 'rb'))\n",
    "\n",
    "x_train, y_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/cxplain_' + 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-29 18:27:58.733150: 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-29 18:27:58.753847: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-29 18:27:58.754449: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x556fa11239b0 executing computations on platform Host. Devices:\n",
      "2021-09-29 18:27:58.754475: I tensorflow/compiler/xla/service/service.cc:158]   StreamExecutor device (0): <undefined>, <undefined>\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "WARNING:tensorflow:From /home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/keras/utils/losses_utils.py:170: to_float (from tensorflow.python.ops.math_ops) is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "Use tf.cast instead.\n",
      "WARNING:tensorflow:From /home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/keras/engine/network.py:1436: update_checkpoint_state (from tensorflow.python.training.checkpoint_management) is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "Use tf.train.CheckpointManager to manage checkpoints rather than manually editing the Checkpoint proto.\n",
      "Train on 900000 samples, validate on 100000 samples\n",
      "WARNING:tensorflow:From /home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/ops/math_ops.py:3066: to_int32 (from tensorflow.python.ops.math_ops) is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "Use tf.cast instead.\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 12s 13us/sample - loss: 0.3785 - dense_2_loss: 0.3139 - all_loss: 0.0317 - lambda_1_loss: 0.0329 - val_loss: 0.2534 - val_dense_2_loss: 0.1891 - val_all_loss: 0.0316 - val_lambda_1_loss: 0.0328\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.2092 - dense_2_loss: 0.1445 - all_loss: 0.0317 - lambda_1_loss: 0.0329 - val_loss: 0.1928 - val_dense_2_loss: 0.1284 - val_all_loss: 0.0316 - val_lambda_1_loss: 0.0328\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.1784 - dense_2_loss: 0.1138 - all_loss: 0.0317 - lambda_1_loss: 0.0329 - val_loss: 0.1651 - val_dense_2_loss: 0.1008 - val_all_loss: 0.0316 - val_lambda_1_loss: 0.0328\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.1658 - dense_2_loss: 0.1011 - all_loss: 0.0317 - lambda_1_loss: 0.0329 - val_loss: 0.1596 - val_dense_2_loss: 0.0953 - val_all_loss: 0.0316 - val_lambda_1_loss: 0.0328\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.1597 - dense_2_loss: 0.0951 - all_loss: 0.0317 - lambda_1_loss: 0.0329 - val_loss: 0.1547 - val_dense_2_loss: 0.0903 - val_all_loss: 0.0316 - val_lambda_1_loss: 0.0328\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:02<00:00,  8.99it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 10s 12us/sample - loss: 0.1788 - dense_7_loss: 0.1129 - all_loss: 0.0330 - lambda_3_loss: 0.0330 - val_loss: 0.0768 - val_dense_7_loss: 0.0111 - val_all_loss: 0.0328 - val_lambda_3_loss: 0.0328\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.0765 - dense_7_loss: 0.0105 - all_loss: 0.0330 - lambda_3_loss: 0.0330 - val_loss: 0.0717 - val_dense_7_loss: 0.0060 - val_all_loss: 0.0328 - val_lambda_3_loss: 0.0328\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.0745 - dense_7_loss: 0.0086 - all_loss: 0.0330 - lambda_3_loss: 0.0330 - val_loss: 0.0710 - val_dense_7_loss: 0.0053 - val_all_loss: 0.0328 - val_lambda_3_loss: 0.0328\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 10s 11us/sample - loss: 0.0740 - dense_7_loss: 0.0081 - all_loss: 0.0330 - lambda_3_loss: 0.0330 - val_loss: 0.0723 - val_dense_7_loss: 0.0066 - val_all_loss: 0.0328 - val_lambda_3_loss: 0.0328\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.0733 - dense_7_loss: 0.0073 - all_loss: 0.0330 - lambda_3_loss: 0.0330 - val_loss: 0.0708 - val_dense_7_loss: 0.0051 - val_all_loss: 0.0328 - val_lambda_3_loss: 0.0328\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:03<00:00,  7.14it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 8s 9us/sample - loss: 0.9995 - dense_9_loss: 0.9335 - all_loss: 0.0330 - lambda_5_loss: 0.0330 - val_loss: 0.9867 - val_dense_9_loss: 0.9210 - val_all_loss: 0.0329 - val_lambda_5_loss: 0.0329\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 9s 11us/sample - loss: 0.9881 - dense_9_loss: 0.9222 - all_loss: 0.0330 - lambda_5_loss: 0.0330 - val_loss: 0.9892 - val_dense_9_loss: 0.9235 - val_all_loss: 0.0329 - val_lambda_5_loss: 0.0329\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 6s 6us/sample - loss: 0.9878 - dense_9_loss: 0.9218 - all_loss: 0.0330 - lambda_5_loss: 0.0330 - val_loss: 0.9872 - val_dense_9_loss: 0.9215 - val_all_loss: 0.0329 - val_lambda_5_loss: 0.0329\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 5s 6us/sample - loss: 0.9876 - dense_9_loss: 0.9216 - all_loss: 0.0330 - lambda_5_loss: 0.0330 - val_loss: 0.9874 - val_dense_9_loss: 0.9217 - val_all_loss: 0.0329 - val_lambda_5_loss: 0.0329\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 5s 6us/sample - loss: 0.9873 - dense_9_loss: 0.9213 - all_loss: 0.0330 - lambda_5_loss: 0.0330 - val_loss: 0.9867 - val_dense_9_loss: 0.9210 - val_all_loss: 0.0329 - val_lambda_5_loss: 0.0329\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:01<00:00, 13.42it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 7s 7us/sample - loss: 0.8443 - dense_12_loss: 0.7033 - all_loss: 0.0535 - lambda_7_loss: 0.0876 - val_loss: 0.6533 - val_dense_12_loss: 0.5122 - val_all_loss: 0.0533 - val_lambda_7_loss: 0.0877\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 7s 7us/sample - loss: 0.5924 - dense_12_loss: 0.4514 - all_loss: 0.0535 - lambda_7_loss: 0.0876 - val_loss: 0.5471 - val_dense_12_loss: 0.4060 - val_all_loss: 0.0533 - val_lambda_7_loss: 0.0877\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 7s 8us/sample - loss: 0.5210 - dense_12_loss: 0.3800 - all_loss: 0.0535 - lambda_7_loss: 0.0876 - val_loss: 0.4937 - val_dense_12_loss: 0.3526 - val_all_loss: 0.0533 - val_lambda_7_loss: 0.0877\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 8s 9us/sample - loss: 0.4832 - dense_12_loss: 0.3422 - all_loss: 0.0535 - lambda_7_loss: 0.0876 - val_loss: 0.4651 - val_dense_12_loss: 0.3240 - val_all_loss: 0.0533 - val_lambda_7_loss: 0.0877\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 8s 8us/sample - loss: 0.4598 - dense_12_loss: 0.3188 - all_loss: 0.0535 - lambda_7_loss: 0.0876 - val_loss: 0.4457 - val_dense_12_loss: 0.3046 - val_all_loss: 0.0533 - val_lambda_7_loss: 0.0877\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:02<00:00,  8.28it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXQAAAD4CAYAAAD8Zh1EAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8QVMy6AAAACXBIWXMAAAsTAAALEwEAmpwYAAA+jUlEQVR4nO3dd3xUVfr48c9JT0hIB0InNAHpiICioCugYncta2+ou27TXXV17a6rrvrV/emqrAq2VREbAnaaUtSQAgEkCSEhIQmENJLMpMzM+f0xASMEmCT3zp2ZPO/XKy9I5uae50J4uPPcc56jtNYIIYTwf0FWByCEEMIYktCFECJASEIXQogAIQldCCEChCR0IYQIECFWDZyUlKQHDhxo1fBCCOGXNm7cuE9rndzWa5Yl9IEDB5KWlmbV8EII4ZeUUoVHek1KLkIIESAkoQshRICQhC6EEAFCEroQQgQISehCCBEgjpnQlVKvKaX2KqWyj/C6Ukr9WymVp5TapJSaYHyYQgghjsWTO/SFwJyjvH4mMLTlYx7wYufDEkII0V7HnIeutV6jlBp4lEPOA97Q7j68G5RScUqpFK11qVFBdkU1jTW889M7OFwOq0PxTdoFe7ZAQ03Hvj8oBOL6Q0wvUCZXHh2NhBXl0mfFdpTLO+2qXVrT2Oyi2enyyniifaLGTuLcPz1n+HmNWFjUByhq9Xlxy9cOS+hKqXm47+Lp37+/AUMHrpc3vcybW99EoawOxTdpDXQyOVZlun9VB/6Mjf6z1i0hauZ95iI1SyPpVQBsdv1gynmNSOht/Sto81+a1no+MB9g0qRJsrPGEexv2s8HOR9wdurZPD79cavD8R2F6+Cr+6H4R0geAWc8DEPPaJWQ28FWCTtXw44VsGMl1LTck8QPgsGnuT8GTYeIWM/OV1XYcq4V7vM21AAKeo/DkTSVvK1Lib30QlIeeqj9sR5BUaWNNbnlfJuzj7U79lHb4EApGNMnlulDk5k+NInx/eMJC5G5D75mlEnnNSKhFwP9Wn3eFygx4Lxd1qLti7A5bFw36jqrQ/EN5dvh6wdh+3KISYFzn4dxv4Gg4I6fMyoBRl3g/tAaKnb8nJA3vQdpr4IKhr6Tfk7wvSdAcMs/mYb9UPDtz/8hVO5wf717HxhxLgyeCYNmQLdEKv/vWbTDSeL113fqj2F/QzPrd1TwXe4+vs0tp6DCBkDv2AjOOj6F6cOSOGlwEvHdwjo1jvBfRiT0JcBtSql3gROBGqmfd1yTs4m3t73N1JSpDE8YbnU41qotg1X/hPQ3ILQbnH4/nHgrhEUZO45SkDTE/XHiPHA0ud8FHEjwqx53xxEeCwNPBnslFP0A2gmhUTBwOkye5076SUN/8Y7BWVdP1TvvEHPGGYQNGNCusBxOF5t21/BtjjuBZxRV43RposKCmZqayLXTBjJ9WDKpSd1QHXmXIgLOMRO6UuodYAaQpJQqBh4AQgG01i8By4GzgDzABshtZScszV/KPvs+Hjv5MatDsU5jLaz9N6x/HpzN7mR5yl+hW5J3xg8Jg4EnuT9Ov89dnslfBfkrIX81RMbDSX90J/B+kyEk/IinqvlgMa79+0m8wbO789ZllHU79rG/pYxyfO9Ybjk1lelDk5kgZRRxBMqqTaInTZqkpdviL7m0i/M/OZ/w4HAWzV3U9e66nM2wcaH7jti2D0Zd6E6oCalWR9YhurmZvNmzCevdhwFvvdnmMbUtZZRv2yijTB+aLGUUcRil1Eat9aS2XrOsfa443Oqi1eys2cnj0x/vWslca9j6CXzzsLsWPeBkmPUw9JlodWSdsv/zz3GUlNLrvvsOfs2TMsrJQ5MZnCxlFNF+ktB9yMItC+ndrTezBs6yOhTvqSqAD278eebKbxbB0Fkdm7niQ7TWVLz6GmGDBxN96qn8VLaf577OZW3ez2WU0X2kjCKMJQndR2SVZ5G+N527TriL0KBQq8PxDkcjLLoaKguMmbniQ+rXraPxp59I+cc/+GLrHm5flEVEaDBnymwUYSJJ6D5iYfZCuod158KhF1odivd8/RCUZsFl/4PjzrY6GkNVvvoqwcnJLIwczrNvpTO+fxwvXzmRHt0jrA5NBDB5j+cDCmoK+GbXN1w6/FKiQg2ekuertn8OG16AyTcHXDJv2LqV+nXrWX38aTy7poCLJ/blnZumSDIXppM7dB/wxtY3CA0K5TcjfmN1KN6xvwQ+vhV6jnav9gwwRS/9l4bQCP4ddTx/P3sEN5w8SB5wCq+QhG6xffZ9fJL3CecMPoekSC/Ns7aSywkfznPXz3+9AEID6671+/Vb6PbVl3w17FReuPlUThnW5ubsQphCSi4We+end2h2NXPNqGusDsU7vn3avWT+7KfcqyoDyFsbCln5qLuD3sWP3SHJXHid3KFbyNZs473t7zGz30wGxQ6yOhzzFa5zL6EffQmMvdzqaAzT7HTx0Kdb+HjNT7y96weizz6LgaMGWx2W6IIkoVvoo7yPqGms4brju0C3BFule755/ECY+4yl88zz9tbxn1V5jO8fzylDkxiQ2K3D56qsb+LWtzby/c5KnuEnwpob6XXTTQZGK4TnJKFbxOFy8ObWNxnfYzzjeoyzOhxzaQ2f3AZ1e+HGryA8xtJwnvpiO59vKePD9N0A9E+IYvrQJKYPTWbq4ERiIz1bB7CtdD83vZFGeW0jz10wkhF3PE7E9OlEDB9mZvhCHJEkdIt8VfgVu+t2c+cJd1odivl++C9sXwaz/wm9x1saSn55HV9sLeN3Mwdz0YS+LT1U9vFxxm7e/n4XwUGKsX3d/cRPGZbE2L5xhAQf/qjp8+xSbl+URUxECItunkr/9V9Stm+fx024hDCDJHQLaK1ZkL2Agd0HMqPfDKvDMVfpJvjyXhg6G6bcanU0vPLdTkKDg7h22iCSY8JJTY7mmmkDaXa6yNhVzbe55azJ3cf/W5HLc9/kEhMewrQhie4EPzSZvvGR/HtFLs9+nXtwsVBydBj5ry0gYtQook480epLFF2YJHQL/FD2A9sqt/Hg1AcJMns/Sys11sHi6yEqEc7/j+X9WcprG1m8sZiLJvQhOeaXLW9Dg4OYPCiByYMSuGPWcKptTazbUeFO8Dn7+GLLHgASuoVRWd/ExRP78uj5xxMRGkzt11/TVFBAn2eelvnmwlKS0C2wIHsBiRGJzB081+pQzPXZnVCRB9d86r1e5kfxxvoCmp0ubpx+7Ha8cVFhnDU6hbNGp6C1Zue+er7N3ccPOyuZkprAlVMGHEzeFa++RmifPsTM6kJN1YRPkoTuZdsrt7O2ZC1/GP8HwoOPvDGC39u0CDLfhlPudO/NaTFbk4M3NxRyxoieDE6Obtf3KqVITY4+WJ75xXnT07FnZNDz3ntRIfLPSVgrgN/v+6bXt7xOZEgklwy/xOpQzFOxA5b+GfpPhVPvsjoaABb9WES1rZmbTzV2s4yKV18jODaWuIu6UFM14bMkoXtRWX0Zn+38jIuGXkRsuIe7yfsbRyMsvg6CQuCiV37eVNnKkJwuXvluJ5MGxDNxQIJh523Mz6duxQrir/gNQVFdpKma8GmS0L3oza1votFcNfIqq0Mxz4GWuOf/B2L7Wh0NAMuzyyiusjPvFGPvzisXLECFhRF/xRWGnleIjpKE7iX7m/azOGcxcwbNoXd0b6vDMUfOFy0tcef5TEtcrTXz1+wgNbkbvxrR07DzOsrLqfn4E2IvOJ+QxETDzitEZ0hC95L3t7+PzWHjulEBusz/Fy1xH7E6moPW7agge/d+5k1PJSjIuCmFlW+9jXY4SLz2WsPOKURnSUL3giZnE29te4upKVMZnjDc6nDMseT30Nzgcy1xX16TT1J0OOeP72PYOZ119VS98w4xv/oVYQMHGnZeITpLEroXLMtfxj77vsBtwlW9C/K+hpP/7FMtcbeV7mdNTjnXnTSQiFDj9iqt+WAxrv37ZZm/8DmS0E3m0i4WblnIcQnHMSVlitXhmGPrJ+5fR19kbRyHmL8mn6iwYK48cYBh59TNzVS8/jqRkyYSOW6cYecVwgiS0E22pngN+TX5XDvq2sBdFr7lY0gZCwnGziLpjJJqO59mlXDZCf2JjfKse6In9n/+OY6SUhKvv8GwcwphFEnoJntr61v07tabWQMDdFl49S7YnQYjz7c6kl947budaOD6kwcaet7K198gLDWV6BmnGnpeIYwgCd1ETpeTzPJMfjXgV4QGGXeX6FMOlFtGnW9pGK3V2Jt554ddzB2TQt944xb8OKqqaMjOJvbcc1FB8k9H+B75qTRRUW0Rjc5Ghsb7zoNCw/lgueXt7wupb3IavpDInpkJQNTECYaeVwijSEI3UW51LkDgJnQfLLc0OpwsWFvA9KFJjOptbHsFe3oGhIQQcfzxhp5XCKNIQjdRblUuCkVqrO/cvRrKB8stH2fspry20fC7cwBbRjoRI0cSFBlp+LmFMIIkdBPlVefRv3t/IkMCNAH4WLnF5dLMX5PPyJTunDzE2P7ruqmJhs3ZRI0fZ+h5hTCSRwldKTVHKbVdKZWnlLq7jddjlVKfKqWylFJblFIBuoKmfXKrchkaJ+UWb1nx0152lNdz86mphk8Rbdi2Dd3YSOR4qZ8L33XMhK6UCgZeAM4ERgKXK6VGHnLY74CtWuuxwAzgaaVUmMGx+pUGRwO7ancxJH6I1aGYwwfLLS+v2UGfuEjOGp1i+Llt6RkARI63dpNrIY7Gkzv0yUCe1jpfa90EvAucd8gxGohR7tuiaKAScBgaqZ/ZUbMDl3YF7h26j5Vb0ndV8WNBFTecPIjQYOMrifaMDEL79CG0Zw/Dzy2EUTz5ye8DFLX6vLjla609D4wASoDNwB+11q5DT6SUmqeUSlNKpZWXl3cwZP+QV5UHBOgMFx8st8xfnU9sZCiXntDP8HNrrbFlpBM5Qcotwrd5ktDbKkbqQz6fDWQCvYFxwPNKqe6HfZPW87XWk7TWk5KTk9sZqn/JrcolLCiM/jH9rQ7FeD5Wbskvr+OLrWVcOaU/3cKN3yGpubgYZ/k+oiZIuUX4Nk8SejHQ+ranL+478dauAz7UbnnATuA4Y0L0T7nVuQyOG0xwkHFd/nyGj5VbXvluJ6FBQYdt4GwUe4bUz4V/8CSh/wgMVUoNannQeRmw5JBjdgGnAyilegLDgXwjA/U3eVV5Um7xgvLaRhZvLOaiiX3oEWNOH3ZbejpB0dGEDw3Av08RUI75/lRr7VBK3QZ8AQQDr2mttyilbml5/SXgEWChUmoz7hLNXVrrfSbG7dNqGmvYa98bmA9Efazc8sb6ApqdLm6cbt67BXt6BpFjx6KCA/DdlggoHhUctdbLgeWHfO2lVr8vAQK0nWD75VTlAATmlEUfKrfYmhy8uaGQX43oyeDkaFPGcNbW0pibS8ws+fEWvk9Wipogr7plhkug3aH7WLll0Y9FVNuaueVUE+/OM7NAa3kgKvyCJHQT5Fbl0j2sOz2iAmzOsg+VW+xNTuavyWfigHgmDkgwb5yMdAgKImLMWNPGEMIoktBNkFuVy5C4IYG3Q5EPlVueX5lLSU0Df51t7qbbtowMwocPJzi6m6njCGEESegG01qTVx2AM1x8qNySt7eW+WvyuXBCH6akJpo2jnY4sGdtIkqmKwo/IQndYGX1ZdQ11zEsfpjVoRjLR8otWmvu+3gLkaHB3HPWCFPHati+HW2zyQpR4TckoRvswKYWQ+ICbIbLlo+h1xjLyy2fZJawPr+CO+ccR1J0uKlj2TMyAaRlrvAbktANllvVktADacrigXLLqAssDaPG3syjy7Yxtl8cl082v6WCPT2dkJ49Cend2/SxhDCC8Y0vurjc6lx6detF97DDWtn4Lx8ptzzz5XYq6xtZcO0JBAeZ/8DZlpFB5ITxgfdwWwQsuUM32IEZLgHFB8otm4treHNDIVdNGcDovsbuFdqW5tJSHKWl8kBU+BVJ6AZqdjWTX5MfWDNcfKDc4nRp/v7xZhK6hXOHydMUD/i5IZc8EBX+QxK6gXbt34XD5QisFaI+UG753w+7yCqu4b65I+geEeqVMW3pGajISCKO885/IEIYQRK6gQ48EA2oKYsWl1vKaxt58vOfmDY4kXPHeu/hpD0jg8jRo1Gh3vkPRAgjSEI3UE5VDsEqmEGxg6wOxRjVRZaXW/752TYamp08fN7xXns46aqvp+Gnn4iU/i3Cz0hCN1BedR4Dug8gLDhA9se2uNyyIb+CD9N3M++UVIb0MKebYlvsm7PB6SRKFhQJPyMJ3UC5VbmB9UB0y0eWlVuaHC7u+zibvvGR3DbTu3+m9ox0ACLHSkMu4V8koRvE1myjuK44cKYsWlxueW3tTnL31vHQuaOIDPPuxhK29AzChw4hONb86ZFCGEkSukF2VO8ACJw7dAvLLcVVNp77OpdZI3ty+oieXh1bu1zYMzNluqLwS5LQDXKgh8uwuACZ4WJhueXhT7cCcP85I70+dmNeHq7aWtkQWvglSegGya3KJTIkkj4xfawOpfMsLLd8s20PX27dwx9OH0rf+Civj29Pdy8okh2KhD+ShG6Q3OpcBscOJkgFwB+pReUWe5OTB5ZsYWiPaG442Zqpn/aMDIITEwntb37zLyGMFgDZxzcE1AwXi8otL6zMo7jKziPnH09YiDU/mraMDCLHj5OGXMIvSUI3QIW9gsqGysCY4WJRuSVvbx0vr9lh+i5ER+PYt4/mXbuIkgeiwk9JQjdAXnUeECAzXCwot2ituf+TbK/sQnQ0tgMNuaR+LvyUJHQDHOjhEhAJ3YJyy5KsEtbt8M4uREdjT89AhYURMWqUZTEI0RmS0A2QW51LfHg8iRHWlAoMY0G5ZX9DM48s3cbYvrFe2YXoaOzp6UQcfzxBYQHSukF0OZLQDXDggajfP0jL+8r964hzvTbkR+m72VfXyMPnHe+VXYiOxNXYiH3rVpmuKPyaJPROcmkXedV5gVFuKc2CiDhIHOy1IX8sqKR3bARj+8V5bcy2NGRnQ3OzLCgSfk0SeiftrtuN3WEPjE0tSjdByhjw4juN9MIqJgyI99p4R2JLb2nIJQld+DFJ6J104IHokHg/n7LobIY9WyDFex0GS6rtlNQ0MMkHEro9I5OwAQMISUiwOhQhOkwSeicdmLLo93PQ9+WAsxF6eS+hpxVWATBpoLVJVGvt3qFI+p8LP+dRQldKzVFKbVdK5Sml7j7CMTOUUplKqS1KqdXGhum7cqty6RPdh26h3awOpXNKN7l/TRnjtSHTC6uICgvmuF4xXhuzLU07C3BWVcn8c+H3Qo51gFIqGHgBOAMoBn5USi3RWm9tdUwc8B9gjtZ6l1Kqh0nx+pzcqtwAqZ9nQWgUJHrvnUZaYSXj+sUREmztG0V7y4KiKKmfCz/nyb+kyUCe1jpfa90EvAucd8gxvwE+1FrvAtBa7zU2TN/U5GyicH9hYMxwKdsEPY+HIO9sJlHf6GBbaa1P1M9tGekExcYSlmrNRthCGMWThN4HKGr1eXHL11obBsQrpVYppTYqpa5u60RKqXlKqTSlVFp5eXnHIvYhO2t24tAO/0/oLlfLDBfv1c+ziqpxurRPzHCxp2cQNW4cKkgeKQn/5slPcFtz2PQhn4cAE4GzgdnAfUqpw3Z60FrP11pP0lpPSk5ObnewvubAphZ+/0C0aic01Xq1fp5WWIVSML6/tQndUVVFU36+TFcUAeGYNXTcd+T9Wn3eFyhp45h9Wut6oF4ptQYYC+QYEqWPyqvKIyQohIGxA60OpXPKWh6I9vJuQh/WI4bYyFCvjdkWe2YmIA25RGDw5A79R2CoUmqQUioMuAxYcsgxnwDTlVIhSqko4ERgm7Gh+p7c6lwGxQ4iNMjapNRppVkQFAo9vNPp0OXSZBRWMXGgb5RbCAkhcvRoq0MRotOOeYeutXYopW4DvgCCgde01luUUre0vP6S1nqbUupzYBPgAl7RWmebGbgvyK3KZVyPcVaH0Xmlm6DHcRDinU6HOXtrqW10+MQDUXtGBhEjRhAUGWl1KEJ0miclF7TWy4Hlh3ztpUM+/xfwL+NC8211TXWU1pdySfwlVofSOVq779CHz/HakGkF7gVFEy1O6LqpCfvmzcRfdqmlcQhhFHms30EHN7Xw9znotaVg2+fVFaLphVUkRYfTP8H7m0C31rBtG7qxkUjZoUgECEnoHZRT5X7e6/c9XCxYIZpWWMWkAfGWtxs+uEORzHARAUISegflVuXSLbQbvbv1tjqUzinNApR7UZEX7K1tYFelzfJyC7gfiIb26UNozy6zsFkEOEnoHZRXnceQuCGW32V2Wtkm93L/8GivDJfe0pDL6hkuWmtsGenSkEsEFEnoHaC1Jrc61/9XiIL7Dt2LK0TTCqoICwni+N6xXhuzLc27d+Ms30fk+HGWxiGEkSShd0C5vZyaxhr/XyFqq4SaIq/Xz8f2jSUsxOKGXC0bWkTJHboIIJLQOyCvyj3DZVj8Yd0N/IuXV4g2NDvZUlLDxAHWbyJhS08nqFs3wocGwLssIVpIQu+AgOnhUprl/tVLJZdNxTU0O7WPLCjKJHLsWFSwd7pLCuENktA7IKcqh6TIJOIjrE9MnVK6CWL7QZR37pjTCisBLO+w6KytpTEnRx6IioAjCb0D8qrz/H9BEbjv0L3YkCu9sIrU5G4kdAvz2phtsWdkgNZESUMuEWAkobeT0+VkR/UO/5/h0lgHFXleK7dordnYsqDISq6GBvY+9TTBCQlEjhtnaSxCGM2jXi7iZ0W1RTQ6G/2/fr5nC6C9NsNlR3k9VbZmyxcU7XniCRpzcuj33/kERVnbekAIo8kdejsd6OHi9zNcvPxA9OCCIgtnuOz/4kuq33mXhBuuJ3r6dMviEMIsktDbKbcqF4UiNc7P958sy4KoJIhJ8cpwaYWVxEWFMji5m1fGO1Tz7t2U3ncfEaNH0+OPf7QkBiHMJgm9nXKrc+kX04/IED/vn12a5S63eKl1QVphFRP7W9OQSzc3s/uOv4DLRZ9nnkaFWftQVgizSEJvp9yqAFjy72iCvT95rdxSWd9Efnm9Zf1byp9/AXtmJikPP0RYv37H/gYh/JQk9HZocDSwq3aX/yf08m3gavbalMWD9XMLNoSuX7+eivnzifv1xXQ/6yyvjy+EN0lCb4f8mnxc2uX/c9C9/EA0rbCK0GDF2H5xXhnvAEdFBbvvvJOw1FR63nOPV8cWwgoybbEdcqtalvwHwqYWYTEQP8grw6UXVjGqdywRod5bZq9dLkru/huumv30f+VV2TNUdAlyh94OedV5hAWF0T+mv9WhdE5pFvQaDUHm//U3OVxkFVd7ff555YKF1H/7LT3v+RsRw/18iqkQHpKE3g65VbkMjhtMSJAfv7FxOWFPttfKLdklNTQ6XF5dIWrftIm9//d/xMyaRdylsgG06DokobdDblWu/68QrdgBzTavrRD9eUGRdxK6s7aW3Xf8hZAeyaQ88rD/7yglRDv48a2md9U01rDXvtf/Z7h4+4FoQRX9EiLp0T3C9LG01pQ98ADNJSUMePNNgmOt3RVJCG+TO3QPHXgg6vcJvSwLgsMhyfy6staatMIqJnlpuX/NBx+wf/lnJP/hD9JJUXRJktA9FFCbWvQcCcGhpg9VVGlnX12jV8otjXl5lD36D6KmTiHxphtNH08IXyQJ3UN5VXnEhMXQM6qn1aF0nNbuKYtem3/u3tDC7ITuamhg9+13EBQVRe8nnkB5YfaOEL5Iaugeyq3OZWjcUP9+yFZTBA3VXlshmlZYRUx4CMN6xpg6TuuWuKE9epg6lhC+TG5lPKC1Jq8qz//r5wcfiI7zynDphVWMHxBPcJB5/wkebIl7vbTEFUISugf22PZQ21wbAEv+N4EKdtfQTVZjb2b7nlpT+7f8oiXun6QlrhCS0D2QU5UDBMAMl9Is9+yWUPOXwWcWVaM1TDKxw2LJ3/8uLXGFaEUSugfWlawjPDicEYkjrA6lc8q890B0Y0ElQQrGmdSQy7l/P7YN35NwzTXSEleIFh4ldKXUHKXUdqVUnlLq7qMcd4JSyqmUuti4EK2ltWZV0Sqmpkz1700t6vZCbanXVoimFVYxIqU73cLNee5uz8oCrYk6YZIp5xfCHx0zoSulgoEXgDOBkcDlSqnDirAtxz0BfGF0kFbKrc5ld91uZvSbYXUonVO6yf2rF+7QHU4XmUXVpvZvsaWnQ3AwkaNHmzaGEP7Gkzv0yUCe1jpfa90EvAuc18Zxvwc+APYaGJ/lVhWtAuDUfqdaGkenlbXMcOllfgL8qawWW5OTiQPNWyFqz8gkYvhwgrpZs0epEL7Ik4TeByhq9Xlxy9cOUkr1AS4AXjraiZRS85RSaUqptPLy8vbGaomVu1YyJmkMSZFJVofSOaVZED8QIszvb5JWYO6CIu1wYN+0icgJE0w5vxD+ypOE3tYkYn3I588Cd2mtnUc7kdZ6vtZ6ktZ6UnJysochWmevbS/ZFdnM7D/T6lA6z4srRDfuqiYlNoI+ceY8c2jYvh1tsxE5fpwp5xfCX3nyxKoYaD2NoC9Qcsgxk4B3W1ZRJgFnKaUcWuuPjQjSKquLVwMwo+8MawPprIYaqNoJ46/0ynAbCyqZYGL93J6eAUCU3KEL8QueJPQfgaFKqUHAbuAy4DetD9BaH9zLTCm1EFjq78kc3PXzvtF9GRw32OpQOqdss/tXL6wQLam2U1LTwE1mJvSMdEJSUghNSTFtDCH80TFLLlprB3Ab7tkr24BFWustSqlblFK3mB2gVWzNNjaUbGBGvxn+3b8FWs1wMX/K4saWDS3MbJlry8gkSsotQhzGo0nCWuvlwPJDvtbmA1Ct9bWdD8t660vW0+RqYma/QKifZ0F0L4g2v3HVxsIqIkODOS7FnIZczaWlOEpLibz+elPOL4Q/k5WiR7CyaCXdw7ozvmcAbJTgxRWiaYWVjOsXR2iwOT9atvR0ACJlAwshDiMJvQ1Ol5M1xWuY3nc6oUHmbwRhqmY7lG/3SrmlvtHBttJaU/u32DMyUVFRRAwfbtoYQvgrSeht2LRvE1WNVf6/OhRgz1bQTq/coWcVVeN0aZNnuKQTOWYMKkRa+QtxKEnobVi5ayUhQSGc1Pskq0PpvIMrRM2/Q08rrEIpmGBSy1xXfT0N27fLfqFCHIEk9DasLFrJCT1PICbM3J12vKI0CyLiIK6/6UNtLKxiWI8YYiPNKVPZN28Gp5PI8ZLQhWiLJPRD7KzZScH+gsBYHQotK0THgMlTL10uTfquKlPLLbb0dFCKyLHeecArhL+RhH6I1UUBsjoUwNkMe7Z4pdySs7eW2gaHqR0W7ekZhA8ZQnD37qaNIYQ/k4R+iJVFKzku4ThSogNgFeK+HHA2emWF6MEFRSbNcNEuF/bMTGnIJcRRSEJvpbKhkszyzMCY3QLeXSFaUEVSdBj9E6JMOX9jbh6uujp5ICrEUUhCb+Xb4m9xaVcAJfQsCI2CxCGmDuNwulifX8HEAfGmtUmwZ7QsKJIHokIckST0VlYVraJHVA9GJhy2IZN/KtsEPY+HoGBTh/ksu4zSmgYunNDXtDHsGRkEJyURKvuHCnFEktBbNDobWVuylpn9Zvp/My4Al8vdZdHkcovWmvlr8klN6sYZI3qaNo4tPYOo8eMD4+9GCJNIQm/xfen32B32wCm3VO2Exv2mrxBdn1/B5t013HRKKkFB5iRbR3k5zUVFUm4R4hgkobdYVbSKqJAoJveabHUoxihreSBq8pTFl1fnkxQdzgXj+xz74A6yZRzY0EISuhBHIwkdcGkXq4tWc1KfkwgLDrM6HGOUZkFQCPQYYdoQP5XtZ3VOOddOG0BEqHl1ent6BiosjIiRAfJsQwiTSIcjYFvFNvba9wZG7/MDSje5k3lIuGlDzF+TT1RYMFdOGWDaGAC2jHQiRo9GhQXIf7biqJqbmykuLqahocHqUCwVERFB3759CQ31vJWGJHRgRdEKglQQ0/tMtzoUY2jtvkMfNse0IUqq7SzJLOGqqQOIizIv0boaGmjYuo3Ea68xbQzhW4qLi4mJiWHgwIFd9iG41pqKigqKi4sZNGjQsb+hhZRccNfPx/cYT1xEnNWhGKO2FGz7TH0gumDtTjRww8me/7B1REN2NjQ3EzleVoh2FQ0NDSQmJnbZZA6glCIxMbHd71K6fELfXbebnKqcwCu3gGlTFmvszbzzQxFnj06hb7w5K0MPsKW7H4hGyh6iXUp7k/mlL6/n0pfXmxSNNTryH1qXT+irilYBBM50RXCXW1DuRUUm+N/3u6hrdDDvlFRTzt+aPSODsEGDCIk3r+mXEIFCEnrRKlJjUxnQ3dwHe15Vtsm93D882vBTNzqcLFi7k5OHJHF8n1jDz9+a1hp7RobsHyq8qqioiJkzZzJixAhGjRrFc889B8C1117L4sWLLY7u6Lr0Q9H9TftJK0vj6lFXWx2KcWyVUPAdDJ1lyuk/ySxhb20jT/3a/J7kTTt34qyuJkoWFAkvCgkJ4emnn2bChAnU1tYyceJEzjjjDNPHdTgchHRya8UundDX7l6LQzsCq37+zcPuFaIn/cHwU7tc7mX+I1K6M31okuHnP5S9ZUGRtMztuh76dAtbS/Yf87itpe5jPKmjj+zdnQfOGXXE11NSUkhJcbfPjomJYcSIEezevfsXxzz88MN8+umn2O12pk2bxssvv0x+fj6//vWvSU93N5LLzc3lsssuY+PGjWzcuJHbb7+duro6kpKSWLhwISkpKcyYMYNp06axdu1azj33XO64445jxn80XbrksrJoJQkRCYxOGm11KMYoToONC+HEW6CX8de0cvte8vbWcfMpqV6ZgWBLTyc4NpawdkzbEsJIBQUFZGRkcOKJJ/7i67fddhs//vgj2dnZ2O12li5dyuDBg4mNjSUzMxOABQsWcO2119Lc3Mzvf/97Fi9ezMaNG7n++uu59957D56rurqa1atXdzqZQxe+Q292NfNd8XecPuB0gk3uRugVTgcs/TPE9IIZfzNliJfX5NM7NoKzx3hn8w97egaR0pCrSzvanXRrB+7M37t5qmFj19XVcdFFF/Hss8/S/ZBdslauXMmTTz6JzWajsrKSUaNGcc4553DjjTeyYMECnnnmGd577z1++OEHtm/fTnZ29sGyjdPpPPgOAODSSy81LOYum9A37tlIbXNt4JRb0l51Pwy9eAFEGL9FW8auKn7YWcnfzx5BaLD5b+wcVVU07dxJ7AUXmD6WEIdqbm7moosu4oorruDCCy/8xWsNDQ389re/JS0tjX79+vHggw8enC9+0UUX8dBDD3HaaacxceJEEhMTKSkpYdSoUaxf33Y5qFu3bobF3WVLLquKVhEeHM6UlClWh9J5tWWw4lEYfBqMMicBzl+TT/eIEC6b3N+U8x/KnpEJQJTMPxdeprXmhhtuYMSIEdx+++2HvX4geSclJVFXV/eLmS8RERHMnj2bW2+9leuuuw6A4cOHU15efjChNzc3s2XLFlNi75IJXWvNqqJVTEmZQlSouQtjvOKLe8HRCGc9BSaUJwr21fP5ljKunDKA6HDvvKmzZ2RAaCgRowPk+YbwG2vXruXNN99kxYoVjBs3jnHjxrF8+fKDr8fFxXHTTTcxevRozj//fE444YRffP8VV1yBUopZs9wzzcLCwli8eDF33XUXY8eOZdy4caxbt86U2LtkySW3Opfddbu5cfSNVofSefmrIHsxnHo3JA42ZYj/fptPaFAQ104baMr522LLSCdi5AiCIiK8NqbwX0bWzk8++WS01od9/ayzzjr4+0cffZRHH320ze//7rvvuP766wkO/vnZ3Lhx41izZs1hx65atarzAbfSJRP6yl0rgQBYHepohGV3QPwgOPnPpgyxr66RxRuLuXBCH3p0905y1U1NNGzOJv6yy7wynhBGueCCC9ixYwcrVqywZHyPErpSag7wHBAMvKK1fvyQ168A7mr5tA64VWudZWSgRlpVtIoxSWNIijR/LrWp1v0bKvLgig8g1Jxk+8b6QhodLm6cbv4y/wMatm1DNzbK/HPhdz766CNLxz9mDV0pFQy8AJwJjAQuV0odutPATuBUrfUY4BFgvtGBGmWvbS/ZFdn+f3deVQBrnoKR58HQX5kyhK3JwRvrC/jViJ4M6WF8G4EjjisNuYToEE8eik4G8rTW+VrrJuBd4LzWB2it12mtq1o+3QCYt/17J60uXg34eblFa1h+p3tHojmPH/v4Dno/rZhqWzO3nOq9u3MAe3o6oX37Etqjh1fHFcLfeZLQ+wBFrT4vbvnakdwAfNbWC0qpeUqpNKVUWnl5uedRGmhV0Sr6RPdhSNwQS8Y3xE/LIPcL9wKi7r1NGcLhdPHKd/lM6B/HpIEJpozRFq01tkxpyCXaacHZ7o8uzpOE3tY8uMMfAQNKqZm4E/pdbb2utZ6vtZ6ktZ6UnJzseZQGsTXb2FCygZn9Zvrv6sPGOvjsLugxCk682bRhPssuo6jSzrxTzJk5cyTNxcU4y/cRJfVzIdrNk4ReDPRr9XlfoOTQg5RSY4BXgPO01hXGhGes9SXraXI1+ffq0DVPwv5imPsMBHu+12B7aO1uwjUoqRtnjOxpyhhHYm9pbBQpHRaFxZxOJ+PHj2fu3LmAf7TP9SSh/wgMVUoNUkqFAZcBS1ofoJTqD3wIXKW1zjE+TGOsLFpJTFgM43v6abLYuw3WvwDjr4T+5q1wXZ9fwebdNdw0PZXgIO++k7FlZBAUHU34ED8uiYmA8NxzzzFixAivjedwODp9jmNOW9RaO5RStwFf4J62+JrWeotS6paW118C7gcSgf+0lDIcWutJnY7OQHaHnRVFKzi176mEBplzZ2sqrd1zzsNj4FcPmzrUy6vzSYoO48IJR3tUYg57egaR48ahggOgYZrovM/uhrLNxz6urGXbRU/q6L1Gw5lHn0xQXFzMsmXLuPfee3nmmWcOe92v2+dqrZdrrYdprQdrrf/R8rWXWpI5WusbtdbxWutxLR8+lcwBvij4gtqmWi4ceuGxD/ZFWe9C4Vr41UPQLdG0YX4q28/qnHKumTqQiFDvJlXn/v005ubKdEVhuT/96U88+eSTBAW1nSKlfa7FFm1fRGpsKpN6+tz/Ncdmq4Qv/w59J8P4q0wdav6afCJDg7lqqve35LNnbQKt5YGo+Nkx7qQPOnBnft2yTg+5dOlSevTowcSJE4+4NF/a51poa8VWNu/bzN2T7/bP2S0rHgF7lftB6BHuGIxQVGljSWYJV04ZQFxUmGnjHIk9Ix2CgogcM8brYwtxwNq1a1myZAnLly+noaGB/fv3c+WVVx7cHk7a51rs/Zz3iQiO4JzB51gdSvsVb4S0BabtQnRA9u4aLn15PSHBihtOtmaHIFt6BuHHDSfIwB9wIdrrn//8J8XFxRQUFPDuu+9y2mmn8dZbbx18XdrnWqiuqY5l+cs4c9CZdA8zfuMHU7mcsPRP7l2IZpqzCxHA0k0lXPzSOjTw/s3T6Jfg/ZbC2uHAvmkTUeOl3CJ8m7TPtdDS/KXYHXYuGX6J1aG0348tuxD9eqF7dovBXC7N/32dw/9bkcfEAfG8eOUEesRY0662Yft2tM0m889FxxhQO2/LjBkzmDFjBgALFy48+HVpn2sBrTWLchYxImEEoxI925vQZ9TucdfOB58GI883/PR1jQ7+/F4mX23dwyWT+vLI+ccTHmLdVEF7S0OuKFnyL/yYX7TP9VdZ5VnkVuXywNQH/O9h6Jfm7UK0q8LGjW/8yI7yeh44ZyTXThto+Z+PPSODkF69CO1tTm8aIbzB6va5AZ3Q39v+HtGh0Zw16KxjH+xL8lfB5vdN2YVoXd4+fvu/dLSG16+bzMlDfaMnvC0jQ+afC9FJAftQtKqhii8LvmRu6lz/2jfU0QjL/mL4LkRaa15fV8BVr/1AUnQ4n/zuJJ9J5s2lpThKS+WBqBCdFLB36Et2LKHJ1cSvh//a6lDaZ92/oSLX0F2ImhwuHliSzTs/FHH6cT149rJxxET4TvsDe0bLhhayoEh00HWfu6cILpizwOJIrBWQCd2lXbyf8z7je4xnWPwwq8PxnAm7EO2ra+TWtzbyY0EVv50xmDtmDfd6w61jsaVnoCIjiRjuR39XQviggCy5fF/6PYX7C/1rqmLrXYhm/9OQU24pqeG859eyqbiG5y4bx51zjvO5ZA7ulrmRY8agQn3nXYPo2qKj3VsulpSUcPHFF1scjecCMqG/n/M+ceFxnDHgDKtD8VzrXYhiO9/lcNmmUi56cR0urVl8yzTOG+f9zomecNXX07B9u+xQJHxS7969Te+BbkTb3AMCruSy17aXFbtWcNXIqwgPDrc6HM801R+2C1FZTQMvrd5Bo8PZ7tPttztYtrnU8sVCnrBlZILTSZQsKBJteOKHJ/ip8qdjHnfgmAO19KM5LuE47prc5qZqhykoKGDu3LlkZ2ezcOFClixZgs1mY8eOHVxwwQU8+eSTAHz55Zc88MADNDY2MnjwYBYsWEB0dHSbbXaVUoa3zT0g4BL6R7kf4dROfj3Mjx6Grn7CvQvRxa9CcCjNThe3vr2R7N01xHewSdaVU/pz39yRli4WOpbmPXsou/9+ghMS5IGo8AuZmZlkZGQQHh7O8OHD+f3vf09kZCSPPvooX3/9Nd26deOJJ57gmWee4f777+e2227j/vvvB+Cqq65i6dKlnHOOu6fUgba5RgqohO50OVmcu5ipKVPp372/1eF4po1diJ75KoeMXdW88JsJnD0m5Rgn8E/O6mqKbrwRZ00N/V9/neCWmqUQrXl6J+2tWS6nn346sbGxAIwcOZLCwkKqq6vZunUrJ510EgBNTU1MnToVOHKbXTC2be4BAZXQv939LWX1Zdx1gmc/BJZrYxeiNTnlvLhqB5dP7h+wydxlt1N0629pKiik33/nE3m8n7VlEF1WePjPZdzg4GAcDgdaa8444wzeeeedXxx7tDa7YGzb3AMC6qHoou2LSI5M5tR+pxp74j1b4PN7oG6vsec9ZBei8tpGbl+UxbCe0dw/d6SxY/kI3dxM8R//iD0ri95PPUW3KebtjSqEN0yZMoW1a9eSl5cHgM1mIycn56htds0SMHfou+t2893u77h57M3G7hm6dQl8dAs018PWT+Cyt6H3uM6f95BdiFwuze2LMqltaOZ/N51IZJjv1r47SrtclNx7L/VrvqXXww/RffYsq0MSotOSk5NZuHAhl19+OY2NjYC7G+OwYcMOttkdOHDgYW12TaG1tuRj4sSJ2kjPbnxWj3l9jC6tKzXmhE6n1isf1/qB7lrPn6l1zpdaPz1S60d6ar35g86f/9M/af1gnNalm7TWWr+4Kk8PuGupfntDYefP7YNcLpcue+wxvXX4cbr8xZesDkf4sK1bt1odgs9o688CSNNHyKsBUXJpdjbzYe6HnNL3FHp169X5EzbVw/vXwKrHYOzlcO1yGHoGzFsJKWNh8XXwzSPgcnXs/IfsQpSxq4qnvtjO2aNTuHxyv87H74MqXp5P5etvEH/1VSTePM/qcIQISAGR0L8p+obKhkouHW7AU+PqXfDqbPhpKcz6B5z/4s89VaJ7wDWfwoSr4dun4L0robG2fedvvQvRjL9RY2/m9+9k0Cs2gscuHG15G1szVC1aRPmzz9L93HPoebef7usqhB8IiIT+/vb36RPdh2m9p3XuRAVrYf4Md1K/4n2YdtvhvchDwuCcf8OZ/4Kcz+GVM6Ay3/MxDuxCNPsxdHgM93y4mdKaBv59+XhiIwNv6fv+L7+k7MGH6HbKdHr/4x8oEze5FqKr8/t/Xfk1+fxQ9gMXD7uYINWJy0l7Dd44FyIT4KYVMOQozbGUghPnwVUfQl0Z/Pc0dw/zY6ktc+9ClDoTRl3Auz8WsWxzKX+ZNZwJ/eM7HruPqt+wgZI7/kLkmDH0ffZZ6dUihMn8PqG/v/19QoJCuGDIBR07gbMZlt4OS//sTrQ3fQNJQzz73tQZ7uQf3QvevBC+f9k9t/xIvvw7OBrg7KfJ2VvHg0u2MH1oEjefktqx2H2YPXsLxb/9HWEDB9DvpRcJivKjnvTC7xRedTWFV11tdRiW8+uE3uBo4JMdn3BG/zNIjExs/wnqK+DNCyDtVZj2B/jNexAR275zJKTCjV/BsNnw2Z3w6R/A0XT4cQd2ITr5zzR0H8ht/0snJiKEpy8ZS5APdkDsjMadOymaN4/guDj6vfIKwXFxVockRJfg1wn9i4IvqG2q7dgmFmXZ8N8ZUPQDXDAfZj0CQR2c+x0eA5e+DdP/AulvwOvn/HIR0sFdiAbCyX/m4aVbydlTxzOXjPPpxlkd0bxnD0U33AhAv1dfIbRnT4sjEqLr8OuEvmj7IlJjU5nUc1L7vnHrEnh1lrvcct1nMNaA2TFBQXD6fXDxa1CaBfNnQkmm+7UDuxCd9RTLtlXzv+93ccupgzllWHLnx/UhB/uzVFfTb/58wgcNsjokIboUv10puq1iG5v2beLuye2YBudywZp/ueeX95novqvubnC/lOMvgoTB8O4V8Nocd5Jf8xSMOJeixJO4+9/fMq5fHHfMCqzdeaQ/izBD2WOP0bjt2O1zG35yH+NJHT18xHH0uueeI75eX1/PJZdcQnFxMU6nk7/+9a8sW7aMRYsWAbBq1SqefvppPv30U6Kjo/nd737H119/TXx8PI899hh33nknu3bt4tlnn+Xcc8/18EqN4bd36O/nvE9EcARzU+ce/UCtoXw7bHgRXp/7y8VCRifzA3qP+3kR0hf3gAqmedZj/OHdDNDw/y4fT2iw3/7RH6S1pjE3l4qFCym86mrpzyICwueff07v3r3JysoiOzub888/nw0bNlBfXw/Ae++9d7BTYn19PTNmzGDjxo3ExMTw97//na+++oqPPvroYNtcb/LLO/S6pjqW5i9lzqA5xIa38RCzvgJ2roIdK2DHSti/2/31hMFw5pMwed7h88uNFt0DrlkCq5+EXqN55vv6gy1x+yX474wPR0UF9evWU792LfXr1uHY635WEJaaSu8nHpf+LMJQR7uTbu3AnfmAN9/o9JijR4/mL3/5C3fddRdz585l+vTpzJkzh08//ZSLL76YZcuWHdzYIiwsjDlz5hz8vvDwcEJDQxk9ejQFBQWdjqW9PEroSqk5wHNAMPCK1vrxQ15XLa+fBdiAa7XW6QbHetCy/GXYHXYuGdayZ6ijCYq+b0ngK9w1bLR7xkrqDEj9Kwye6X4o6U0h4XD6fXybW86Lq37wy5a4rsZG7Onp1K9bR93atTRu3QZAcGwsUdOmEn3SSXSbNo3Q3r0tjlQIYwwbNoyNGzeyfPly/va3vzFr1iwuvfRSXnjhBRISEjjhhBOIiYkBIDQ09GDJNygo6GB73aCgIEO3lvPUMRO6UioYeAE4AygGflRKLdFab2112JnA0JaPE4EXW341nNaa97a/x4juqRyf9y18+QgUfOfuhhgUAn1PgJn3wODToPf4js9cMUh5bSN/fs9/WuJqrWnKy6Nu7Vrq167D9uOP6IYGCAkhavx4kv/0J7qddBIRI0egggOvI6QQJSUlJCQkcOWVVxIdHc3ChQu59957ueGGG/jvf/9rysYURvHkDn0ykKe1zgdQSr0LnAe0TujnAW+0dALboJSKU0qlaK1LjQ74i2dv55YPtpHscJLv2oJDhWKnJw0qggbCcakm4POWD+s5XZp/aE3/hChKvvmX1eEck7OmBmf5PsBdRom7+GK6nTSNqBMmExxtfEN+IXzN5s2b+etf/0pQUBChoaG8+OKLBAcHM3fuXBYuXMjrr79udYhH5ElC7wMUtfq8mMPvvts6pg/wi4SulJoHzAPo379jW8Tp+GRqE4KJcvVhT3AiTUG+P4+7b3wkMX4y3zwoIpyoE06QMorwK0bUzg+YPXs2s2fPPuzrzz//PM8///wvvlZXV3fw9w8++OARX/MWTxJ6W08PD13f7skxaK3nA/MBJk2adJQ18kd25rX3wLWePSgRQoiuxJO5c8VA6ybdfYGSDhwjhBDCRJ4k9B+BoUqpQUqpMOAyYMkhxywBrlZuU4AaM+rnQoiuQR+tyV0X0ZE/g2OWXLTWDqXUbcAXuKctvqa13qKUuqXl9ZeA5binLObhnrZ4XbsjEUIIICIigoqKChITE7vsZihaayoqKoiIaN+zN2XV/4STJk3SaWlplowthPBdzc3NFBcX09DQYHUoloqIiKBv376EHrKPgFJqo9a6zQZWfrlSVAgRuEJDQxkkjd06xP8bigghhAAkoQshRMCQhC6EEAHCsoeiSqlyoLCd35YE7DMhHF/SFa4R5DoDjVyn9wzQWre5O45lCb0jlFJpR3q6Gyi6wjWCXGegkev0DVJyEUKIACEJXQghAoS/JfT5VgfgBV3hGkGuM9DIdfoAv6qhCyGEODJ/u0MXQghxBJLQhRAiQPhcQldKzVFKbVdK5Sml7m7jdaWU+nfL65uUUhOsiLOzPLjOK1qub5NSap1SaqwVcXbWsa6z1XEnKKWcSqmLvRmfUTy5TqXUDKVUplJqi1JqtbdjNIIHP7exSqlPlVJZLdfpl51XlVKvKaX2KqWyj/C6b+YhrbXPfOBuz7sDSAXCgCxg5CHHnAV8hnuXpCnA91bHbdJ1TgPiW35/ZqBeZ6vjVuBuw3yx1XGb9PcZh3sf3v4tn/ewOm6TrvMe4ImW3ycDlUCY1bF34FpPASYA2Ud43SfzkK/doR/ckFpr3QQc2JC6tYMbUmutNwBxSqkUbwfaSce8Tq31Oq11VcunG3DvAuVvPPn7BPg98AGw15vBGciT6/wN8KHWeheA1tofr9WT69RAjHI3Mo/GndAd3g2z87TWa3DHfiQ+mYd8LaEfabPp9h7j69p7DTfgvhvwN8e8TqVUH+AC4CUvxmU0T/4+hwHxSqlVSqmNSqmrvRadcTy5zueBEbi3oNwM/FFr7fJOeF7lk3nI1/qhG7YhtY/z+BqUUjNxJ/STTY3IHJ5c57PAXVprpx/vTuPJdYYAE4HTgUhgvVJqg9Y6x+zgDOTJdc4GMoHTgMHAV0qpb7XW+02Ozdt8Mg/5WkLvKhtSe3QNSqkxwCvAmVrrCi/FZiRPrnMS8G5LMk8CzlJKObTWH3slQmN4+nO7T2tdD9QrpdYAYwF/SuieXOd1wOPaXWjOU0rtBI4DfvBOiF7jk3nI10ouXWVD6mNep1KqP/AhcJWf3cW1dszr1FoP0loP1FoPBBYDv/WzZA6e/dx+AkxXSoUopaKAE4FtXo6zszy5zl2434WglOoJDAfyvRqld/hkHvKpO3TdRTak9vA67wcSgf+03L06tA93eWuLh9fp9zy5Tq31NqXU58AmwAW8orVuc0qcr/Lw7/MRYKFSajPussRdWmur2822m1LqHWAGkKSUKgYeAELBt/OQLP0XQogA4WslFyGEEB0kCV0IIQKEJHQhhAgQktCFECJASEIXQogAIQldCCEChCR0IYQIEP8fLwtE3Ciw7PwAAAAASUVORK5CYII=\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",
    "                model_builder = MLPModelBuilder(num_layers=2, num_units=200, activation=activation, verbose=1,\n",
    "                                        batch_size=1000, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                        with_bn=True)\n",
    "            \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",
    "                model_builder = MLPModelBuilder(num_layers=4, num_units=50, activation=activation, verbose=1,\n",
    "                                batch_size=1000, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                with_bn=True)\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",
    "                model_builder = MLPModelBuilder(num_layers=1, num_units=200, activation=activation, verbose=1,\n",
    "                                        batch_size=1000, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                        with_bn=True)\n",
    "                \n",
    "            elif classifiers[j] == 'svm':\n",
    "                activation = 'relu' if datatype in ['orange_skin', 'XOR'] else 'selu'\n",
    "                pred_model = pickle.load(open('models/' + datatype + '_svm.pk', 'rb'))\n",
    "                model_builder = MLPModelBuilder(num_layers=2, num_units=200, activation=activation, verbose=1,\n",
    "                                                batch_size=1000, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                                with_bn=True)\n",
    "#             fname = 'explained_weights/cxplain/' + 'cxplain_' + datatype + '_' + classifiers[j] + '_' + str(\n",
    "#                 i) + '.gz'\n",
    "#             explanations = np.loadtxt(fname, delimiter=',')\n",
    "            if classifiers[j] == 'svm':\n",
    "                training_indices = np.random.choice(len(x_train), int(0.01 * len(x_train)), replace=False)\n",
    "                explainer = CXPlain(pred_model, model_builder, masking_operation, loss, num_models=1)\n",
    "                explainer.fit(x_train, y_train)\n",
    "                for k in tqdm(range(len(epsilon_range))):\n",
    "                    _, total_astuteness[i, j, k], _ = calculate_robust_astute_sampled(data=x_val,\n",
    "                                                                                      explainer=explainer,\n",
    "                                                                                      explainer_type='cxplain',\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=None)\n",
    "            else:\n",
    "                training_indices = np.random.choice(len(x_train), int(0.01 * len(x_train)), replace=False)\n",
    "                explainer = CXPlain(pred_model, model_builder, masking_operation, loss, num_models=1)\n",
    "                explainer.fit(x_train, y_train)\n",
    "                for k in tqdm(range(len(epsilon_range))):\n",
    "                    _, total_astuteness[i, j, k], _ = calculate_robust_astute_sampled(data=x_val,\n",
    "                                                                                      explainer=explainer,\n",
    "                                                                                      explainer_type='cxplain',\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=None)\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/cxplain_' + 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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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
}
