{
 "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_[10, 10, 10, 70]gt'\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 21:36:32.832748: 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 21:36:32.853940: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-09-29 21:36:32.854573: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x555e4f73c790 executing computations on platform Host. Devices:\n",
      "2021-09-29 21:36:32.854599: 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 [==============================] - 11s 12us/sample - loss: 0.4129 - dense_2_loss: 0.3762 - all_loss: 0.0080 - lambda_1_loss: 0.0286 - val_loss: 0.2734 - val_dense_2_loss: 0.2365 - val_all_loss: 0.0081 - val_lambda_1_loss: 0.0288\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 12s 13us/sample - loss: 0.2280 - dense_2_loss: 0.1913 - all_loss: 0.0080 - lambda_1_loss: 0.0286 - val_loss: 0.2159 - val_dense_2_loss: 0.1790 - val_all_loss: 0.0081 - val_lambda_1_loss: 0.0288\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 23s 26us/sample - loss: 0.1878 - dense_2_loss: 0.1512 - all_loss: 0.0080 - lambda_1_loss: 0.0286 - val_loss: 0.1711 - val_dense_2_loss: 0.1343 - val_all_loss: 0.0081 - val_lambda_1_loss: 0.0288\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 16s 18us/sample - loss: 0.1705 - dense_2_loss: 0.1339 - all_loss: 0.0080 - lambda_1_loss: 0.0286 - val_loss: 0.1705 - val_dense_2_loss: 0.1337 - val_all_loss: 0.0081 - val_lambda_1_loss: 0.0288\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 15s 17us/sample - loss: 0.1601 - dense_2_loss: 0.1235 - all_loss: 0.0080 - lambda_1_loss: 0.0286 - val_loss: 0.1498 - val_dense_2_loss: 0.1129 - val_all_loss: 0.0081 - val_lambda_1_loss: 0.0288\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:04<00:00,  5.48it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 15s 16us/sample - loss: 1.0711 - dense_7_loss: 0.2448 - all_loss: 0.4067 - lambda_3_loss: 0.4196 - val_loss: 1.0161 - val_dense_7_loss: 0.1906 - val_all_loss: 0.4063 - val_lambda_3_loss: 0.4192\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 13s 15us/sample - loss: 0.9822 - dense_7_loss: 0.1559 - all_loss: 0.4067 - lambda_3_loss: 0.4196 - val_loss: 0.9645 - val_dense_7_loss: 0.1390 - val_all_loss: 0.4063 - val_lambda_3_loss: 0.4192\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 13s 15us/sample - loss: 0.9568 - dense_7_loss: 0.1305 - all_loss: 0.4067 - lambda_3_loss: 0.4196 - val_loss: 0.9473 - val_dense_7_loss: 0.1218 - val_all_loss: 0.4063 - val_lambda_3_loss: 0.4192\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 13s 15us/sample - loss: 0.9442 - dense_7_loss: 0.1179 - all_loss: 0.4067 - lambda_3_loss: 0.4196 - val_loss: 0.9400 - val_dense_7_loss: 0.1145 - val_all_loss: 0.4063 - val_lambda_3_loss: 0.4192\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 13s 15us/sample - loss: 0.9360 - dense_7_loss: 0.1096 - all_loss: 0.4067 - lambda_3_loss: 0.4196 - val_loss: 0.9337 - val_dense_7_loss: 0.1082 - val_all_loss: 0.4063 - val_lambda_3_loss: 0.4192\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:03<00:00,  5.63it/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: 1.1806 - dense_9_loss: 0.7728 - all_loss: 0.2029 - lambda_5_loss: 0.2049 - val_loss: 1.1629 - val_dense_9_loss: 0.7553 - val_all_loss: 0.2028 - val_lambda_5_loss: 0.2048\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 7s 8us/sample - loss: 1.1642 - dense_9_loss: 0.7565 - all_loss: 0.2029 - lambda_5_loss: 0.2049 - val_loss: 1.1630 - val_dense_9_loss: 0.7554 - val_all_loss: 0.2028 - val_lambda_5_loss: 0.2048\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 7s 8us/sample - loss: 1.1640 - dense_9_loss: 0.7562 - all_loss: 0.2029 - lambda_5_loss: 0.2049 - val_loss: 1.1641 - val_dense_9_loss: 0.7565 - val_all_loss: 0.2028 - val_lambda_5_loss: 0.2048\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 7s 7us/sample - loss: 1.1635 - dense_9_loss: 0.7558 - all_loss: 0.2029 - lambda_5_loss: 0.2049 - val_loss: 1.1621 - val_dense_9_loss: 0.7545 - val_all_loss: 0.2028 - val_lambda_5_loss: 0.2048\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 7s 7us/sample - loss: 1.1636 - dense_9_loss: 0.7558 - all_loss: 0.2029 - lambda_5_loss: 0.2049 - val_loss: 1.1617 - val_dense_9_loss: 0.7542 - val_all_loss: 0.2028 - val_lambda_5_loss: 0.2048\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:02<00:00,  8.96it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 900000 samples, validate on 100000 samples\n",
      "Epoch 1/5\n",
      "900000/900000 [==============================] - 16s 18us/sample - loss: 0.5610 - dense_12_loss: 0.4073 - all_loss: 0.0728 - lambda_7_loss: 0.0809 - val_loss: 0.4423 - val_dense_12_loss: 0.2880 - val_all_loss: 0.0731 - val_lambda_7_loss: 0.0813\n",
      "Epoch 2/5\n",
      "900000/900000 [==============================] - 15s 17us/sample - loss: 0.3901 - dense_12_loss: 0.2364 - all_loss: 0.0728 - lambda_7_loss: 0.0809 - val_loss: 0.3685 - val_dense_12_loss: 0.2142 - val_all_loss: 0.0731 - val_lambda_7_loss: 0.0813\n",
      "Epoch 3/5\n",
      "900000/900000 [==============================] - 16s 18us/sample - loss: 0.3542 - dense_12_loss: 0.2005 - all_loss: 0.0728 - lambda_7_loss: 0.0809 - val_loss: 0.3371 - val_dense_12_loss: 0.1827 - val_all_loss: 0.0731 - val_lambda_7_loss: 0.0813\n",
      "Epoch 4/5\n",
      "900000/900000 [==============================] - 15s 17us/sample - loss: 0.3328 - dense_12_loss: 0.1791 - all_loss: 0.0728 - lambda_7_loss: 0.0809 - val_loss: 0.3169 - val_dense_12_loss: 0.1626 - val_all_loss: 0.0731 - val_lambda_7_loss: 0.0813\n",
      "Epoch 5/5\n",
      "900000/900000 [==============================] - 15s 17us/sample - loss: 0.3164 - dense_12_loss: 0.1627 - all_loss: 0.0728 - lambda_7_loss: 0.0809 - val_loss: 0.3065 - val_dense_12_loss: 0.1522 - val_all_loss: 0.0731 - val_lambda_7_loss: 0.0813\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:04<00:00,  5.21it/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
}
