{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Graph Dynamical Systems\n",
    "\n",
    "This notebooks contains the experiments to evaluate graph edit networks on simple graph dynamical systems, namely the edit cycles, degree rules, and game of life datasets."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Hyperparameter setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "import numpy as np\n",
    "import torch\n",
    "import pytorch_graph_edit_networks as gen\n",
    "from torch_geometric.utils import dense_to_sparse\n",
    "import baseline_models\n",
    "import os\n",
    "import time\n",
    "import hep_th\n",
    "\n",
    "# model hyperparameters\n",
    "num_layers = 2\n",
    "dim_hid = 64\n",
    "\n",
    "# training hyperparameters\n",
    "learning_rate  = 1E-3\n",
    "weight_decay   = 1E-5\n",
    "loss_threshold = 1E-3\n",
    "max_epochs     = 50000\n",
    "print_step     = 1000\n",
    "\n",
    "R = 5        # number of repetitions for each experiment\n",
    "N_test = 10  # number of test time series we use to evaluate learning afterwards"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Model setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "pycharm": {
     "name": "#%%\n"
    }
   },
   "outputs": [],
   "source": [
    "# SETUP FUNCTIONS\n",
    "def setup_vgae(dim_in, nonlin):\n",
    "    return baseline_models.VGAE(num_layers=num_layers,\n",
    "                                dim_in=dim_in,\n",
    "                                dim_hid=dim_hid,\n",
    "                                beta=1E-3,\n",
    "                                sigma_scaling=1E-3,\n",
    "                                nonlin=nonlin)\n",
    "\n",
    "\n",
    "def setup_vgrnn(dim_in, nonlin):\n",
    "    return baseline_models.VGRNN(num_layers=num_layers,\n",
    "                                 dim_in=dim_in,\n",
    "                                 dim_hid=dim_hid)\n",
    "\n",
    "\n",
    "def setup_gen(dim_in, nonlin):\n",
    "    return gen.GEN(num_layers=num_layers,\n",
    "                   dim_in=dim_in,\n",
    "                   dim_hid=dim_hid,\n",
    "                   nonlin=nonlin)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "pycharm": {
     "name": "#%%\n"
    }
   },
   "outputs": [],
   "source": [
    "# LOSS FUNCTIONS\n",
    "loss_fun = gen.GEN_loss()\n",
    "crossent_loss_fun = gen.GEN_loss_crossent()\n",
    "def vgae_loss(model, A, X, delta, Epsilon, state=None):\n",
    "    B = A + Epsilon\n",
    "    # delete all outgoing and incoming edges of deleted nodes\n",
    "    B[delta < -0.5, :] = 0\n",
    "    B[:, delta < -0.5] = 0\n",
    "    loss = model.compute_loss(torch.tensor(A, dtype=torch.float),\n",
    "                              torch.tensor(B, dtype=torch.float),\n",
    "                              torch.tensor(X, dtype=torch.float))\n",
    "\n",
    "    return loss, state\n",
    "\n",
    "\n",
    "def vgrnn_loss(model, A, X, delta, Epsilon, state=None):\n",
    "    A = torch.tensor(A, dtype=torch.float)\n",
    "    edge_index = dense_to_sparse(A)\n",
    "\n",
    "    predicted = model(X, edge_index, hidden_in=state)\n",
    "    predicted, state = predicted[:-1], predicted[-1]\n",
    "\n",
    "    target = A + Epsilon\n",
    "    target[delta < -0.5, :] = 0\n",
    "    target[:, delta < -0.5] = 0\n",
    "\n",
    "    return model.compute_loss(*predicted, target), state\n",
    "\n",
    "\n",
    "def gen_loss_crossent(model, A, X, delta, Epsilon, state=None):\n",
    "    delta_pred, Epsilon_pred = model(torch.tensor(A, dtype=torch.float),\n",
    "                                     torch.tensor(X, dtype=torch.float))\n",
    "    loss = crossent_loss_fun(delta_pred, Epsilon_pred,\n",
    "                             torch.tensor(delta, dtype=torch.float),\n",
    "                             torch.tensor(Epsilon, dtype=torch.float),\n",
    "                             torch.tensor(A, dtype=torch.float))\n",
    "\n",
    "    return loss, state\n",
    "\n",
    "\n",
    "def gen_loss(model, A, X, delta, Epsilon, state=None):\n",
    "    delta_pred, Epsilon_pred = model(torch.tensor(A, dtype=torch.float),\n",
    "                                     torch.tensor(X, dtype=torch.float))\n",
    "    loss = loss_fun(delta_pred, Epsilon_pred,\n",
    "                    torch.tensor(delta, dtype=torch.float),\n",
    "                    torch.tensor(Epsilon, dtype=torch.float),\n",
    "                    torch.tensor(A, dtype=torch.float))\n",
    "\n",
    "    return loss, state"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "pycharm": {
     "name": "#%%\n"
    }
   },
   "outputs": [],
   "source": [
    "# PREDICTION FUNCTIONS\n",
    "def vgae_pred(model, A, X, state=None):\n",
    "    B = model(torch.tensor(A, dtype=torch.float), torch.tensor(X, dtype=torch.float))\n",
    "    B = B.detach().numpy()\n",
    "    Epsilon = B - A\n",
    "    delta = np.zeros(A.shape[0])\n",
    "    delta[np.sum(B, 1) < 0.5] = -1.\n",
    "    Epsilon[delta < -0.5, :] = 0.\n",
    "    Epsilon[:, delta < -0.5] = 0.\n",
    "    return delta, Epsilon, state\n",
    "\n",
    "\n",
    "def vgrnn_pred(model, A, X, state=None):\n",
    "    A = torch.tensor(A, dtype=torch.float)\n",
    "    edge_index = dense_to_sparse(A)\n",
    "\n",
    "    predicted = model(X, edge_index, hidden_in=state)\n",
    "    predicted, state = predicted[:-1], predicted[-1]\n",
    "\n",
    "    Epsilon = predicted - A\n",
    "    delta = np.zeros(A.shape[0])\n",
    "    delta[np.sum(predicted, 1) < 0.5] = -1.\n",
    "    Epsilon[delta < -0.5, :] = 0.\n",
    "    Epsilon[:, delta < -0.5] = 0.\n",
    "\n",
    "    return delta, Epsilon, state\n",
    "\n",
    "\n",
    "def gen_pred(model, A, X, state=None):\n",
    "    delta_pred, Epsilon_pred = model(torch.tensor(A, dtype=torch.float), torch.tensor(X, dtype=torch.float))\n",
    "    delta_pred = delta_pred.detach().numpy()\n",
    "    Epsilon_pred = Epsilon_pred.detach().numpy()\n",
    "    delta = np.zeros(A.shape[0])\n",
    "    delta[delta_pred > 0.5] = 1.\n",
    "    delta[delta_pred < -0.5] = -1.\n",
    "    Epsilon = np.zeros(A.shape)\n",
    "    Epsilon[np.logical_and(A > 0.5, Epsilon_pred < -0.5)] = -1.\n",
    "    Epsilon[np.logical_and(A < 0.5, Epsilon_pred > +0.5)] = +1.\n",
    "    return delta, Epsilon, state\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "pycharm": {
     "name": "#%%\n"
    }
   },
   "outputs": [],
   "source": [
    "# EVALUATION FUNCTIONS\n",
    "eval_criteria = ['node_ins_recall',\n",
    "                 'node_ins_precision',\n",
    "                 'node_del_recall',\n",
    "                 'node_del_precision',\n",
    "                 'edge_ins_recall',\n",
    "                 'edge_ins_precision',\n",
    "                 'edge_del_recall',\n",
    "                 'edge_del_precision']\n",
    "# set up a function to compute precision and recall\n",
    "def prec_rec(X, Y):\n",
    "    # X is the prediction, Y is the target\n",
    "    target_insertions = Y > 0.5\n",
    "    predicted_insertions = X > 0.5\n",
    "    target_deletions = Y < -0.5\n",
    "    predicted_deletions = X < -0.5\n",
    "    # first, check the insertion recall\n",
    "    if np.sum(target_insertions) < 0.5:\n",
    "        ins_rec = 1.\n",
    "    else:\n",
    "        ins_rec  = np.mean(X[target_insertions] > 0.5)\n",
    "    # then the insertion precision\n",
    "    if np.sum(predicted_insertions) < 0.5:\n",
    "        ins_prec = 1.\n",
    "    else:\n",
    "        ins_prec = np.mean(Y[predicted_insertions] > 0.5)\n",
    "    # then the deletion recall\n",
    "    if np.sum(target_deletions) < 0.5:\n",
    "        del_rec = 1.\n",
    "    else:\n",
    "        del_rec  = np.mean(X[target_deletions] < -0.5)\n",
    "    # and finally the deletion precision\n",
    "    if np.sum(predicted_deletions) < 0.5:\n",
    "        del_prec = 1.\n",
    "    else:\n",
    "        del_prec = np.mean(Y[predicted_deletions] < -0.5)\n",
    "    return ins_rec, ins_prec, del_rec, del_prec"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Dataset setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "import graph_edit_cycles\n",
    "import degree_rules\n",
    "import game_of_life\n",
    "import random\n",
    "\n",
    "\n",
    "# DATASET SETUP\n",
    "def generate_edit_cycle():\n",
    "    As, Xs, tuples = graph_edit_cycles.generate_time_series(random.randrange(3), random.randrange(12), random.randrange(4, 12))\n",
    "    deltas = []\n",
    "    Epsilons = []\n",
    "    for tpl in tuples:\n",
    "        deltas.append(tpl[0])\n",
    "        Epsilons.append(tpl[1])\n",
    "    return As, Xs, deltas, Epsilons\n",
    "\n",
    "\n",
    "def generate_degree_rules():\n",
    "    # the initial number of nodes in each graph\n",
    "    n_init = 8\n",
    "    # the maximum number of nodes that can occur in each graph during evolution\n",
    "    n_max  = n_init * 4\n",
    "    return degree_rules.generate_time_series_from_random_matrix(n_init, n_max = n_max)\n",
    "\n",
    "\n",
    "def generate_game_of_life():\n",
    "    # set hyper-parameters for the game of life random grid generation\n",
    "    grid_size = 10\n",
    "    num_shapes = 1\n",
    "    p = 0.1\n",
    "    T_max = 10\n",
    "    A, Xs, deltas = game_of_life.generate_random_time_series(grid_size, num_shapes, p, T_max)\n",
    "    As = [A] * len(Xs)\n",
    "    Epsilons = [np.zeros_like(A)] * len(Xs)\n",
    "    return As, Xs, deltas, Epsilons"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "pycharm": {
     "name": "#%%\n"
    }
   },
   "outputs": [],
   "source": [
    "# CONFIG FOR EXPERIMENTS\n",
    "\n",
    "# Models\n",
    "models = ['VGAE', 'VGRNN', 'GEN_crossent', 'GEN']\n",
    "setup_funs = [setup_vgae, setup_vgrnn, setup_gen, setup_gen]\n",
    "loss_funs = [vgae_loss, vgrnn_loss, gen_loss_crossent, gen_loss]\n",
    "pred_funs = [vgae_pred, vgrnn_pred, gen_pred, gen_pred]\n",
    "\n",
    "# Datasets\n",
    "datasets = ['edit_cycles', 'degree_rules', 'game_of_life']\n",
    "dim_ins  = [4, 32, 1]\n",
    "generator_funs = [generate_edit_cycle, generate_degree_rules, generate_game_of_life]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Actual Experiment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--- data set edit_cycles ---\n",
      "\n",
      "--- model VGAE ---\n",
      "node_ins_recall: 0.616033 +- 0.0137438\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.690418 +- 0.0622833\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN_crossent ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 1 +- 0\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 1 +- 0\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "\n",
      "--- data set degree_rules ---\n",
      "\n",
      "--- model VGAE ---\n",
      "node_ins_recall: 0.146334 +- 0.033497\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.95537 +- 0.0244676\n",
      "edge_ins_recall: 0.881462 +- 0.0254464\n",
      "edge_ins_precision: 0.973631 +- 0.0516198\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 0.965385 +- 0.0692308\n",
      "--- model GEN_crossent ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 1 +- 0\n",
      "edge_ins_recall: 0.967017 +- 0.0453511\n",
      "edge_ins_precision: 0.9869 +- 0.0167524\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 1 +- 0\n",
      "edge_ins_recall: 0.970246 +- 0.0569909\n",
      "edge_ins_precision: 0.985629 +- 0.0287424\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "\n",
      "--- data set game_of_life ---\n",
      "\n",
      "--- model VGAE ---\n",
      "node_ins_recall: 0.268 +- 0.0808455\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.0341 +- 0.00352307\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN_crossent ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.979732 +- 0.0405352\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 0.997893 +- 0.0042144\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.9991 +- 0.00111355\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n"
     ]
    }
   ],
   "source": [
    "# iterate over all datasets\n",
    "for d in range(len(datasets)):\n",
    "    print('\\n--- data set %s ---\\n' % datasets[d])\n",
    "    # load partial runtime results if possible\n",
    "    runtimes_file = 'results/%s_runtimes.csv' % datasets[d]\n",
    "    if os.path.exists(runtimes_file):\n",
    "        runtimes = np.loadtxt(runtimes_file, skiprows = 1, delimiter = '\\t')\n",
    "    else:\n",
    "        runtimes = np.full((R, len(models)), np.nan)\n",
    "    # iterate over all models\n",
    "    for k in range(len(models)):\n",
    "        print('--- model %s ---' % models[k])\n",
    "        # load partial results if possible\n",
    "        results_file = 'results/%s_%s_results.csv' % (datasets[d], models[k])\n",
    "        curves_file  = 'results/%s_%s_learning_curves.csv' % (datasets[d], models[k])\n",
    "        if os.path.exists(results_file):\n",
    "            results = np.loadtxt(results_file, skiprows = 1, delimiter = '\\t')\n",
    "            learning_curves = np.loadtxt(curves_file, delimiter = '\\t')\n",
    "        else:\n",
    "            results = np.full((R, len(eval_criteria)), np.nan)\n",
    "            learning_curves = np.full((max_epochs, R), np.nan)\n",
    "        # iterate over experimental repeats\n",
    "        for r in range(R):\n",
    "            # check if this repeat is already evaluated; if so, skip it\n",
    "            if not np.isnan(learning_curves[0, r]):\n",
    "                continue\n",
    "            print('-- repeat %d of %d --' % (r+1, R))\n",
    "            start_time = time.time()\n",
    "            # set up model\n",
    "            if datasets[d] == 'game_of_life':\n",
    "                nonlin = torch.nn.Sigmoid()\n",
    "            else:\n",
    "                nonlin = torch.nn.ReLU()\n",
    "            model = setup_funs[k](dim_ins[d], nonlin)\n",
    "            # set up optimizer\n",
    "            optimizer = torch.optim.Adam(model.parameters(), lr=learning_rate, weight_decay=weight_decay)\n",
    "            # initialize moving loss average for printing\n",
    "            loss_avg = None\n",
    "            # start training\n",
    "            for epoch in range(max_epochs):\n",
    "                optimizer.zero_grad()\n",
    "                # sample a time series from the data set\n",
    "                As, Xs, deltas, Epsilons = generator_funs[d]()\n",
    "                # compute the loss over all time steps\n",
    "                loss = 0.\n",
    "                state = None\n",
    "                for t in range(len(As)):\n",
    "                    # compute loss\n",
    "                    loss_obj, state = loss_funs[k](model, As[t], Xs[t], deltas[t], Epsilons[t], state=state)\n",
    "                    # compute gradient\n",
    "                    loss_obj.backward()\n",
    "                    # accumulate loss\n",
    "                    loss += loss_obj.item()\n",
    "                # perform an optimizer step\n",
    "                optimizer.step()\n",
    "                # store the current loss value in the learning curve\n",
    "                learning_curves[epoch, r] = loss\n",
    "                # compute a new moving average over the loss\n",
    "                if loss_avg is None:\n",
    "                    loss_avg = loss\n",
    "                else:\n",
    "                    loss_avg = loss_avg * 0.9 + 0.1 * loss\n",
    "                # print every print_step steps\n",
    "                if(epoch+1) % print_step == 0:\n",
    "                    print('loss avg after %d epochs: %g' % (epoch+1, loss_avg))\n",
    "                # stop early if the moving average is small\n",
    "                if loss_avg < loss_threshold:\n",
    "                    break\n",
    "            # perform evaluation on new time series\n",
    "            results[r, :] = 0.\n",
    "            T = 0\n",
    "            for j in range(N_test):\n",
    "                # get a random time series from the dataset\n",
    "                As, Xs, deltas, Epsilons = generator_funs[d]()\n",
    "                state = None\n",
    "                for t in range(len(As)):\n",
    "                    # predict the current time step with the network\n",
    "                    delta, Epsilon, state = pred_funs[k](model, As[t], Xs[t], state=state)\n",
    "                    # assess node edit precision and recall\n",
    "                    results[r, :4] += prec_rec(delta, deltas[t])\n",
    "                    # assess edge edit precision and recall\n",
    "                    results[r, 4:] += prec_rec(Epsilon, Epsilons[t])\n",
    "                    # TODO this is only for debugging purposes\n",
    "                    if (d != 1 and k == 2 and np.any(results[r, :] < 0.99)) or np.any(np.isnan(results[r, :])):\n",
    "                        print('delta (predicted) = %s' % str(delta))\n",
    "                        print('delta (target) = %s' % str(deltas[t]))\n",
    "                        print('Epsilon (predicted) = %s' % str(Epsilon))\n",
    "                        print('Epsilon (target) = %s' % str(Epsilons[t]))\n",
    "                        raise ValueError('stop')\n",
    "                        \n",
    "                T += len(As)\n",
    "            results[r, :] /= T\n",
    "            # store runtime\n",
    "            runtimes[r, k] = time.time() - start_time\n",
    "            np.savetxt(runtimes_file, runtimes, delimiter = '\\t', fmt = '%g', header = '\\t'.join(models), comments = '')\n",
    "            # store results\n",
    "            np.savetxt(results_file, results, delimiter = '\\t', fmt = '%g', header = '\\t'.join(eval_criteria), comments = '')\n",
    "            # store learning curves\n",
    "            np.savetxt(curves_file, learning_curves, delimiter = '\\t', fmt = '%g')\n",
    "        # print results\n",
    "        for crit in range(len(eval_criteria)):\n",
    "            print('%s: %g +- %g' % (eval_criteria[crit], np.mean(results[:, crit]), np.std(results[:, crit])))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/phait/dev/tf2/lib/python3.7/site-packages/ipykernel_launcher.py:9: RuntimeWarning: Mean of empty slice\n",
      "  if __name__ == '__main__':\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x243 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# visualize learning curves\n",
    "import matplotlib.pyplot as plt\n",
    "smoothing_steps = 10\n",
    "fig, axes = plt.subplots(ncols=1, nrows=len(datasets))\n",
    "for d in range(len(datasets)):\n",
    "    for k in range(len(models)):\n",
    "        curves_file  = 'results/%s_%s_learning_curves.csv' % (datasets[d], models[k])\n",
    "        learning_curves = np.loadtxt(curves_file, delimiter = '\\t')\n",
    "        acum = np.cumsum(np.nanmean(learning_curves, 1))\n",
    "        axes[d].semilogy((acum[smoothing_steps:] - acum[:-smoothing_steps])/smoothing_steps)\n",
    "    axes[d].set_xlabel('epoch')\n",
    "    axes[d].set_ylabel('loss')\n",
    "    axes[d].set_title(datasets[d])\n",
    "    axes[d].legend(models)\n",
    "plt.show()"
   ]
  }
 ],
 "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.7.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
