{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "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,
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'switch'\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,
   "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,
   "metadata": {},
   "outputs": [],
   "source": [
    "median_rad = 0.5 * np.median(pdist(x_val))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "save_astuteness_file = 'plots/cxplain_' + datatype + '_astuteness_classifiers.pk'\n",
    "classifiers = ['2layer', '4layer', 'linear', 'svm']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "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:27:53.073250: 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:27:53.093844: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-29 18:27:53.094382: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x55d49202e8f0 executing computations on platform Host. Devices:\n",
      "2021-09-29 18:27:53.094403: 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 [==============================] - 13s 15us/sample - loss: 0.9978 - dense_2_loss: 0.2236 - all_loss: 0.3829 - lambda_1_loss: 0.3914 - val_loss: 0.9249 - val_dense_2_loss: 0.1488 - val_all_loss: 0.3838 - val_lambda_1_loss: 0.3923\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 13s 15us/sample - loss: 0.8853 - dense_2_loss: 0.1110 - all_loss: 0.3829 - lambda_1_loss: 0.3914 - val_loss: 0.8731 - val_dense_2_loss: 0.0970 - val_all_loss: 0.3838 - val_lambda_1_loss: 0.3923\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 13s 15us/sample - loss: 0.8648 - dense_2_loss: 0.0906 - all_loss: 0.3829 - lambda_1_loss: 0.3914 - val_loss: 0.8703 - val_dense_2_loss: 0.0942 - val_all_loss: 0.3838 - val_lambda_1_loss: 0.3923\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 13s 15us/sample - loss: 0.8543 - dense_2_loss: 0.0800 - all_loss: 0.3829 - lambda_1_loss: 0.3914 - val_loss: 0.8514 - val_dense_2_loss: 0.0753 - val_all_loss: 0.3838 - val_lambda_1_loss: 0.3923\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 13s 15us/sample - loss: 0.8477 - dense_2_loss: 0.0734 - all_loss: 0.3829 - lambda_1_loss: 0.3914 - val_loss: 0.8494 - val_dense_2_loss: 0.0733 - val_all_loss: 0.3838 - val_lambda_1_loss: 0.3923\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:03<00:00,  7.25it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 14s 15us/sample - loss: 0.9535 - dense_7_loss: 0.2509 - all_loss: 0.3377 - lambda_3_loss: 0.3649 - val_loss: 0.8528 - val_dense_7_loss: 0.1485 - val_all_loss: 0.3384 - val_lambda_3_loss: 0.3659\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 12s 13us/sample - loss: 0.8278 - dense_7_loss: 0.1251 - all_loss: 0.3377 - lambda_3_loss: 0.3649 - val_loss: 0.8159 - val_dense_7_loss: 0.1116 - val_all_loss: 0.3384 - val_lambda_3_loss: 0.3659\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 10s 11us/sample - loss: 0.8094 - dense_7_loss: 0.1068 - all_loss: 0.3377 - lambda_3_loss: 0.3649 - val_loss: 0.7994 - val_dense_7_loss: 0.0952 - val_all_loss: 0.3384 - val_lambda_3_loss: 0.3659\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.7997 - dense_7_loss: 0.0970 - all_loss: 0.3377 - lambda_3_loss: 0.3649 - val_loss: 0.7920 - val_dense_7_loss: 0.0877 - val_all_loss: 0.3384 - val_lambda_3_loss: 0.3659\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 12s 14us/sample - loss: 0.7922 - dense_7_loss: 0.0896 - all_loss: 0.3377 - lambda_3_loss: 0.3649 - val_loss: 0.7864 - val_dense_7_loss: 0.0821 - val_all_loss: 0.3384 - val_lambda_3_loss: 0.3659\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:03<00:00,  6.76it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 6s 7us/sample - loss: 0.9713 - dense_9_loss: 0.6607 - all_loss: 0.1477 - lambda_5_loss: 0.1629 - val_loss: 0.9166 - val_dense_9_loss: 0.6058 - val_all_loss: 0.1478 - val_lambda_5_loss: 0.1631\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 5s 5us/sample - loss: 0.9210 - dense_9_loss: 0.6105 - all_loss: 0.1477 - lambda_5_loss: 0.1629 - val_loss: 0.9153 - val_dense_9_loss: 0.6045 - val_all_loss: 0.1478 - val_lambda_5_loss: 0.1631\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 5s 5us/sample - loss: 0.9209 - dense_9_loss: 0.6103 - all_loss: 0.1477 - lambda_5_loss: 0.1629 - val_loss: 0.9165 - val_dense_9_loss: 0.6056 - val_all_loss: 0.1478 - val_lambda_5_loss: 0.1631\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 5s 5us/sample - loss: 0.9209 - dense_9_loss: 0.6103 - all_loss: 0.1477 - lambda_5_loss: 0.1629 - val_loss: 0.9185 - val_dense_9_loss: 0.6076 - val_all_loss: 0.1478 - val_lambda_5_loss: 0.1631\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 5s 5us/sample - loss: 0.9206 - dense_9_loss: 0.6100 - all_loss: 0.1477 - lambda_5_loss: 0.1629 - val_loss: 0.9146 - val_dense_9_loss: 0.6038 - val_all_loss: 0.1478 - val_lambda_5_loss: 0.1631\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:01<00:00, 11.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 10s 11us/sample - loss: 0.4644 - dense_12_loss: 0.3639 - all_loss: 0.0290 - lambda_7_loss: 0.0715 - val_loss: 0.3555 - val_dense_12_loss: 0.2547 - val_all_loss: 0.0290 - val_lambda_7_loss: 0.0718\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 9s 10us/sample - loss: 0.3257 - dense_12_loss: 0.2251 - all_loss: 0.0290 - lambda_7_loss: 0.0715 - val_loss: 0.3193 - val_dense_12_loss: 0.2185 - val_all_loss: 0.0290 - val_lambda_7_loss: 0.0718\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 10s 11us/sample - loss: 0.3002 - dense_12_loss: 0.1997 - all_loss: 0.0290 - lambda_7_loss: 0.0715 - val_loss: 0.2864 - val_dense_12_loss: 0.1857 - val_all_loss: 0.0290 - val_lambda_7_loss: 0.0718\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.2852 - dense_12_loss: 0.1846 - all_loss: 0.0290 - lambda_7_loss: 0.0715 - val_loss: 0.2825 - val_dense_12_loss: 0.1817 - val_all_loss: 0.0290 - val_lambda_7_loss: 0.0718\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 11s 12us/sample - loss: 0.2748 - dense_12_loss: 0.1743 - all_loss: 0.0290 - lambda_7_loss: 0.0715 - val_loss: 0.2731 - val_dense_12_loss: 0.1723 - val_all_loss: 0.0290 - val_lambda_7_loss: 0.0718\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:02<00:00,  8.34it/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,
   "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
}
