{
 "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",
    "import mat73\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-10-01 11:44:06.990587: 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-01 11:44:07.009978: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-10-01 11:44:07.010446: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x55ee9e0f14e0 executing computations on platform Host. Devices:\n",
      "2021-10-01 11:44:07.010461: 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 ['telescope']:\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",
    "    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]\n",
    "    input_shape = x_train.shape[-1]\n",
    "\n",
    "    median_rad =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
}
