{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "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 pickle\n",
    "import numpy as np\n",
    "import argparse\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "import pandas as pd\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.model_selection import train_test_split\n",
    "from scipy.spatial.distance import pdist\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",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "from tqdm import tqdm\n",
    "\n",
    "from utils.explanations import calculate_robust_astute_sampled\n",
    "\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'telescope'\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 = pd.read_csv('data/magic04.data').values\n",
    "data = data_dict[:, :-1]\n",
    "labels = data_dict[:, -1]\n",
    "labels[labels == 'h'] = 0\n",
    "labels[labels == 'g'] = 1\n",
    "x_train, x_val, y_train, y_val = train_test_split(data, labels, test_size=0.05, stratify=labels, random_state=42)\n",
    "x_train = StandardScaler().fit_transform(x_train)\n",
    "x_val = StandardScaler().fit_transform(x_val)\n",
    "input_shape = x_train.shape[-1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "median_rad = np.median(pdist(x_train))"
   ]
  },
  {
   "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-10-02 22:23:49.143245: 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-10-02 22:23:49.165911: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-10-02 22:23:49.166287: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x55aba6f348a0 executing computations on platform Host. Devices:\n",
      "2021-10-02 22:23:49.166301: I tensorflow/compiler/xla/service/service.cc:158]   StreamExecutor device (0): <undefined>, <undefined>\n",
      "100%|███████████████████████████████████████████| 22/22 [04:36<00:00, 12.56s/it]\n",
      "100%|███████████████████████████████████████████| 22/22 [07:15<00:00, 19.81s/it]\n",
      "100%|███████████████████████████████████████████| 22/22 [04:43<00:00, 12.89s/it]\n",
      "100%|██████████████████████████████████████████| 22/22 [45:45<00:00, 124.81s/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'\n",
    "\n",
    "                model_input = Input(shape=(input_shape,), dtype='float32')\n",
    "\n",
    "                net = Dense(32, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "\n",
    "                preds = Dense(1, activation='sigmoid', name='dense3',\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",
    "\n",
    "                net = Dense(32, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "                net = Dense(32, activation=activation, name='dense2',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = Dense(32, activation=activation, name='dense3',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = Dense(32, activation=activation, name='dense4',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                preds = Dense(1, activation='sigmoid', 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(32, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "\n",
    "                preds = Dense(1, activation='sigmoid', name='dense3',\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
}
