{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "ba331695",
   "metadata": {
    "scrolled": true
   },
   "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",
    "import argparse\n",
    "import pickle\n",
    "from tqdm import tqdm\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "import cxplain\n",
    "from cxplain import MLPModelBuilder, ZeroMasking, CXPlain\n",
    "from tensorflow.python.keras.losses import categorical_crossentropy\n",
    "\n",
    "from scipy.spatial.distance import pdist\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",
    "\n",
    "from utils.explanations import calculate_stability, calculate_robust_astute_sampled\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ad84b906",
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'nonlinear_additive'\n",
    "run_times = 1\n",
    "prop_points = 0.05\n",
    "calculate = True\n",
    "epsilon_range = np.arange(0.01, 1.1, 0.05)\n",
    "masking_operation = ZeroMasking()\n",
    "loss = categorical_crossentropy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "90d1fe9f",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_dict = pickle.load(open('data/' + datatype + '.pk', 'rb'))\n",
    "\n",
    "x_train, y_train, x_val, _, _, input_shape = data_dict['x_train'], data_dict['y_train'], \\\n",
    "                                       data_dict['x_val'], data_dict['y_val'], \\\n",
    "                                       data_dict['datatype_val'], data_dict['input_shape']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "b5dc9319",
   "metadata": {},
   "outputs": [],
   "source": [
    "median_rad = 0.5 * np.median(pdist(x_val))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "b4693e46",
   "metadata": {},
   "outputs": [],
   "source": [
    "save_astuteness_file = 'plots/cxplain_' + datatype + '_astuteness_classifiers.pk'\n",
    "classifiers = ['2layer', '4layer', 'linear', 'svm']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "2a397a01",
   "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-29 18:18:17.903552: 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-29 18:18:17.926041: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-29 18:18:17.926384: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x55f3eb144f50 executing computations on platform Host. Devices:\n",
      "2021-09-29 18:18:17.926408: 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 900000 samples, validate on 100000 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",
      "900000/900000 [==============================] - 16s 18us/sample - loss: 0.8948 - dense_2_loss: 0.2394 - all_loss: 0.2325 - lambda_1_loss: 0.4229 - val_loss: 0.7407 - val_dense_2_loss: 0.0838 - val_all_loss: 0.2333 - val_lambda_1_loss: 0.4236\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 13s 14us/sample - loss: 0.7280 - dense_2_loss: 0.0726 - all_loss: 0.2325 - lambda_1_loss: 0.4229 - val_loss: 0.7162 - val_dense_2_loss: 0.0593 - val_all_loss: 0.2333 - val_lambda_1_loss: 0.4236\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 12s 13us/sample - loss: 0.7131 - dense_2_loss: 0.0577 - all_loss: 0.2325 - lambda_1_loss: 0.4229 - val_loss: 0.7078 - val_dense_2_loss: 0.0509 - val_all_loss: 0.2333 - val_lambda_1_loss: 0.4236\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.7048 - dense_2_loss: 0.0494 - all_loss: 0.2325 - lambda_1_loss: 0.4229 - val_loss: 0.6976 - val_dense_2_loss: 0.0407 - val_all_loss: 0.2333 - val_lambda_1_loss: 0.4236\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.6992 - dense_2_loss: 0.0438 - all_loss: 0.2325 - lambda_1_loss: 0.4229 - val_loss: 0.6964 - val_dense_2_loss: 0.0395 - val_all_loss: 0.2333 - val_lambda_1_loss: 0.4236\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:01<00:00, 15.74it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 8s 9us/sample - loss: 0.8327 - dense_7_loss: 0.2914 - all_loss: 0.1899 - lambda_3_loss: 0.3515 - val_loss: 0.6186 - val_dense_7_loss: 0.0760 - val_all_loss: 0.1906 - val_lambda_3_loss: 0.3520\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 14s 16us/sample - loss: 0.6130 - dense_7_loss: 0.0716 - all_loss: 0.1899 - lambda_3_loss: 0.3515 - val_loss: 0.6011 - val_dense_7_loss: 0.0586 - val_all_loss: 0.1906 - val_lambda_3_loss: 0.3520\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 10s 11us/sample - loss: 0.6015 - dense_7_loss: 0.0601 - all_loss: 0.1899 - lambda_3_loss: 0.3515 - val_loss: 0.5945 - val_dense_7_loss: 0.0520 - val_all_loss: 0.1906 - val_lambda_3_loss: 0.3520\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 9s 10us/sample - loss: 0.5959 - dense_7_loss: 0.0545 - all_loss: 0.1899 - lambda_3_loss: 0.3515 - val_loss: 0.5941 - val_dense_7_loss: 0.0515 - val_all_loss: 0.1906 - val_lambda_3_loss: 0.3520\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 10s 11us/sample - loss: 0.5914 - dense_7_loss: 0.0500 - all_loss: 0.1899 - lambda_3_loss: 0.3515 - val_loss: 0.5833 - val_dense_7_loss: 0.0407 - val_all_loss: 0.1906 - val_lambda_3_loss: 0.3520\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:02<00:00,  8.09it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 8s 9us/sample - loss: 0.6496 - dense_9_loss: 0.3768 - all_loss: 0.0429 - lambda_5_loss: 0.2299 - val_loss: 0.5472 - val_dense_9_loss: 0.2745 - val_all_loss: 0.0428 - val_lambda_5_loss: 0.2298\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 7s 8us/sample - loss: 0.5470 - dense_9_loss: 0.2743 - all_loss: 0.0429 - lambda_5_loss: 0.2299 - val_loss: 0.5466 - val_dense_9_loss: 0.2740 - val_all_loss: 0.0428 - val_lambda_5_loss: 0.2298\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 7s 7us/sample - loss: 0.5468 - dense_9_loss: 0.2740 - all_loss: 0.0429 - lambda_5_loss: 0.2299 - val_loss: 0.5462 - val_dense_9_loss: 0.2736 - val_all_loss: 0.0428 - val_lambda_5_loss: 0.2298\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 7s 8us/sample - loss: 0.5468 - dense_9_loss: 0.2740 - all_loss: 0.0429 - lambda_5_loss: 0.2299 - val_loss: 0.5465 - val_dense_9_loss: 0.2738 - val_all_loss: 0.0428 - val_lambda_5_loss: 0.2298\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 7s 8us/sample - loss: 0.5466 - dense_9_loss: 0.2739 - all_loss: 0.0429 - lambda_5_loss: 0.2299 - val_loss: 0.5467 - val_dense_9_loss: 0.2740 - val_all_loss: 0.0428 - val_lambda_5_loss: 0.2298\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:02<00:00,  8.72it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 16s 17us/sample - loss: 0.6126 - dense_12_loss: 0.2993 - all_loss: 0.0156 - lambda_7_loss: 0.2977 - val_loss: 0.4523 - val_dense_12_loss: 0.1381 - val_all_loss: 0.0154 - val_lambda_7_loss: 0.2988\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 14s 15us/sample - loss: 0.4462 - dense_12_loss: 0.1329 - all_loss: 0.0156 - lambda_7_loss: 0.2977 - val_loss: 0.4398 - val_dense_12_loss: 0.1256 - val_all_loss: 0.0154 - val_lambda_7_loss: 0.2988\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 14s 15us/sample - loss: 0.4351 - dense_12_loss: 0.1219 - all_loss: 0.0156 - lambda_7_loss: 0.2977 - val_loss: 0.4280 - val_dense_12_loss: 0.1138 - val_all_loss: 0.0154 - val_lambda_7_loss: 0.2988\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 17s 19us/sample - loss: 0.4276 - dense_12_loss: 0.1143 - all_loss: 0.0156 - lambda_7_loss: 0.2977 - val_loss: 0.4222 - val_dense_12_loss: 0.1080 - val_all_loss: 0.0154 - val_lambda_7_loss: 0.2988\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 18s 20us/sample - loss: 0.4218 - dense_12_loss: 0.1085 - all_loss: 0.0156 - lambda_7_loss: 0.2977 - val_loss: 0.4172 - val_dense_12_loss: 0.1030 - val_all_loss: 0.0154 - val_lambda_7_loss: 0.2988\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:05<00:00,  4.40it/s]\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' if datatype in ['orange_skin', 'XOR'] else 'selu'\n",
    "                model_input = Input(shape=(input_shape,), dtype='float32')\n",
    "                net = Dense(200, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "                net = BatchNormalization()(net)  # Add batchnorm for stability.\n",
    "                net = Dense(200, activation=activation, name='dense2',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = BatchNormalization()(net)\n",
    "                preds = Dense(2, activation='softmax', name='dense4',\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=2, num_units=200, activation=activation, verbose=1,\n",
    "                                        batch_size=1000, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                        with_bn=True)\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",
    "                net = Dense(50, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "                net = BatchNormalization()(net)  # Add batchnorm for stability.\n",
    "                net = Dense(50, activation=activation, name='dense2',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = BatchNormalization()(net)\n",
    "                net = Dense(50, activation=activation, name='dense3',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = BatchNormalization()(net)\n",
    "                net = Dense(50, activation=activation, name='dense4',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = BatchNormalization()(net)\n",
    "                preds = Dense(2, activation='softmax', 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=50, activation=activation, verbose=1,\n",
    "                                batch_size=1000, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                with_bn=True)\n",
    "\n",
    "\n",
    "            elif classifiers[j] == 'linear':\n",
    "                activation = None\n",
    "\n",
    "                model_input = Input(shape=(input_shape,), dtype='float32')\n",
    "\n",
    "                net = Dense(200, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "                net = BatchNormalization()(net)  # Add batchnorm for stability.\n",
    "\n",
    "                preds = Dense(2, activation='softmax', name='dense4',\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=200, activation=activation, verbose=1,\n",
    "                                        batch_size=1000, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                        with_bn=True)\n",
    "                \n",
    "            elif classifiers[j] == 'svm':\n",
    "                activation = 'relu' if datatype in ['orange_skin', 'XOR'] else 'selu'\n",
    "                pred_model = pickle.load(open('models/' + datatype + '_svm.pk', 'rb'))\n",
    "                model_builder = MLPModelBuilder(num_layers=2, num_units=200, activation=activation, verbose=1,\n",
    "                                                batch_size=1000, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                                with_bn=True)\n",
    "#             fname = 'explained_weights/cxplain/' + 'cxplain_' + datatype + '_' + classifiers[j] + '_' + str(\n",
    "#                 i) + '.gz'\n",
    "#             explanations = np.loadtxt(fname, delimiter=',')\n",
    "            if classifiers[j] == 'svm':\n",
    "                training_indices = np.random.choice(len(x_train), int(0.01 * len(x_train)), replace=False)\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",
    "                training_indices = np.random.choice(len(x_train), int(0.01 * len(x_train)), replace=False)\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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "483ec596",
   "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
}
