{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "\n",
    "os.chdir('..')\n",
    "\n",
    "import tqdm\n",
    "\n",
    "import numpy as np\n",
    "import tensorflow as tf\n",
    "\n",
    "# Get rid of the deprecation warnings\n",
    "tf.compat.v1.logging.set_verbosity(tf.compat.v1.logging.ERROR)\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib as mpl\n",
    "\n",
    "#from tensorflow.python.lib.io import tf_record\n",
    "#from tensorflow.core.util import event_pb2\n",
    "#from tensorflow.python.framework import tensor_util\n",
    "\n",
    "from models.models import DeepStatisticalSolver"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This notebook considers the linear systems derived from the Poisson equation.\n",
    "\n",
    "\n",
    "The current notebook is composed of the following parts:\n",
    "- Reload a trained model\n",
    "- Compute the important metrics (same as the ones displayed in the paper)\n",
    "- Visualize the evolution of the loss, latent variables and actual prediction\n",
    "- Generalize to datasets of varying size (change the amount of nodes)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Reloading a model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Enter path to model. Three pre-trained models are available : best_linear_0, best_linear_1, best_linear_2\n",
    "model_path = \"results/best_linear_0\"\n",
    "\n",
    "# Enter path to data to build architecture\n",
    "data_dir = 'datasets/linear_systems'\n",
    "\n",
    "# Initialize a tensorflow session\n",
    "sess = tf.Session()\n",
    "\n",
    "# Build the Deep Statistical Solver\n",
    "model = DeepStatisticalSolver(sess, \n",
    "                          model_to_restore=model_path, \n",
    "                          default_data_directory=data_dir)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load the test set (change the mode variable if you want val or train)\n",
    "mode = 'test'\n",
    "\n",
    "# Import numpy data\n",
    "A_np = np.load(os.path.join(data_dir, 'A_'+mode+'.npy'))\n",
    "B_np = np.load(os.path.join(data_dir, 'B_'+mode+'.npy'))\n",
    "U_np = np.load(os.path.join(data_dir, 'U_'+mode+'.npy'))\n",
    "coord_np = np.load(os.path.join(data_dir, 'coord_'+mode+'.npy'))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Computing metrics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# For the whole dataset, we will compute both the individual loss of each sample, \n",
    "# and the final predictions\n",
    "individual_losses_DSS_ = None\n",
    "individual_losses_LU_ = None\n",
    "U_DSS_ = None\n",
    "U_LU_ = None\n",
    "\n",
    "# In order to split the dataset, define the size of the batches that will be fed to the model.\n",
    "# For very large dataset, it can be useful to have a small value for BATCH_SIZE\n",
    "BATCH_SIZE = 100\n",
    "\n",
    "# Get total amount of samples and split the dataset\n",
    "n_samples_tot = A_np.shape[0]\n",
    "n_batches = n_samples_tot // BATCH_SIZE\n",
    "batched_indices = np.array_split(np.arange(n_samples_tot), n_batches) \n",
    "\n",
    "# Iterate over the batches\n",
    "for indices in tqdm.tqdm(batched_indices):\n",
    "    \n",
    "    # Compute individual losses and final predictions for the DSS\n",
    "    feed_dict = {model.A:A_np[indices], model.B:B_np[indices]}\n",
    "    U_DSS, individual_losses_DSS = sess.run([model.U_final, \n",
    "                                         model.cost_per_sample[str(model.correction_updates)]], \n",
    "                                         feed_dict = feed_dict\n",
    "                                        )\n",
    "    \n",
    "    feed_dict = {model.A:A_np[indices], model.B:B_np[indices], model.U_final:U_np[indices]}\n",
    "    individual_losses_LU = sess.run(model.cost_per_sample[str(model.correction_updates)], \n",
    "                                         feed_dict = feed_dict\n",
    "                                        )\n",
    "        \n",
    "    # Getting prediction\n",
    "    U_DSS = np.reshape(U_DSS[:, :, 0], -1)\n",
    "\n",
    "    # Getting LU solution\n",
    "    U_LU = np.reshape(U_np[indices, :, 0], -1)\n",
    "\n",
    "    if individual_losses_DSS_ is None:\n",
    "        individual_losses_DSS_ = individual_losses_DSS\n",
    "        individual_losses_LU_ = individual_losses_LU\n",
    "        U_DSS_ = U_DSS\n",
    "        U_LU_ = U_LU\n",
    "        \n",
    "    else:\n",
    "        individual_losses_DSS_ = np.concatenate([individual_losses_DSS_, individual_losses_DSS])\n",
    "        individual_losses_LU_ = np.concatenate([individual_losses_LU_, individual_losses_LU])\n",
    "        U_DSS_ = np.concatenate([U_DSS_, U_DSS])\n",
    "        U_LU_ = np.concatenate([U_LU_, U_LU])\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print('Losses - DSS')\n",
    "print('    10th percentile = {}'.format(np.percentile(individual_losses_DSS_, 10)))\n",
    "print('    50th percentile = {}'.format(np.percentile(individual_losses_DSS_, 50)))\n",
    "print('    90th percentile = {}'.format(np.percentile(individual_losses_DSS_, 90)))\n",
    "print('Losses - LU')\n",
    "print('(Keep in mind that the maximum precision achieved by tf.float32 is around 1e-14')\n",
    "print('Thus the loss computed by tensorflow is very noisy w.r.t. LU method)')\n",
    "print('    10th percentile = {}'.format(np.percentile(individual_losses_LU_, 10)))\n",
    "print('    50th percentile = {}'.format(np.percentile(individual_losses_LU_, 50)))\n",
    "print('    90th percentile = {}'.format(np.percentile(individual_losses_LU_, 90)))\n",
    "print('Correlation between methods DSS and LU')\n",
    "print('    Correlation = {}'.format(np.corrcoef(U_LU_,U_DSS_)[1,0]))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We observe a correlation of 99.99% between our prediction and the LU method!"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Visualize loss, H and U"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Let's pick a random sample\n",
    "sample = np.random.randint(0,A_np.shape[0])\n",
    "\n",
    "# Predict and gather latent variables, intermediate predictions and the loss\n",
    "feed_dict = {model.A:A_np[sample:sample+1], model.B:B_np[sample:sample+1]}\n",
    "H_list, U_list, loss_list = sess.run([model.H, model.U, model.loss], feed_dict = feed_dict)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Convert those dictionaries into actual np array\n",
    "U = np.array([u[0,:,0] for u in U_list.values()])\n",
    "H = np.array([h[0,:,:] for h in H_list.values()])\n",
    "loss = np.array([l for l in loss_list.values()])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Loss visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plotting the loss\n",
    "plt.plot(np.arange(1,loss.shape[0]+1),loss)\n",
    "plt.scatter(np.arange(1,loss.shape[0]+1), loss, marker='+')\n",
    "plt.yscale('log')\n",
    "plt.ylabel('Intermediate losses')\n",
    "plt.xlabel(r'update $k$')\n",
    "plt.title('Evolution of the loss across the Deep Statistical Solver')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The loss decreases by 5 orders of magnitude between the first prediction and the final"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Structure of the graph"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "edges_or = A_np[sample, :, 0].astype(np.int32)\n",
    "edges_ex = A_np[sample, :, 1].astype(np.int32)\n",
    "\n",
    "# Creating mesh coordinates\n",
    "coordinates = coord_np[sample]\n",
    "coordinates_or_x = coordinates[edges_or, 0]\n",
    "coordinates_or_y = coordinates[edges_or, 1]\n",
    "coordinates_ex_x = coordinates[edges_ex, 0]\n",
    "coordinates_ex_y = coordinates[edges_ex, 1]\n",
    "\n",
    "coordinates_edge_x = np.c_[coordinates_or_x, coordinates_ex_x]\n",
    "coordinates_edge_y = np.c_[coordinates_or_y, coordinates_ex_y]\n",
    "\n",
    "plt.figure(figsize=[5,5])\n",
    "plt.tick_params(\n",
    "    axis='x',          # changes apply to the x-axis\n",
    "    which='both',      # both major and minor ticks are affected\n",
    "    bottom=False,      # ticks along the bottom edge are off\n",
    "    top=False,         # ticks along the top edge are off\n",
    "    labelbottom=False) # labels along the bottom edge are off\n",
    "plt.tick_params(\n",
    "    axis='y',          # changes apply to the x-axis\n",
    "    which='both',      # both major and minor ticks are affected\n",
    "    bottom=False,      # ticks along the bottom edge are off\n",
    "    top=False,         # ticks along the top edge are off\n",
    "    left=False,\n",
    "    right=False,\n",
    "    labelleft=False) # labels along the bottom edge are off\n",
    "\n",
    "plt.plot(coordinates_edge_x.T, coordinates_edge_y.T, 'k', zorder=1)\n",
    "plt.xlim([1,2])\n",
    "plt.ylim([1,2])\n",
    "plt.title(r'Mesh $\\tilde{\\mathbb{G}}$')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This is the undirected and unweighted graph that bears the structure of the problem instance.\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Intermediate predictions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "max_x = np.max(U)\n",
    "min_x = np.min(U)\n",
    "\n",
    "for frame in range(U.shape[0]):\n",
    "    plt.figure(figsize=[3,2.5], dpi=300)\n",
    "    \n",
    "    plt.tick_params(\n",
    "        axis='x',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        labelbottom=False) # labels along the bottom edge are off\n",
    "    plt.tick_params(\n",
    "        axis='y',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        left=False,\n",
    "        right=False,\n",
    "        labelleft=False) # labels along the bottom edge are off\n",
    "    \n",
    "    plt.scatter(coordinates[:,0], coordinates[:,1], c=U[frame,:], zorder=2)\n",
    "    plt.clim(min_x, max_x)\n",
    "    plt.xlim([0.9,2.1])\n",
    "    plt.ylim([0.9,2.1])\n",
    "    plt.colorbar()\n",
    "    plt.title(r'Intermediate prediction $\\widehat{\\mathbf{U}}^{'+str(frame+1)+'}$')\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "One can see that the information propagates from the boundaries to the inside of the shape"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Latent variables evolution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "# Select which of the d components of H you want to visualize\n",
    "component_of_H = 2\n",
    "\n",
    "max_h = np.max(H[:,:,component_of_H])\n",
    "min_h = np.min(H[:,:,component_of_H])\n",
    "\n",
    "for frame in range(H.shape[0]):\n",
    "    plt.figure(figsize=[3,2.5], dpi=300)\n",
    "    \n",
    "    plt.tick_params(\n",
    "        axis='x',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        labelbottom=False) # labels along the bottom edge are off\n",
    "    plt.tick_params(\n",
    "        axis='y',          # changes apply to the x-axis\n",
    "        which='both',      # both major and minor ticks are affected\n",
    "        bottom=False,      # ticks along the bottom edge are off\n",
    "        top=False,         # ticks along the top edge are off\n",
    "        left=False,\n",
    "        right=False,\n",
    "        labelleft=False) # labels along the bottom edge are off\n",
    "    \n",
    "    plt.scatter(coordinates[:,0], coordinates[:,1], c=H[frame,:,component_of_H], zorder=2)\n",
    "    plt.clim(min_h, max_h)\n",
    "    plt.xlim([0.9,2.1])\n",
    "    plt.ylim([0.9,2.1])\n",
    "    plt.colorbar()\n",
    "    plt.title(r'$\\mathbf{H}^{'+str(frame+1)+'} - component {'+str(component_of_H)+'}$')\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Change in distribution"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "As explained in the paper, we train our Deep Statistical Solver on a certain distribution of problems. This part aims at looking at what happens when the test distribution is different from the training distribution "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Varying number of nodes"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "In this first experiment, we keep pretty much the same data generation process, except for the number of node output by the discretization process. We have thus generated some other datasets"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sizes = [100, 250, 500, 750, 1000]\n",
    "n_path = 'datasets/linear_systems/varying_size'\n",
    "\n",
    "A_modified = {}\n",
    "B_modified = {}\n",
    "U_modified = {}\n",
    "U_modified_pred = {}\n",
    "corr = {}\n",
    "\n",
    "for size in sizes:\n",
    "    A_modified[size] = np.load(n_path+'/n_{}/A_test.npy'.format(size))#, allow_pickle=True)\n",
    "    B_modified[size] = np.load(n_path+'/n_{}/B_test.npy'.format(size), allow_pickle=True)\n",
    "    U_modified[size] = np.load(n_path+'/n_{}/U_test.npy'.format(size), allow_pickle=True)\n",
    "    feed_dict={model.A:A_modified[size], model.B:B_modified[size]}\n",
    "    U_modified_pred[size] = sess.run(model.U_final, feed_dict=feed_dict)\n",
    "    \n",
    "    U_modified_flat = np.reshape(U_modified[size], -1)\n",
    "    U_modified_pred_flat = np.reshape(U_modified_pred[size], -1)\n",
    "    U_modified_pred_flat = U_modified_pred_flat * (U_modified_flat != 0.)\n",
    "    corr[size] = np.corrcoef(U_modified_flat, U_modified_pred_flat)[0, 1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure(figsize=[2, 2], dpi=300)\n",
    "for size in sizes:\n",
    "    plt.scatter(size, 1-corr[size], color='blue', marker='+')\n",
    "plt.yscale('log')\n",
    "plt.xlabel(r'$<n>$')\n",
    "plt.ylabel(r'$1 - Corr$')\n",
    "\n",
    "#plt.xlim([10**(-3.5), 10**(1.5)])\n",
    "plt.ylim([10**(-6), 10**(-1)])\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "In the test set there are samples with various number of nodes. \n",
    "\n",
    "For samples that have fewer nodes than the amount observed in the train set, we observe an even higher correlation between our method and the LU method.\n",
    "\n",
    "For samples that have more nodes than in the train set, the correlation is slightly worse, although it remains higher than 99%."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Adding noise to the initial test set"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now, we conduct a more extreme experiment, where we gradually increase a noise applied to A and B"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Since our modelling for A and B is not so simple and straightforward, we need the following methods\n",
    "# to convert A and B back to the classical representation, and to solve it\n",
    "\n",
    "def build_A_dense(A_np, B_np):\n",
    "\n",
    "    n_nodes = np.max(A_np[:, 0:2]).astype(np.int32)+1\n",
    "    n_nodes = np.argmin((B_np[:,0]**2 + B_np[:,1]**2 + B_np[:,2]**2) != 0.)\n",
    "    if n_nodes == 0:\n",
    "        n_nodes = B_np[:,0].shape[0]\n",
    "    \n",
    "    A_dense = np.zeros([n_nodes, n_nodes])\n",
    "        \n",
    "    from_index = A_np[:, 0].astype(np.int32)\n",
    "    to_index = A_np[:, 1].astype(np.int32)\n",
    "    A_val = A_np[:, 2]\n",
    "        \n",
    "    A_dense[from_index, to_index] = A_val\n",
    "    A_dense = A_dense - np.diag(np.sum(A_dense, axis=1))\n",
    "        \n",
    "    # Detect constrained nodes\n",
    "    constrained_indices = (B_np[:n_nodes, 1] == 1.)\n",
    "    A_dense[constrained_indices, :] = 0.\n",
    "    A_dense[constrained_indices, constrained_indices] = 1.0\n",
    "    \n",
    "    \n",
    "    B_dense = B_np[:n_nodes, 0] + B_np[:n_nodes, 2]\n",
    "    \n",
    "    return A_dense, B_dense\n",
    "\n",
    "def compute_LU_solution(A_dense, B_dense):\n",
    "    from scipy.linalg import lu_factor, lu_solve\n",
    "    \n",
    "    lu, piv = lu_factor(A_dense)\n",
    "    U = lu_solve((lu, piv), B_dense)\n",
    "    \n",
    "    return U"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_samples = 100\n",
    "\n",
    "noise_levels = 10.**np.arange(-3,2,1.)\n",
    "A_np_noisy = {}\n",
    "A_np_noisy_inv = {}\n",
    "B_np_noisy = {}\n",
    "correlations = {}\n",
    "\n",
    "predictions = {}\n",
    "solutions = {}\n",
    "\n",
    "for noise in noise_levels:\n",
    "    \n",
    "    # Randomly sample from the available samples\n",
    "    indices = np.random.randint(0, A_np.shape[0], n_samples)\n",
    "    \n",
    "    predictions[noise] = None\n",
    "    solutions[noise] = None\n",
    "    \n",
    "    correlations[noise] = []\n",
    "    \n",
    "    noise_A = np.random.lognormal(0, noise, A_np.shape)\n",
    "    noise_A = np.array([[[0., 0., 1.]]]) * noise_A * A_np\n",
    "    cst_A = np.array([[[1., 1., 0.]]]) * A_np\n",
    "    \n",
    "    A_np_noisy[noise] = cst_A + noise_A\n",
    "    \n",
    "    \n",
    "    noise_B = np.random.normal(1, noise, B_np.shape)\n",
    "    noise_B = np.array([[[1.,0.,1.]]]) * noise_B * B_np\n",
    "    cst_B = np.array([[[0.,1.,0.]]]) * B_np\n",
    "    \n",
    "    B_np_noisy[noise] = cst_B + noise_B\n",
    "    \n",
    "    for sample in tqdm.tqdm(indices):\n",
    "\n",
    "        A_dense, B_dense = build_A_dense(A_np_noisy[noise][sample], B_np_noisy[noise][sample])\n",
    "        # Computing solution\n",
    "        try:\n",
    "            U_dense = compute_LU_solution(A_dense, B_dense)\n",
    "        except:\n",
    "            continue\n",
    "        \n",
    "        U_pred = sess.run(model.U_final, feed_dict={model.A:A_np_noisy[noise][sample:sample+1],\n",
    "                                                    model.B:B_np_noisy[noise][sample:sample+1]\n",
    "                                                   })\n",
    "        \n",
    "        \n",
    "        if predictions[noise] is None:\n",
    "            predictions[noise] = U_pred[:,:,0]\n",
    "            solutions[noise] = np.zeros(U_pred.shape[1])\n",
    "            solutions[noise][:U_dense.shape[0]] = U_dense\n",
    "            solutions[noise] = np.expand_dims(solutions[noise], axis=0)\n",
    "            \n",
    "        else :\n",
    "            predictions[noise] = np.concatenate([predictions[noise], U_pred[:,:,0]], axis=0)\n",
    "            solution_sample = np.zeros(U_pred.shape[1])\n",
    "            solution_sample[:U_dense.shape[0]] = U_dense\n",
    "            solution_sample = np.expand_dims(solution_sample, axis=0)\n",
    "            solutions[noise] = np.concatenate([solutions[noise], solution_sample], axis=0)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure(figsize=[2, 2], dpi=300)\n",
    "for noise in noise_levels:\n",
    "    sol = np.reshape(solutions[noise], -1)\n",
    "    pred = np.reshape(predictions[noise], -1)\n",
    "    corr = np.corrcoef(sol, pred)[0,1]\n",
    "    plt.scatter(noise, 1-corr, color='blue', marker='+')\n",
    "plt.xscale('log')\n",
    "plt.yscale('log')\n",
    "plt.xlabel(r'$\\tau$')\n",
    "plt.ylabel(r'$1 - Corr$')\n",
    "\n",
    "plt.xlim([10**(-3.5), 10**(1.5)])\n",
    "plt.ylim([10**(-5), 10**(1)])\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "For small amplitudes of noise, the correlation remains excellent (around 99.99%), but when the noise parameter tau increases, we observe that our predictions no longer match the ones of the LU method. This experiment clearly shows that we have learned to solve linear systems for a certain distribution of problems, and diverging too much from it can cause a major decrease in accuracy."
   ]
  },
  {
   "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.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
