{
 "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",
    "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",
    "import cxplain\n",
    "from cxplain import MLPModelBuilder, ZeroMasking, CXPlain\n",
    "from tensorflow.python.keras.losses import binary_crossentropy\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",
    "import shap\n",
    "\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'rice'\n",
    "run_times = 1\n",
    "prop_points = 1\n",
    "calculate = True\n",
    "epsilon_range = np.arange(0.01, 1.1, 0.05)\n",
    "masking_operation = ZeroMasking()\n",
    "loss = binary_crossentropy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "rice_pd = pd.read_excel('data/Rice_Osmancik_Cammeo_Dataset.xlsx')\n",
    "data = rice_pd.values[:, :-1]\n",
    "labels = rice_pd.values[:, -1]\n",
    "labels[labels == 'Cammeo'] = 0\n",
    "labels[labels == 'Osmancik'] = 1\n",
    "x_train, x_val, y_train, y_val = train_test_split(data, labels, test_size=0.33, 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 = 0.25 * np.median(pdist(x_train))"
   ]
  },
  {
   "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-30 20:29:17.138257: 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-30 20:29:17.141565: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-30 20:29:17.142055: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x564eb9173440 executing computations on platform Host. Devices:\n",
      "2021-09-30 20:29:17.142070: 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 2296 samples, validate on 256 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",
      "2296/2296 [==============================] - 0s 205us/sample - loss: 0.6576 - dense_1_loss: 0.5480 - all_loss: 0.0539 - lambda_1_loss: 0.0556 - val_loss: 0.6038 - val_dense_1_loss: 0.4680 - val_all_loss: 0.0694 - val_lambda_1_loss: 0.0664\n",
      "Epoch 2/5\n",
      "2296/2296 [==============================] - 0s 41us/sample - loss: 0.4836 - dense_1_loss: 0.3740 - all_loss: 0.0538 - lambda_1_loss: 0.0555 - val_loss: 0.5322 - val_dense_1_loss: 0.3963 - val_all_loss: 0.0694 - val_lambda_1_loss: 0.0664\n",
      "Epoch 3/5\n",
      "2296/2296 [==============================] - 0s 42us/sample - loss: 0.4267 - dense_1_loss: 0.3170 - all_loss: 0.0537 - lambda_1_loss: 0.0556 - val_loss: 0.5003 - val_dense_1_loss: 0.3645 - val_all_loss: 0.0694 - val_lambda_1_loss: 0.0664\n",
      "Epoch 4/5\n",
      "2296/2296 [==============================] - 0s 42us/sample - loss: 0.4001 - dense_1_loss: 0.2905 - all_loss: 0.0538 - lambda_1_loss: 0.0556 - val_loss: 0.4797 - val_dense_1_loss: 0.3439 - val_all_loss: 0.0694 - val_lambda_1_loss: 0.0664\n",
      "Epoch 5/5\n",
      "2296/2296 [==============================] - 0s 43us/sample - loss: 0.3825 - dense_1_loss: 0.2734 - all_loss: 0.0539 - lambda_1_loss: 0.0557 - val_loss: 0.4646 - val_dense_1_loss: 0.3288 - val_all_loss: 0.0694 - val_lambda_1_loss: 0.0664\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:27<00:00,  1.24s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 2296 samples, validate on 256 samples\n",
      "Epoch 1/5\n",
      "2296/2296 [==============================] - 1s 328us/sample - loss: 0.8758 - dense_6_loss: 0.7667 - all_loss: 0.0528 - lambda_3_loss: 0.0542 - val_loss: 0.5815 - val_dense_6_loss: 0.4442 - val_all_loss: 0.0661 - val_lambda_3_loss: 0.0712\n",
      "Epoch 2/5\n",
      "2296/2296 [==============================] - 0s 65us/sample - loss: 0.4360 - dense_6_loss: 0.3290 - all_loss: 0.0531 - lambda_3_loss: 0.0544 - val_loss: 0.5155 - val_dense_6_loss: 0.3782 - val_all_loss: 0.0661 - val_lambda_3_loss: 0.0712\n",
      "Epoch 3/5\n",
      "2296/2296 [==============================] - 0s 63us/sample - loss: 0.3925 - dense_6_loss: 0.2847 - all_loss: 0.0528 - lambda_3_loss: 0.0541 - val_loss: 0.4784 - val_dense_6_loss: 0.3411 - val_all_loss: 0.0661 - val_lambda_3_loss: 0.0712\n",
      "Epoch 4/5\n",
      "2296/2296 [==============================] - 0s 65us/sample - loss: 0.3633 - dense_6_loss: 0.2557 - all_loss: 0.0529 - lambda_3_loss: 0.0542 - val_loss: 0.4393 - val_dense_6_loss: 0.3020 - val_all_loss: 0.0661 - val_lambda_3_loss: 0.0712\n",
      "Epoch 5/5\n",
      "2296/2296 [==============================] - 0s 64us/sample - loss: 0.3424 - dense_6_loss: 0.2349 - all_loss: 0.0530 - lambda_3_loss: 0.0543 - val_loss: 0.4207 - val_dense_6_loss: 0.2834 - val_all_loss: 0.0661 - val_lambda_3_loss: 0.0712\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:36<00:00,  1.64s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 2296 samples, validate on 256 samples\n",
      "Epoch 1/5\n",
      "2296/2296 [==============================] - 1s 425us/sample - loss: 0.7523 - dense_8_loss: 0.6428 - all_loss: 0.0544 - lambda_5_loss: 0.0551 - val_loss: 0.7408 - val_dense_8_loss: 0.6015 - val_all_loss: 0.0704 - val_lambda_5_loss: 0.0689\n",
      "Epoch 2/5\n",
      "2296/2296 [==============================] - 0s 98us/sample - loss: 0.6031 - dense_8_loss: 0.4933 - all_loss: 0.0544 - lambda_5_loss: 0.0550 - val_loss: 0.6407 - val_dense_8_loss: 0.5015 - val_all_loss: 0.0703 - val_lambda_5_loss: 0.0689\n",
      "Epoch 3/5\n",
      "2296/2296 [==============================] - 0s 76us/sample - loss: 0.5291 - dense_8_loss: 0.4196 - all_loss: 0.0544 - lambda_5_loss: 0.0551 - val_loss: 0.5926 - val_dense_8_loss: 0.4533 - val_all_loss: 0.0703 - val_lambda_5_loss: 0.0689\n",
      "Epoch 4/5\n",
      "2296/2296 [==============================] - 0s 67us/sample - loss: 0.4985 - dense_8_loss: 0.3894 - all_loss: 0.0546 - lambda_5_loss: 0.0553 - val_loss: 0.5725 - val_dense_8_loss: 0.4332 - val_all_loss: 0.0703 - val_lambda_5_loss: 0.0689\n",
      "Epoch 5/5\n",
      "2296/2296 [==============================] - 0s 68us/sample - loss: 0.4873 - dense_8_loss: 0.3776 - all_loss: 0.0544 - lambda_5_loss: 0.0550 - val_loss: 0.5646 - val_dense_8_loss: 0.4254 - val_all_loss: 0.0703 - val_lambda_5_loss: 0.0689\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:28<00:00,  1.28s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 2296 samples, validate on 256 samples\n",
      "Epoch 1/5\n",
      "2296/2296 [==============================] - 1s 370us/sample - loss: 2.7759 - dense_11_loss: 1.1274 - all_loss: 0.8297 - lambda_7_loss: 0.8175 - val_loss: 2.5733 - val_dense_11_loss: 0.9901 - val_all_loss: 0.7983 - val_lambda_7_loss: 0.7848\n",
      "Epoch 2/5\n",
      "2296/2296 [==============================] - 0s 55us/sample - loss: 2.5634 - dense_11_loss: 0.9156 - all_loss: 0.8297 - lambda_7_loss: 0.8175 - val_loss: 2.5289 - val_dense_11_loss: 0.9457 - val_all_loss: 0.7983 - val_lambda_7_loss: 0.7848\n",
      "Epoch 3/5\n",
      "2296/2296 [==============================] - 0s 52us/sample - loss: 2.5342 - dense_11_loss: 0.8862 - all_loss: 0.8300 - lambda_7_loss: 0.8178 - val_loss: 2.5340 - val_dense_11_loss: 0.9508 - val_all_loss: 0.7983 - val_lambda_7_loss: 0.7848\n",
      "Epoch 4/5\n",
      "2296/2296 [==============================] - 0s 55us/sample - loss: 2.5323 - dense_11_loss: 0.8842 - all_loss: 0.8303 - lambda_7_loss: 0.8181 - val_loss: 2.5242 - val_dense_11_loss: 0.9410 - val_all_loss: 0.7983 - val_lambda_7_loss: 0.7848\n",
      "Epoch 5/5\n",
      "2296/2296 [==============================] - 0s 51us/sample - loss: 2.5296 - dense_11_loss: 0.8818 - all_loss: 0.8300 - lambda_7_loss: 0.8178 - val_loss: 2.5348 - val_dense_11_loss: 0.9516 - val_all_loss: 0.7983 - val_lambda_7_loss: 0.7848\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:30<00:00,  1.40s/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",
    "                model_builder = MLPModelBuilder(num_layers=1, num_units=32, activation=activation, verbose=1,\n",
    "                        batch_size=32, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                        with_bn=False)\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",
    "\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",
    "                model_builder = MLPModelBuilder(num_layers=4, num_units=32, activation=activation, verbose=1,\n",
    "                                batch_size=32, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                with_bn=False)\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",
    "                model_builder = MLPModelBuilder(num_layers=1, num_units=32, activation=activation, verbose=1,\n",
    "                                        batch_size=32, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                        with_bn=False)\n",
    "            elif classifiers[j] == 'svm':\n",
    "                pred_model = pickle.load(open('models/' + datatype + '_svm.pk', 'rb'))\n",
    "                model_builder = MLPModelBuilder(num_layers=2, num_units=32, activation=activation, verbose=1,\n",
    "                                                batch_size=32, learning_rate=0.001, num_epochs=5, \n",
    "                                                early_stopping_patience=15,\n",
    "                                                with_bn=False)\n",
    "            if classifiers[j] == 'svm':\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",
    "                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)\n",
    "plt.show()\n",
    "plt.close()"
   ]
  }
 ],
 "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
}
