{
 "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 numpy as np\n",
    "\n",
    "import numpy as np\n",
    "import pickle\n",
    "from datetime import datetime\n",
    "import time\n",
    "from scipy.spatial.distance import pdist\n",
    "\n",
    "import pandas as pd\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "from utils.explanations import calculate_prob_lipschitz\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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "BATCH_SIZE = 32\n",
    "epochs = 2\n",
    "calculate = True\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "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",
      "WARNING:tensorflow:Output \"dense3\" missing from loss dictionary. We assume this was done on purpose. The fit and evaluate APIs will not be expecting any data to be passed to \"dense3\".\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2021-09-30 21:01:38.842985: 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 21:01:38.865964: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-30 21:01:38.866390: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x55e77d680fb0 executing computations on platform Host. Devices:\n",
      "2021-09-30 21:01:38.866404: I tensorflow/compiler/xla/service/service.cc:158]   StreamExecutor device (0): <undefined>, <undefined>\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training classifier with extra layer\n",
      "WARNING:tensorflow:Output \"dense5\" missing from loss dictionary. We assume this was done on purpose. The fit and evaluate APIs will not be expecting any data to be passed to \"dense5\".\n",
      "Training Linear Classifier\n",
      "WARNING:tensorflow:Output \"dense3\" missing from loss dictionary. We assume this was done on purpose. The fit and evaluate APIs will not be expecting any data to be passed to \"dense3\".\n",
      "SVM\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": [
    "for datatype in ['rice']:\n",
    "    save_lipschitz = 'plots/blackbox_' + datatype + '_lipschitz.pk'\n",
    "    classifiers = ['2layer','4layer','linear','svm']\n",
    "    L_range = np.arange(0, 1.1, 0.1)\n",
    "    total_lipschitz = np.zeros(shape=(len(classifiers), len(L_range)))\n",
    "\n",
    "    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]\n",
    "\n",
    "    median_rad = 0.25 * np.median(pdist(x_train))\n",
    "\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",
    "    model = Model(model_input, preds)\n",
    "\n",
    "    model.load_weights('models/' + datatype + '_blackbox.hdf5',\n",
    "                           by_name=True)\n",
    "    pred_model = Model(model_input, preds)\n",
    "    pred_model.compile(loss=None,\n",
    "                       optimizer='rmsprop',\n",
    "                       metrics=None)\n",
    "\n",
    "    if calculate:\n",
    "        total_lipschitz[0, :] = calculate_prob_lipschitz(x_val, pred_model,\n",
    "                                                   r=median_rad,\n",
    "                                                   L_range=L_range,\n",
    "                                                   num_points=len(x_val))\n",
    "\n",
    "    del pred_model\n",
    "\n",
    "    ###\n",
    "\n",
    "    print(\"Training classifier with extra layer\")\n",
    "\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",
    "    model = Model(model_input, preds)\n",
    "    model.load_weights('models/' + datatype + '_blackbox_extra.hdf5',\n",
    "                       by_name=True)\n",
    "    pred_model = Model(model_input, preds)\n",
    "    pred_model.compile(loss=None,\n",
    "                       optimizer='rmsprop',\n",
    "                       metrics=None)\n",
    "    if calculate:\n",
    "        total_lipschitz[1, :] = calculate_prob_lipschitz(x_val, pred_model,\n",
    "                                                   r=median_rad,\n",
    "                                                   L_range=L_range,\n",
    "                                                   num_points=len(x_val))\n",
    "\n",
    "    del pred_model\n",
    "    print('Training Linear Classifier')\n",
    "\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",
    "    model = Model(model_input, preds)\n",
    "\n",
    "    model.load_weights('models/' + datatype + '_blackbox_linear.hdf5',\n",
    "                       by_name=True)\n",
    "    pred_model = Model(model_input, preds)\n",
    "    pred_model.compile(loss=None,\n",
    "                       optimizer='rmsprop',\n",
    "                       metrics=None)\n",
    "\n",
    "\n",
    "    if calculate:\n",
    "        total_lipschitz[2, :] = calculate_prob_lipschitz(x_val, pred_model,\n",
    "                                                   r=median_rad,\n",
    "                                                   L_range=L_range,\n",
    "                                                   num_points=len(x_val))\n",
    "\n",
    "    del pred_model\n",
    "    ###\n",
    "\n",
    "\n",
    "    print(\"SVM\")\n",
    "    svm_classif = pickle.load(open('models/' + datatype + '_svm.pk', 'rb'))\n",
    "\n",
    "    if calculate:\n",
    "        total_lipschitz[3, :] = calculate_prob_lipschitz(x_val, svm_classif,\n",
    "                                                   r=median_rad,\n",
    "                                                   L_range=L_range,\n",
    "                                                   num_points=len(x_val),\n",
    "                                                   NN=False)\n",
    "\n",
    "\n",
    "    if calculate:\n",
    "        pickle.dump(total_lipschitz, open(save_lipschitz, 'wb'))\n",
    "    else:\n",
    "        total_lipschitz = pickle.load(open(save_lipschitz, 'rb'))\n",
    "\n",
    "    image_name = 'plots/classifiers_' + datatype + '_lipschitz.PNG'\n",
    "    plt.figure()\n",
    "    for i in range(len(classifiers)):\n",
    "        plt.errorbar(x=L_range, y=total_lipschitz[i, :], yerr=0,\n",
    "                     label=classifiers[i], marker='x')\n",
    "    plt.legend()\n",
    "    plt.savefig(image_name)\n",
    "    plt.show()\n",
    "    plt.close()"
   ]
  },
  {
   "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
}
