{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "9f6d9614",
   "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": "13dd3e17",
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'orange_skin'\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": "55089213",
   "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": "97cfe484",
   "metadata": {},
   "outputs": [],
   "source": [
    "median_rad = 0.5 * np.median(pdist(x_val))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "5bf71327",
   "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": "6ae52984",
   "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:29.968188: 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:29.989834: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-29 18:18:29.990132: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x561e93c06f90 executing computations on platform Host. Devices:\n",
      "2021-09-29 18:18:29.990144: 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 [==============================] - 10s 11us/sample - loss: 0.2097 - dense_2_loss: 0.1395 - all_loss: 0.0040 - lambda_1_loss: 0.0662 - val_loss: 0.1302 - val_dense_2_loss: 0.0607 - val_all_loss: 0.0040 - val_lambda_1_loss: 0.0655\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 6s 7us/sample - loss: 0.1315 - dense_2_loss: 0.0613 - all_loss: 0.0040 - lambda_1_loss: 0.0662 - val_loss: 0.1247 - val_dense_2_loss: 0.0552 - val_all_loss: 0.0040 - val_lambda_1_loss: 0.0655\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 14s 15us/sample - loss: 0.1263 - dense_2_loss: 0.0561 - all_loss: 0.0040 - lambda_1_loss: 0.0662 - val_loss: 0.1207 - val_dense_2_loss: 0.0511 - val_all_loss: 0.0040 - val_lambda_1_loss: 0.0655\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.1232 - dense_2_loss: 0.0531 - all_loss: 0.0040 - lambda_1_loss: 0.0662 - val_loss: 0.1177 - val_dense_2_loss: 0.0482 - val_all_loss: 0.0040 - val_lambda_1_loss: 0.0655\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 12s 13us/sample - loss: 0.1208 - dense_2_loss: 0.0506 - all_loss: 0.0040 - lambda_1_loss: 0.0662 - val_loss: 0.1156 - val_dense_2_loss: 0.0461 - val_all_loss: 0.0040 - val_lambda_1_loss: 0.0655\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:02<00:00,  7.61it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 9s 10us/sample - loss: 0.9277 - dense_7_loss: 0.8098 - all_loss: 0.0624 - lambda_3_loss: 0.0556 - val_loss: 0.6740 - val_dense_7_loss: 0.5558 - val_all_loss: 0.0627 - val_lambda_3_loss: 0.0555\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 9s 10us/sample - loss: 0.6328 - dense_7_loss: 0.5148 - all_loss: 0.0624 - lambda_3_loss: 0.0556 - val_loss: 0.5901 - val_dense_7_loss: 0.4719 - val_all_loss: 0.0627 - val_lambda_3_loss: 0.0555\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 6s 7us/sample - loss: 0.5862 - dense_7_loss: 0.4683 - all_loss: 0.0624 - lambda_3_loss: 0.0556 - val_loss: 0.5566 - val_dense_7_loss: 0.4384 - val_all_loss: 0.0627 - val_lambda_3_loss: 0.0555\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 9s 10us/sample - loss: 0.5594 - dense_7_loss: 0.4415 - all_loss: 0.0624 - lambda_3_loss: 0.0556 - val_loss: 0.5318 - val_dense_7_loss: 0.4135 - val_all_loss: 0.0627 - val_lambda_3_loss: 0.0555\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 5s 6us/sample - loss: 0.5394 - dense_7_loss: 0.4215 - all_loss: 0.0624 - lambda_3_loss: 0.0556 - val_loss: 0.5129 - val_dense_7_loss: 0.3947 - val_all_loss: 0.0627 - val_lambda_3_loss: 0.0555\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:01<00:00, 12.53it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 14s 16us/sample - loss: 1.2013 - dense_9_loss: 0.9527 - all_loss: 0.1243 - lambda_5_loss: 0.1243 - val_loss: 1.1924 - val_dense_9_loss: 0.9435 - val_all_loss: 0.1245 - val_lambda_5_loss: 0.1245\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 4s 5us/sample - loss: 1.1941 - dense_9_loss: 0.9454 - all_loss: 0.1243 - lambda_5_loss: 0.1243 - val_loss: 1.1922 - val_dense_9_loss: 0.9432 - val_all_loss: 0.1245 - val_lambda_5_loss: 0.1245\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 5s 6us/sample - loss: 1.1939 - dense_9_loss: 0.9452 - all_loss: 0.1243 - lambda_5_loss: 0.1243 - val_loss: 1.1928 - val_dense_9_loss: 0.9438 - val_all_loss: 0.1245 - val_lambda_5_loss: 0.1245\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 1.1936 - dense_9_loss: 0.9449 - all_loss: 0.1243 - lambda_5_loss: 0.1243 - val_loss: 1.1925 - val_dense_9_loss: 0.9436 - val_all_loss: 0.1245 - val_lambda_5_loss: 0.1245\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 10s 11us/sample - loss: 1.1935 - dense_9_loss: 0.9448 - all_loss: 0.1243 - lambda_5_loss: 0.1243 - val_loss: 1.1931 - val_dense_9_loss: 0.9442 - val_all_loss: 0.1245 - val_lambda_5_loss: 0.1245\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:01<00:00, 15.04it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 17s 18us/sample - loss: 0.3777 - dense_12_loss: 0.2615 - all_loss: 0.0252 - lambda_7_loss: 0.0910 - val_loss: 0.3192 - val_dense_12_loss: 0.2037 - val_all_loss: 0.0253 - val_lambda_7_loss: 0.0902\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 16s 17us/sample - loss: 0.3066 - dense_12_loss: 0.1904 - all_loss: 0.0252 - lambda_7_loss: 0.0910 - val_loss: 0.2890 - val_dense_12_loss: 0.1735 - val_all_loss: 0.0253 - val_lambda_7_loss: 0.0902\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 13s 14us/sample - loss: 0.2845 - dense_12_loss: 0.1683 - all_loss: 0.0252 - lambda_7_loss: 0.0910 - val_loss: 0.2717 - val_dense_12_loss: 0.1562 - val_all_loss: 0.0253 - val_lambda_7_loss: 0.0902\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.2707 - dense_12_loss: 0.1545 - all_loss: 0.0252 - lambda_7_loss: 0.0910 - val_loss: 0.2616 - val_dense_12_loss: 0.1461 - val_all_loss: 0.0253 - val_lambda_7_loss: 0.0902\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 12s 13us/sample - loss: 0.2612 - dense_12_loss: 0.1450 - all_loss: 0.0252 - lambda_7_loss: 0.0910 - val_loss: 0.2528 - val_dense_12_loss: 0.1373 - val_all_loss: 0.0253 - val_lambda_7_loss: 0.0902\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:04<00:00,  5.17it/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": "82a4d63e",
   "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
}
