{
 "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 = 'switch_[10, 10, 10, 70]gt'\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 = ['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 21:36:17.307175: 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 21:36:17.329909: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-29 21:36:17.330392: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x556a0c4eb330 executing computations on platform Host. Devices:\n",
      "2021-09-29 21:36:17.330412: I tensorflow/compiler/xla/service/service.cc:158]   StreamExecutor device (0): <undefined>, <undefined>\n",
      "100%|████████████████████████████████████████| 22/22 [23:22<00:00, 63.77s/it]\n",
      "100%|████████████████████████████████████████| 22/22 [24:30<00:00, 66.85s/it]\n",
      "100%|████████████████████████████████████████| 22/22 [13:27<00:00, 36.70s/it]\n",
      "100%|████████████████████████████████████████| 22/22 [17:01<00:00, 46.42s/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)"
   ]
  },
  {
   "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
}
