{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "d92126f6",
   "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",
    "import pickle\n",
    "\n",
    "from utils.explanations import calculate_robust_astute_sampled\n",
    "\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "e26c52c6",
   "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": "67d1408d",
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'rice'\n",
    "run_times = 4\n",
    "prop_points = 0.1\n",
    "calculate = True\n",
    "epsilon_range = np.arange(0.01, 1.1, 0.05)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "1fb59510",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "22"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(epsilon_range)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "c89119f9",
   "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": 7,
   "id": "c1688da4",
   "metadata": {},
   "outputs": [],
   "source": [
    "median_rad = 1 * np.median(pdist(x_train))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "a0b9789b",
   "metadata": {},
   "outputs": [],
   "source": [
    "save_astuteness_file = 'plots/rise_' + 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": 9,
   "id": "5e5a9c34",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 1 of 4\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-19 09:17:46.823286: 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-19 09:17:46.846052: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2022-05-19 09:17:46.846760: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x55f0f014b970 executing computations on platform Host. Devices:\n",
      "2022-05-19 09:17:46.846788: I tensorflow/compiler/xla/service/service.cc:158]   StreamExecutor device (0): <undefined>, <undefined>\n",
      "100%|█████████████████████████████████████| 22/22 [12:02<00:00, 32.84s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [12:59<00:00, 35.42s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [12:56<00:00, 35.29s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 2 of 4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████| 22/22 [13:04<00:00, 35.68s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [12:56<00:00, 35.29s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [13:34<00:00, 37.03s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 3 of 4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████| 22/22 [13:13<00:00, 36.06s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [13:20<00:00, 36.40s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [13:51<00:00, 37.80s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Completing Run 4 of 4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████| 22/22 [13:47<00:00, 37.61s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [13:59<00:00, 38.17s/it]\n",
      "100%|█████████████████████████████████████| 22/22 [13:51<00:00, 37.80s/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/rice_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",
    "            fname = 'explained_weights/rise/' + 'rise_' + 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=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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "39831f61",
   "metadata": {},
   "outputs": [
    {
     "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": [
    "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(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": null,
   "id": "24fdcc03",
   "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
}
