{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:526: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_qint8 = np.dtype([(\"qint8\", np.int8, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:527: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_quint8 = np.dtype([(\"quint8\", np.uint8, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:528: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_qint16 = np.dtype([(\"qint16\", np.int16, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:529: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_quint16 = np.dtype([(\"quint16\", np.uint16, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:530: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  _np_qint32 = np.dtype([(\"qint32\", np.int32, 1)])\n",
      "/home/zulqarnain/anaconda3/envs/old_tf/lib/python3.7/site-packages/tensorflow/python/framework/dtypes.py:535: FutureWarning: Passing (type, 1) or '1type' as a synonym of type is deprecated; in a future version of numpy, it will be understood as (type, (1,)) / '(1,)type'.\n",
      "  np_resource = np.dtype([(\"resource\", np.ubyte, 1)])\n"
     ]
    }
   ],
   "source": [
    "import pickle\n",
    "import numpy as np\n",
    "import argparse\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "import pandas as pd\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.model_selection import train_test_split\n",
    "from scipy.spatial.distance import pdist\n",
    "\n",
    "import cxplain\n",
    "from cxplain import MLPModelBuilder, ZeroMasking, CXPlain\n",
    "from tensorflow.python.keras.losses import binary_crossentropy, categorical_crossentropy\n",
    "\n",
    "from tensorflow.python.keras.layers import Dense, Input, Flatten, Add, Multiply, Lambda\n",
    "from tensorflow.python.keras.layers.normalization import BatchNormalization\n",
    "from tensorflow.python.keras import regularizers\n",
    "from tensorflow.python.keras.models import Model, Sequential\n",
    "from tensorflow.python.keras import optimizers\n",
    "from tensorflow.python.keras.callbacks import ModelCheckpoint\n",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "from tqdm import tqdm\n",
    "\n",
    "from utils.explanations import calculate_robust_astute_sampled\n",
    "import shap\n",
    "\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "datatype = 'telescope'\n",
    "run_times = 1\n",
    "prop_points = 1\n",
    "calculate = True\n",
    "epsilon_range = np.arange(0.01, 1.1, 0.05)\n",
    "masking_operation = ZeroMasking()\n",
    "loss = binary_crossentropy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_dict = pd.read_csv('data/magic04.data').values\n",
    "data = data_dict[:, :-1]\n",
    "labels = data_dict[:, -1]\n",
    "labels[labels == 'h'] = 0\n",
    "labels[labels == 'g'] = 1\n",
    "x_train, x_val, y_train, y_val = train_test_split(data, labels, test_size=0.05, stratify=labels, random_state=42)\n",
    "x_train = StandardScaler().fit_transform(x_train)\n",
    "x_val = StandardScaler().fit_transform(x_val)\n",
    "input_shape = x_train.shape[-1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "median_rad = np.median(pdist(x_train))"
   ]
  },
  {
   "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-10-01 12:06:19.521995: I tensorflow/core/platform/cpu_feature_guard.cc:141] Your CPU supports instructions that this TensorFlow binary was not compiled to use: AVX2 FMA\n",
      "2021-10-01 12:06:19.545968: I tensorflow/core/platform/profile_utils/cpu_utils.cc:94] CPU Frequency: 3600000000 Hz\n",
      "2021-10-01 12:06:19.546531: I tensorflow/compiler/xla/service/service.cc:150] XLA service 0x5628d088ecc0 executing computations on platform Host. Devices:\n",
      "2021-10-01 12:06:19.546555: 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 16261 samples, validate on 1807 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",
      "16261/16261 [==============================] - 1s 65us/sample - loss: 0.9599 - dense_1_loss: 0.7393 - all_loss: 0.0975 - lambda_1_loss: 0.1223 - val_loss: 0.7932 - val_dense_1_loss: 0.5812 - val_all_loss: 0.0970 - val_lambda_1_loss: 0.1208\n",
      "Epoch 2/5\n",
      "16261/16261 [==============================] - 1s 49us/sample - loss: 0.7669 - dense_1_loss: 0.5466 - all_loss: 0.0975 - lambda_1_loss: 0.1222 - val_loss: 0.7357 - val_dense_1_loss: 0.5223 - val_all_loss: 0.0964 - val_lambda_1_loss: 0.1202\n",
      "Epoch 3/5\n",
      "16261/16261 [==============================] - 1s 49us/sample - loss: 0.7253 - dense_1_loss: 0.5056 - all_loss: 0.0975 - lambda_1_loss: 0.1222 - val_loss: 0.7027 - val_dense_1_loss: 0.4858 - val_all_loss: 0.0951 - val_lambda_1_loss: 0.1196\n",
      "Epoch 4/5\n",
      "16261/16261 [==============================] - 1s 49us/sample - loss: 0.6991 - dense_1_loss: 0.4794 - all_loss: 0.0977 - lambda_1_loss: 0.1222 - val_loss: 0.6811 - val_dense_1_loss: 0.4658 - val_all_loss: 0.0960 - val_lambda_1_loss: 0.1201\n",
      "Epoch 5/5\n",
      "16261/16261 [==============================] - 1s 49us/sample - loss: 0.6817 - dense_1_loss: 0.4616 - all_loss: 0.0977 - lambda_1_loss: 0.1223 - val_loss: 0.6683 - val_dense_1_loss: 0.4525 - val_all_loss: 0.0958 - val_lambda_1_loss: 0.1196\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:22<00:00,  1.01s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 16261 samples, validate on 1807 samples\n",
      "Epoch 1/5\n",
      "16261/16261 [==============================] - 2s 94us/sample - loss: 0.9141 - dense_6_loss: 0.6991 - all_loss: 0.0918 - lambda_3_loss: 0.1234 - val_loss: 0.8196 - val_dense_6_loss: 0.6030 - val_all_loss: 0.0918 - val_lambda_3_loss: 0.1225\n",
      "Epoch 2/5\n",
      "16261/16261 [==============================] - 1s 58us/sample - loss: 0.7426 - dense_6_loss: 0.5269 - all_loss: 0.0917 - lambda_3_loss: 0.1232 - val_loss: 0.7011 - val_dense_6_loss: 0.4847 - val_all_loss: 0.0920 - val_lambda_3_loss: 0.1231\n",
      "Epoch 3/5\n",
      "16261/16261 [==============================] - 1s 59us/sample - loss: 0.6776 - dense_6_loss: 0.4621 - all_loss: 0.0918 - lambda_3_loss: 0.1231 - val_loss: 0.6687 - val_dense_6_loss: 0.4562 - val_all_loss: 0.0934 - val_lambda_3_loss: 0.1243\n",
      "Epoch 4/5\n",
      "16261/16261 [==============================] - 1s 59us/sample - loss: 0.6524 - dense_6_loss: 0.4372 - all_loss: 0.0919 - lambda_3_loss: 0.1233 - val_loss: 0.6630 - val_dense_6_loss: 0.4458 - val_all_loss: 0.0916 - val_lambda_3_loss: 0.1224\n",
      "Epoch 5/5\n",
      "16261/16261 [==============================] - 2s 94us/sample - loss: 0.6372 - dense_6_loss: 0.4215 - all_loss: 0.0917 - lambda_3_loss: 0.1231 - val_loss: 0.6432 - val_dense_6_loss: 0.4286 - val_all_loss: 0.0923 - val_lambda_3_loss: 0.1233\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:28<00:00,  1.30s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 16261 samples, validate on 1807 samples\n",
      "Epoch 1/5\n",
      "16261/16261 [==============================] - 2s 104us/sample - loss: 1.1303 - dense_8_loss: 0.8056 - all_loss: 0.1476 - lambda_5_loss: 0.1759 - val_loss: 0.8804 - val_dense_8_loss: 0.5640 - val_all_loss: 0.1458 - val_lambda_5_loss: 0.1705\n",
      "Epoch 2/5\n",
      "16261/16261 [==============================] - 1s 60us/sample - loss: 0.8731 - dense_8_loss: 0.5488 - all_loss: 0.1476 - lambda_5_loss: 0.1758 - val_loss: 0.8666 - val_dense_8_loss: 0.5507 - val_all_loss: 0.1451 - val_lambda_5_loss: 0.1704\n",
      "Epoch 3/5\n",
      "16261/16261 [==============================] - 1s 56us/sample - loss: 0.8663 - dense_8_loss: 0.5422 - all_loss: 0.1477 - lambda_5_loss: 0.1759 - val_loss: 0.8621 - val_dense_8_loss: 0.5462 - val_all_loss: 0.1450 - val_lambda_5_loss: 0.1696\n",
      "Epoch 4/5\n",
      "16261/16261 [==============================] - 1s 56us/sample - loss: 0.8652 - dense_8_loss: 0.5412 - all_loss: 0.1475 - lambda_5_loss: 0.1759 - val_loss: 0.8635 - val_dense_8_loss: 0.5520 - val_all_loss: 0.1456 - val_lambda_5_loss: 0.1704\n",
      "Epoch 5/5\n",
      "16261/16261 [==============================] - 1s 65us/sample - loss: 0.8646 - dense_8_loss: 0.5407 - all_loss: 0.1480 - lambda_5_loss: 0.1763 - val_loss: 0.8624 - val_dense_8_loss: 0.5450 - val_all_loss: 0.1446 - val_lambda_5_loss: 0.1691\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:22<00:00,  1.04s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train on 16261 samples, validate on 1807 samples\n",
      "Epoch 1/5\n",
      "16261/16261 [==============================] - 2s 107us/sample - loss: 2.4078 - dense_11_loss: 1.0359 - all_loss: 0.7140 - lambda_7_loss: 0.6578 - val_loss: 2.3277 - val_dense_11_loss: 0.9739 - val_all_loss: 0.7079 - val_lambda_7_loss: 0.6460\n",
      "Epoch 2/5\n",
      "16261/16261 [==============================] - 1s 57us/sample - loss: 2.3663 - dense_11_loss: 0.9950 - all_loss: 0.7137 - lambda_7_loss: 0.6578 - val_loss: 2.3319 - val_dense_11_loss: 0.9763 - val_all_loss: 0.7087 - val_lambda_7_loss: 0.6458\n",
      "Epoch 3/5\n",
      "16261/16261 [==============================] - 1s 61us/sample - loss: 2.3649 - dense_11_loss: 0.9932 - all_loss: 0.7141 - lambda_7_loss: 0.6578 - val_loss: 2.3290 - val_dense_11_loss: 0.9730 - val_all_loss: 0.7076 - val_lambda_7_loss: 0.6449\n",
      "Epoch 4/5\n",
      "16261/16261 [==============================] - 1s 59us/sample - loss: 2.3650 - dense_11_loss: 0.9928 - all_loss: 0.7140 - lambda_7_loss: 0.6579 - val_loss: 2.3304 - val_dense_11_loss: 0.9746 - val_all_loss: 0.7093 - val_lambda_7_loss: 0.6460\n",
      "Epoch 5/5\n",
      "16261/16261 [==============================] - 1s 61us/sample - loss: 2.3647 - dense_11_loss: 0.9927 - all_loss: 0.7139 - lambda_7_loss: 0.6579 - val_loss: 2.3333 - val_dense_11_loss: 0.9781 - val_all_loss: 0.7086 - val_lambda_7_loss: 0.6464\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 22/22 [00:28<00:00,  1.27s/it]\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'\n",
    "\n",
    "                model_input = Input(shape=(input_shape,), dtype='float32')\n",
    "\n",
    "                net = Dense(32, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "\n",
    "                preds = Dense(1, activation='sigmoid', name='dense3',\n",
    "                              kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                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=1, num_units=32, activation=activation, verbose=1,\n",
    "                        batch_size=32, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                        with_bn=False)\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",
    "\n",
    "                net = Dense(32, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "                net = Dense(32, activation=activation, name='dense2',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = Dense(32, activation=activation, name='dense3',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                net = Dense(32, activation=activation, name='dense4',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                preds = Dense(1, activation='sigmoid', name='dense5',\n",
    "                              kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                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=32, activation=activation, verbose=1,\n",
    "                                batch_size=32, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                with_bn=False)\n",
    "\n",
    "\n",
    "            elif classifiers[j] == 'linear':\n",
    "                activation = None\n",
    "\n",
    "                model_input = Input(shape=(input_shape,), dtype='float32')\n",
    "\n",
    "                net = Dense(32, activation=activation, name='dense1',\n",
    "                            kernel_regularizer=regularizers.l2(1e-3))(model_input)\n",
    "\n",
    "                preds = Dense(1, activation='sigmoid', name='dense3',\n",
    "                              kernel_regularizer=regularizers.l2(1e-3))(net)\n",
    "                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=32, activation=activation, verbose=1,\n",
    "                                        batch_size=32, learning_rate=0.001, num_epochs=5, early_stopping_patience=15,\n",
    "                                        with_bn=False)\n",
    "            elif classifiers[j] == 'svm':\n",
    "                pred_model = pickle.load(open('models/' + datatype + '_svm.pk', 'rb'))\n",
    "                model_builder = MLPModelBuilder(num_layers=2, num_units=32, activation=activation, verbose=1,\n",
    "                                                batch_size=32, learning_rate=0.001, num_epochs=5, \n",
    "                                                early_stopping_patience=15,\n",
    "                                                with_bn=False)\n",
    "            if classifiers[j] == 'svm':\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",
    "                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)\n",
    "plt.show()\n",
    "plt.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(18068, 10)"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(951, 10)"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_val.shape"
   ]
  },
  {
   "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
}
