{
 "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 = 'nonlinear_additive'\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:41.746497: 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:41.773975: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-28 22:23:41.774929: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x558cb1ce8d40 executing computations on platform Host. Devices:\n",
      "2021-09-28 22:23:41.774963: I tensorflow/compiler/xla/service/service.cc:158]   StreamExecutor device (0): <undefined>, <undefined>\n",
      "100%|███████████████████████████████████████| 22/22 [37:43<00:00, 102.89s/it]\n",
      "100%|███████████████████████████████████████| 22/22 [45:02<00:00, 122.84s/it]\n",
      "100%|████████████████████████████████████████| 22/22 [22:22<00:00, 61.00s/it]\n",
      "100%|████████████████████████████████████████| 22/22 [17:41<00:00, 48.24s/it]\n"
     ]
    },
    {
     "data": {
      "image/png": "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\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
}
