{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "785b8f6d",
   "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 matplotlib.pyplot as plt\n",
    "\n",
    "import pickle\n",
    "import numpy as np\n",
    "import argparse\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",
    "import shap\n",
    "\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "693d6453",
   "metadata": {},
   "outputs": [],
   "source": [
    "def set_all_weights(model, all_layer_weights):\n",
    "    count = 0\n",
    "    for layer in model.layers:\n",
    "        if type(layer) is Dense:\n",
    "            count += 1\n",
    "    if count == len(all_layer_weights):\n",
    "        c = 0\n",
    "        for layer in model.layers:\n",
    "            if type(layer) is Dense:\n",
    "                layer.set_weights(all_layer_weights[c])\n",
    "                c += 1\n",
    "        return model\n",
    "    else:\n",
    "        print(\"models don't match\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "a9041049",
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'telescope'\n",
    "run_times = 5\n",
    "prop_points = 1\n",
    "calculate = True\n",
    "epsilon_range = np.arange(0.01, 1.1, 0.05)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "566c22c8",
   "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": 6,
   "id": "a41ae9e1",
   "metadata": {},
   "outputs": [],
   "source": [
    "median_rad = 1 * np.median(pdist(x_train))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "c6e6640e",
   "metadata": {},
   "outputs": [],
   "source": [
    "save_astuteness_file = 'plots/shap_' + datatype + '_astuteness_classifiers_lip.pk'\n",
    "lambda_dense_list =[float(0.7), float(1), float(\"inf\")]\n",
    "lambda_names = ['Regularized High', 'Regularized Low', 'Not Regularized']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "ae6f1a03",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 1 of 5\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": [
      "2022-05-18 10:47:42.115166: 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",
      "2022-05-18 10:47:42.134209: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2022-05-18 10:47:42.134631: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x559af57c80f0 executing computations on platform Host. Devices:\n",
      "2022-05-18 10:47:42.134640: I tensorflow/compiler/xla/service/service.cc:158]   StreamExecutor device (0): <undefined>, <undefined>\n",
      "Using TensorFlow backend.\n",
      "100%|█████████████████████████████████████| 22/22 [18:51<00:00, 51.42s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [19:55<00:00, 54.34s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [20:01<00:00, 54.60s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 2 of 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████| 22/22 [23:42<00:00, 64.66s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [20:55<00:00, 57.09s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [19:11<00:00, 52.32s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 3 of 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████| 22/22 [20:11<00:00, 55.09s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [21:47<00:00, 59.44s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [21:41<00:00, 59.15s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 4 of 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████| 22/22 [36:51<00:00, 100.55s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [32:02<00:00, 87.39s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [31:35<00:00, 86.16s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 5 of 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████| 22/22 [30:32<00:00, 83.31s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [29:10<00:00, 79.57s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [32:51<00:00, 89.62s/it]\n"
     ]
    }
   ],
   "source": [
    "if calculate:\n",
    "    total_astuteness = np.zeros(shape=(run_times, len(lambda_names), len(epsilon_range)))\n",
    "    for i in range(run_times):\n",
    "        print('Completing Run ' + str(i + 1) + ' of ' + str(run_times))\n",
    "        for (j, lambda_dense) in enumerate(lambda_dense_list):\n",
    "            all_layer_weights = pickle.load(open('extracted_weights/telescope_l2_' + str(j) + '.pk', 'rb'))\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",
    "            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 = set_all_weights(bbox_model, all_layer_weights)\n",
    "            pred_model = Model(model_input, preds)\n",
    "            background = x_train[np.random.choice(len(x_train), 100, replace=False)]\n",
    "            explainer = shap.GradientExplainer(pred_model, background)\n",
    "            fname = 'explained_weights/shap/' + 'shap_' + datatype + '_' + str(j) + '_' + str(i) + '_lip.gz'\n",
    "            explanations = np.loadtxt(fname, delimiter=',')\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='shap',\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'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "9c857594",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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0dHio4QFhNEnto/R5QgvFxISQnJyMvv37yc6OppFixYZXRJWq5W//OUv/OY3vzHk9adNm8bEiRMNee1659C11hal1AzgS+xti0u11ilKqWmVjy8GZgExwGuV8z4WrXVS05V9TpH2nYrir2m2l3RXrye/zhnbGRIiEowuRTSjZ9alkJp7ut7jUo/Yj6maSz+fnu1aMvvmXk7XcPXVV7N3714AMjMzmT59OidOnCA0NJQ33niDSy65hMzMTO644w6sVisjRozg5ZdfpqioiG+//Za5c+fy2WefATBjxgySkpK46667znqN+++/nx07dlBaWsptt93GM888A0BCQgJTpkxh48aNzJgxgw0bNjBy5EgSEhK45557AHvQ79+/H611nfUdOHCA8ePHY7FYGD78wpaZnjNnDuHh4cycOZMdO3Zw9913ExYWxsCBA/niiy/Yv38/ALm5uQwfPpzMzExGjx7NCy+8cEGvV51Tzcla6/Va64u11l201s9W3re4MszRWt+jtY7SWidWfjRfmAOczoWKYp/fGDrVnMqK1BW0CmlFy8CWRpcjfIjVauXrr7/mllvsLbJTp05lwYIF7Nq1i7lz5/LAAw8A8NBDD/HQQw+xY8cO2rVr1+DXefbZZ9m5cyd79+7lu+++c/wAAXvf9n//+1/GjRvnuC8pKYnk5GSSk5MZPnw4M2fOrLe+qh8abdvW/f5TaWkpiYmJjo9Zs2bVetzkyZNZvHgx27Ztw2QynfVYcnIy77//Pvv27eP999/n0KFDtT5HQzh1YZHbk23nsNgszNk6h8igSOLCm79rVBjL2ZG0q7tcqoLtl19+oX///gwdOpSioiK2bt3K7bf/ulRzeXk5ANu2bePjjz8GYPz48Y6AddYHH3zAkiVLsFgsHDlyhNTUVHr37g3A2LFjz/t9u3fvZuPGjeetb8uWLaxZswaACRMm8PjjtfZ3OKaaqixfvvycef5Tp05RWFjIgAEDHOdb9RsIwA033EBERAQAPXv25ODBg3To0LiN270j0KsW5fLhOfSVqStJy0/jpWtf4saEG40uR/iIqmArKChg5MiRLFq0iLvuuovIyMizAq8+/v7+2Gw2x+3a+q8PHDjA3Llz2bFjB1FRUdx1111nHRcWFlbrc6ekpDB79mw2bdqEyWTCZrOdtz5X9fVrXWszoENQUJDja5PJhMViafRr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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "astuteness_mean = total_astuteness.mean(axis=0)\n",
    "astuteness_std = total_astuteness.std(axis=0)\n",
    "image_name = 'plots/shap_' + datatype + '_astuteness_classifiers.PNG'\n",
    "fig, ax = plt.subplots()\n",
    "for i in range(len(lambda_names)):\n",
    "    ax.errorbar(x=epsilon_range, y=astuteness_mean[i, :], yerr=astuteness_std[i, :],\n",
    "                label=lambda_names[i])\n",
    "plt.legend()\n",
    "plt.savefig(image_name)\n",
    "plt.show()\n",
    "plt.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "7fdb1a2d",
   "metadata": {},
   "outputs": [],
   "source": [
    "pickle.dump(total_astuteness, open(save_astuteness_file, 'wb'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5b34476f",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
