{
  "cells": [
    {
      "cell_type": "code",
      "source": [
        "#!g1.1\n",
        "import torch\n",
        "import torch.nn as nn\n",
        "from model_rnn import MainModel\n",
        "from dataset import LocalizationDataset\n",
        "from torch.utils.data import DataLoader\n",
        "from tqdm import tqdm\n",
        "import numpy as np\n",
        "import pandas as pd\n",
        "from ModelParams import ModelParams\n",
        "from matplotlib import pyplot as plt\n",
        "import matplotlib as mpl\n",
        "from IPython.display import clear_output"
      ],
      "outputs": [],
      "execution_count": 2,
      "metadata": {
        "cellId": "9c89p320il9z9rlv6wv1pm",
        "id": "3035bb16",
        "gather": {
          "logged": 1695927616350
        }
      },
      "id": "3035bb16"
    },
    {
      "cell_type": "code",
      "source": [
        "#!g1.1\n",
        "device = 'cuda:0' if torch.cuda.is_available() else 'cpu'\n",
        "\n",
        "num_tracks = 10000\n",
        "track_data = np.array(pd.read_csv('trajectories.csv', header=None))\n",
        "world = np.loadtxt('environment.csv', delimiter=',')\n",
        "points = []\n",
        "for i, line in enumerate(world):\n",
        "    nb_y = world.shape[0] - i - 1\n",
        "    for j, block in enumerate(line):\n",
        "        if block == 2:\n",
        "            points.extend([[j+0.5, nb_y+0.5]])\n",
        "\n",
        "points = torch.from_numpy(np.array(points)).double().to(device)\n",
        "\n",
        "all_data = dict()\n",
        "\n",
        "track_len = track_data.shape[0] // num_tracks\n",
        "track = np.zeros((num_tracks, track_len, track_data.shape[1]))\n",
        "for i in range(num_tracks):\n",
        "    track[i] = track_data[i*track_len:(i+1)*track_len]\n",
        "\n",
        "eval_test_numbers = np.random.choice(num_tracks, size=num_tracks//5, replace=False)\n",
        "eval_numbers = eval_test_numbers[:len(eval_test_numbers)//2]\n",
        "test_numbers = eval_test_numbers[len(eval_test_numbers)//2:]\n",
        "train_numbers = np.setdiff1d(np.arange(num_tracks), eval_test_numbers)\n",
        "\n",
        "train_data = dict()\n",
        "eval_data = dict()\n",
        "test_data = dict()\n",
        "\n",
        "train_data['tracks'] = track[train_numbers]\n",
        "train_data['map'] = world\n",
        "\n",
        "eval_data['tracks'] = track[eval_numbers]\n",
        "eval_data['map'] = world\n",
        "\n",
        "test_data['tracks'] = track[test_numbers]\n",
        "test_data['map'] = world\n",
        "\n",
        "\n",
        "train_dataset = LocalizationDataset(train_data)\n",
        "eval_dataset = LocalizationDataset(eval_data)\n",
        "test_dataset = LocalizationDataset(test_data)\n",
        "\n",
        "params = ModelParams()\n",
        "train_loader = DataLoader(train_dataset, batch_size=params.batch_size, pin_memory=True, shuffle=True)\n",
        "eval_loader = DataLoader(eval_dataset, batch_size=params.batch_size, pin_memory=True, shuffle=False)\n",
        "test_loader = DataLoader(test_dataset, batch_size=params.batch_size, pin_memory=True, shuffle=False)\n",
        "\n",
        "device = 'cuda:0' if torch.cuda.is_available() else 'cpu'"
      ],
      "outputs": [],
      "execution_count": 3,
      "metadata": {
        "cellId": "5gg9z2ucnml262zy30ts37h",
        "id": "85d4720a",
        "gather": {
          "logged": 1695927623515
        }
      },
      "id": "85d4720a"
    },
    {
      "cell_type": "code",
      "source": [
        "measurement_train = torch.zeros((0, params.obs_num)).double()\n",
        "motion_train = torch.zeros((0, 3)).double()\n",
        "for iteration, data in enumerate(train_loader):\n",
        "    _, measurement, location, motion = data\n",
        "    measurement_train = torch.cat((measurement_train, measurement.view(-1, params.obs_num)), dim=0)\n",
        "    motion_train = torch.cat((motion_train, motion.view(-1, 3)), dim=0)\n",
        "motion_mu, motion_std = motion_train.mean(dim=0).unsqueeze(0).unsqueeze(1), motion_train.std(dim=0).unsqueeze(0).unsqueeze(1)\n",
        "measurement_mu, measurement_std = measurement_train.mean().unsqueeze(0).unsqueeze(1).unsqueeze(2), measurement_train.std().unsqueeze(0).unsqueeze(1).unsqueeze(2)"
      ],
      "outputs": [],
      "execution_count": 4,
      "metadata": {
        "jupyter": {
          "source_hidden": false,
          "outputs_hidden": false
        },
        "nteract": {
          "transient": {
            "deleting": false
          }
        },
        "gather": {
          "logged": 1695927631164
        }
      },
      "id": "614e297e-00cd-46cf-b8ce-728c17a9d345"
    },
    {
      "cell_type": "code",
      "source": [
        "#!g1.1\n",
        "model = MainModel(points).to(device).double()\n",
        "epochs = 5000\n",
        "grad_clip = 3\n",
        "MSE_min = np.Inf\n",
        "MSE_accum = []\n",
        "optimizer = torch.optim.RMSprop(model.parameters(), lr=5e-4)\n",
        "\n",
        "for epoch in tqdm(range(epochs)):\n",
        "    model.train()\n",
        "    train_loss = []\n",
        "    curr_loss1 = 0\n",
        "    for iteration, data in enumerate(train_loader):\n",
        "        _, measurement, location, motion = data\n",
        "        measurement = ((measurement - measurement_mu) / measurement_std).to(device).double()\n",
        "        motion = ((motion - motion_mu) / motion_std).to(device).double()\n",
        "        location = location.to(device).double()\n",
        "\n",
        "        model.zero_grad()\n",
        "        loss, last_loss, pred, particle_pred = model.step(measurement, motion, location, params.bpdecay)\n",
        "        loss.backward()\n",
        "        torch.nn.utils.clip_grad_norm_(model.parameters(), grad_clip)\n",
        "        optimizer.step()\n",
        "        curr_loss1 += loss.to('cpu').detach().numpy() / len(train_numbers)\n",
        "    train_loss.append(curr_loss1)\n",
        "\n",
        "    model.eval()\n",
        "    eval_loss = []\n",
        "    curr_loss2 = 0\n",
        "    y_pred = []\n",
        "    y_true = []\n",
        "    with torch.no_grad():\n",
        "        for iteration, data in enumerate(eval_loader):\n",
        "            _, measurement, location, motion = data\n",
        "            batch_size, seq_len = motion.size(0), motion.size(1)\n",
        "            measurement = ((measurement - measurement_mu) / measurement_std).to(device).double()\n",
        "            motion = ((motion - motion_mu) / motion_std).to(device).double()\n",
        "            location = location.to(device).double()\n",
        "            loss, last_loss, pred, particle_pred = model.step(measurement, motion, location, params.bpdecay)\n",
        "            curr_loss2 += loss.to('cpu').detach().numpy() / len(eval_numbers)\n",
        "            y_true_curr = location[:, :, :2]\n",
        "            y_pred_curr = torch.cat((pred[:, :, :1] * model.width, pred[:, :, 1:2] * model.height), dim=2)\n",
        "            y_true.extend(y_true_curr.to('cpu').numpy())\n",
        "            y_pred.extend(y_pred_curr.detach().to('cpu').numpy())\n",
        "\n",
        "    eval_loss.append(curr_loss2)\n",
        "    y_true = np.array(y_true)\n",
        "    y_pred = np.array(y_pred)\n",
        "    y_true = y_true.reshape(len(y_true), track_len, 2)\n",
        "    y_pred = y_pred.reshape(len(y_pred), track_len, 2)\n",
        "\n",
        "    X = (y_pred - y_true).reshape(y_pred.shape[0]*track_len, 2)\n",
        "    MSE = (X**2).sum(axis=-1).mean()\n",
        "    print('epoch, MSE', epoch, np.round(MSE, 3))\n",
        "    if MSE < MSE_min:\n",
        "        state = {\n",
        "            'epoch': epoch,\n",
        "            'state_dict': model.state_dict(),\n",
        "            'optimizer': optimizer.state_dict(),\n",
        "        }\n",
        "        filepath='mbpfn10.ptm'\n",
        "        torch.save(state, filepath)\n",
        "        MSE_min = MSE\n",
        "\n",
        "    MSE_accum.append(MSE)\n",
        "\n",
        "    clear_output(True)\n",
        "    plt.figure()\n",
        "    plt.title(str(np.round(MSE_min, 2)))\n",
        "    plt.plot(MSE_accum)\n",
        "    plt.plot([MSE_min]*(len(MSE_accum)))\n",
        "    plt.show()"
      ],
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": "epoch, MSE 352 0.899\n"
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": "<Figure size 640x480 with 1 Axes>",
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAh8AAAGzCAYAAACPa3XZAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAABHmUlEQVR4nO3dd3iT5eI+8DujSXe6F22h7F1kVUQZwmGqeFRUxB84vuLAvVFxHk8d53icB8dxgIq4cSGKTJEyCpRNoVA6aQtdadpmNHl+f6R5IbSFFpK8pe/9ua5cV/OO5HmSam+eqRJCCBARERH5iFruAhAREZGyMHwQERGRTzF8EBERkU8xfBAREZFPMXwQERGRTzF8EBERkU8xfBAREZFPMXwQERGRTzF8EBERkU8xfBAREZFPMXwQUZtYLBY89thjSEhIQEBAANLS0rBixYpW3fvHH39g7NixiIqKQlhYGIYPH45PP/30tPesX78eKpUKKpUKx48f90QViEhmDB9E1CY33XQTXnvtNcycORNvvPEGNBoNpkyZgvXr15/2vh9//BETJkyA1WrFs88+ixdffBEBAQGYNWsW/vOf/zR7j8PhwD333IOgoCBvVIWIZKLixnJE1FqbN29GWloaXn31VTz88MMAALPZjP79+yMmJgYbNmxo8d4JEyZgz549OHz4MPR6PQCgoaEBvXv3RlBQEHbs2NHknnfffRdPPfUUbrzxRrzxxhs4duwYoqKivFM5IvIZtnwQUat988030Gg0mDNnjnTM398ft956KzIyMlBQUNDivUajEeHh4VLwAACtVouoqCgEBAQ0ub6iogJPPfUUnn/+eYSFhXm0HkQkL4YPImq17du3o2fPnggNDXU7Pnz4cABAVlZWi/eOGTMGe/bswfz585GTk4NDhw7hhRdeQGZmJh599NEm18+fPx9xcXG4/fbbPVoHIpKfVu4CENH54+jRo4iPj29y3HWsuLi4xXvnz5+P3NxcvPjii/jHP/4BAAgMDMS3336LadOmuV27c+dOvPfee1i2bBk0Go0Ha0BE7QFbPoio1err6926TVz8/f2l8y3R6/Xo2bMnrrnmGnzxxRf47LPPMHToUNx4443YuHGj27X33nsvJk+ejAkTJni2AkTULrDlg4haLSAgABaLpclxs9ksnW/J3XffjY0bN2Lbtm1Qq53/7rn22mvR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"
          },
          "metadata": {}
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": "\r  7%|▋         | 353/5000 [2:46:25<24:07:00, 18.68s/it]"
        }
      ],
      "execution_count": null,
      "metadata": {
        "cellId": "shfnuoxcnqcsqqk5ghc0r",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 469
        },
        "id": "7ea6dfc0",
        "outputId": "35be7436-d5a4-4c1f-f807-24bf30bc7cae",
        "gather": {
          "logged": 1695927396322
        }
      },
      "id": "7ea6dfc0"
    },
    {
      "cell_type": "code",
      "source": [
        "plt.plot(MSE_accum)\n",
        "plt.plot([MSE_min]*(len(MSE_accum)))"
      ],
      "outputs": [
        {
          "output_type": "execute_result",
          "execution_count": 10,
          "data": {
            "text/plain": "[<matplotlib.lines.Line2D at 0x7fcb243836d0>]"
          },
          "metadata": {}
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": "<Figure size 640x480 with 1 Axes>",
            "image/png": "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O1bR0dFKTk7WI488Io/HE8y3clZ79tln5XA4NGnSJHMb58U6+/fv109+8hMlJSUpKipKffv21erVq839hmHoySefVIcOHRQVFaVRo0Zpx44dfs9x5MgRjR8/XnFxcUpISNAvfvELlZSUBPutnDW8Xq+mTJmirl27KioqSt26ddNTTz3ld/0SzkszGDYwe/ZsIyIiwnjjjTeMzZs3G3fddZeRkJBg5OXlWd20s9bo0aONN99809i0aZOxfv1645prrjEyMjKMkpIS85h77rnHSE9PN+bNm2esXr3auPjii41LLrnE3O/xeIw+ffoYo0aNMtatW2d89tlnRrt27YzJkydb8ZbOOitXrjS6dOli9OvXz3jwwQfN7ZwXaxw5csTo3Lmz8fOf/9xYsWKFsWvXLuPLL780srKyzGOeffZZIz4+3vj444+NDRs2GNdff73RtWtXo7y83Dzm6quvNvr3728sX77c+Oabb4zu3bsbt912mxVv6awwdepUIykpyZgzZ46xe/du44MPPjDatGljPPfcc+YxnJdTZ4vwMXjwYGPixInmfa/Xa6SlpRmZmZkWtspe8vPzDUnGokWLDMMwjMLCQiM8PNz44IMPzGO+//57Q5KxbNkywzAM47PPPjNCQkKM3Nxc85jp06cbcXFxhtvtDu4bOMsUFxcbPXr0MObOnWtcccUVZvjgvFjnscceMy699NLj7vf5fEZqaqrx5z//2dxWWFhoOJ1O47333jMMwzC2bNliSDJWrVplHvP5558bDofD2L9/f8s1/iw2duxY48477/Tb9sMf/tAYP368YRicl+Y667tdKisrtWbNGo0aNcrcFhISolGjRmnZsmUWtsxeioqKJEmJiYmSpDVr1qiqqsrvvPTs2VMZGRnmeVm2bJn69u2rlJQU85jRo0fL5XJp8+bNQWz92WfixIkaO3as3/df4rxY6ZNPPtGgQYP0ox/9SMnJyRowYIBee+01c//u3buVm5vrd27i4+M1ZMgQv3OTkJCgQYMGmceMGjVKISEhWrFiRfDezFnkkksu0bx587R9+3ZJ0oYNG7RkyRKNGTNGEueluVrdheUC7fDhw/J6vX6/KCUpJSVFW7dutahV9uLz+TRp0iQNGzZMffr0kSTl5uYqIiJCCQkJfsempKQoNzfXPKax81a7D80ze/ZsrV27VqtWrWqwj/NinV27dmn69Ol6+OGH9cQTT2jVqlV64IEHFBERoQkTJpjf28a+9/XPTXJyst/+sLAwJSYmcm6a6fHHH5fL5VLPnj0VGhoqr9erqVOnavz48ZLEeWmmsz58wHoTJ07Upk2btGTJEqubYns5OTl68MEHNXfuXEVGRlrdHNTj8/k0aNAgPfPMM5KkAQMGaNOmTXrllVc0YcIEi1tnX++//75mzpypWbNmqXfv3lq/fr0mTZqktLQ0zstpOOu7Xdq1a6fQ0NAGo/Xz8vKUmppqUavs4/7779ecOXO0YMECderUydyempqqyspKFRYW+h1f/7ykpqY2et5q9+HUrVmzRvn5+Ro4cKDCwsIUFhamRYsW6fnnn1dYWJhSUlI4Lxbp0KGDevXq5bft/PPPV3Z2tqS6721Tv8tSU1OVn5/vt9/j8ejIkSOcm2Z65JFH9Pjjj+vWW29V37599dOf/lQPPfSQMjMzJXFemuusDx8RERG68MILNW/ePHObz+fTvHnzNHToUAtbdnYzDEP333+/PvroI82fP19du3b123/hhRcqPDzc77xs27ZN2dnZ5nkZOnSoNm7c6PdDO3fuXMXFxTX4JY2TM3LkSG3cuFHr1683b4MGDdL48ePNrzkv1hg2bFiD6ejbt29X586dJUldu3ZVamqq37lxuVxasWKF37kpLCzUmjVrzGPmz58vn8+nIUOGBOFdnH3KysoUEuL/URkaGiqfzyeJ89JsVo94DYbZs2cbTqfTeOutt4wtW7YYd999t5GQkOA3Wh+Bde+99xrx8fHGwoULjYMHD5q3srIy85h77rnHyMjIMObPn2+sXr3aGDp0qDF06FBzf+2UzquuuspYv3698cUXXxjt27dnSmeA1Z/tYhicF6usXLnSCAsLM6ZOnWrs2LHDmDlzphEdHW28++675jHPPvuskZCQYPznP/8xvvvuO+OGG25odErngAEDjBUrVhhLliwxevToYespnadrwoQJRseOHc2pth9++KHRrl0749FHHzWP4bycOluED8MwjBdeeMHIyMgwIiIijMGDBxvLly+3uklnNUmN3t58803zmPLycuO+++4z2rZta0RHRxs33nijcfDgQb/n2bNnjzFmzBgjKirKaNeunfHrX//aqKqqCvK7ObsdGz44L9b573//a/Tp08dwOp1Gz549jRkzZvjt9/l8xpQpU4yUlBTD6XQaI0eONLZt2+Z3TEFBgXHbbbcZbdq0MeLi4ow77rjDKC4uDubbOKu4XC7jwQcfNDIyMozIyEjjnHPOMX7729/6TSvnvJw6h2HUW6YNAACghZ31Yz4AAEDrQvgAAABBRfgAAABBRfgAAABBRfgAAABBRfgAAABBRfgAAABBRfgAAABBRfgAAABBRfgAAABBRfgAAABBRfgAAABB9f8Bbi2vZfrdogUAAAAASUVORK5CYII="
          },
          "metadata": {}
        }
      ],
      "execution_count": 10,
      "metadata": {
        "jupyter": {
          "source_hidden": false,
          "outputs_hidden": false
        },
        "nteract": {
          "transient": {
            "deleting": false
          }
        },
        "gather": {
          "logged": 1695955050383
        }
      },
      "id": "46fd87b2-beb7-4450-b2ef-5a43996eb7c6"
    },
    {
      "cell_type": "code",
      "source": [
        "#!g1.1\n",
        "def load_checkpoint(filepath='mbpfn10.ptm'):\n",
        "    checkpoint = torch.load(filepath, map_location=torch.device('cpu'))\n",
        "    print('epoch: ', checkpoint['epoch'])\n",
        "    model.load_state_dict(checkpoint['state_dict'])\n",
        "    for parameter in model.parameters():\n",
        "        parameter.requires_grad = False\n",
        "    model.eval()\n",
        "    return model\n",
        "\n",
        "model = MainModel(points).to(device).double()\n",
        "\n",
        "model = load_checkpoint()\n",
        "\n",
        "model.eval()\n",
        "eval_loss = []\n",
        "curr_loss2 = 0\n",
        "y_pred = []\n",
        "y_true = []\n",
        "with torch.no_grad():\n",
        "    for iteration, data in enumerate(test_loader):\n",
        "        env_map, measurement, location, motion = data\n",
        "        batch_size, seq_len = motion.size(0), motion.size(1)\n",
        "        env_map = env_map.to(device).double()\n",
        "        measurement = ((measurement - measurement_mu) / measurement_std).to(device).double()\n",
        "        motion = ((motion - motion_mu) / motion_std).to(device).double()\n",
        "        location = location.to(device).double()\n",
        "        loss, last_loss, pred, particle_pred = model.step(measurement, motion, location, 0.1)\n",
        "        curr_loss2 += loss.to('cpu').detach().numpy() / len(eval_numbers)\n",
        "        y_true_curr = location[:, :, :2]\n",
        "        y_pred_curr = pred[:, :, :2]\n",
        "        y_pred_curr[:, :, 0] *= model.width\n",
        "        y_pred_curr[:, :, 1] *= model.height\n",
        "        y_true.extend(y_true_curr.to('cpu').numpy())\n",
        "        y_pred.extend(y_pred_curr.detach().to('cpu').numpy())\n",
        "\n",
        "eval_loss.append(curr_loss2)\n",
        "y_true = np.array(y_true)\n",
        "y_pred = np.array(y_pred)\n",
        "y_true = y_true.reshape(len(y_true), track_len, 2)\n",
        "y_pred = y_pred.reshape(len(y_pred), track_len, 2)\n",
        "\n",
        "X = (y_pred - y_true).reshape(y_pred.shape[0]*seq_len, 2)\n",
        "MSE_arr = (X**2).sum(axis=-1)\n",
        "Y = ((((y_pred - y_true)[:, -1])**2).sum(axis=-1)**0.5)\n",
        "print('MSE mean and std, final point mean and std', MSE_arr.mean(), MSE_arr.std(), Y.mean(), Y.std())"
      ],
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": "epoch:  308\nMSE mean and std, final point mean and std 0.7667002961003556 4.01041581392178 0.13136740511567477 0.08936940708403202\n"
        }
      ],
      "execution_count": 15,
      "metadata": {
        "cellId": "6gvjtmok1kwmqr1hwl0hzf",
        "id": "d398ad25",
        "outputId": "bd9d046a-4498-4686-ae27-b407ec2cdac8",
        "gather": {
          "logged": 1695955240112
        }
      },
      "id": "d398ad25"
    },
    {
      "cell_type": "code",
      "source": [
        "#!g1.1\n"
      ],
      "outputs": [],
      "execution_count": null,
      "metadata": {
        "cellId": "xw34g7w45tx4usilocup",
        "id": "5cad5998"
      },
      "id": "5cad5998"
    },
    {
      "cell_type": "code",
      "source": [
        "#!g1.1\n"
      ],
      "outputs": [],
      "execution_count": null,
      "metadata": {
        "cellId": "ukmfxbee2u6ztki76nc53",
        "id": "0b810608"
      },
      "id": "0b810608"
    },
    {
      "cell_type": "code",
      "source": [
        "#!g1.1\n"
      ],
      "outputs": [],
      "execution_count": null,
      "metadata": {
        "cellId": "echs5rsk1mvvnca9mufrzg",
        "id": "6f9bee46"
      },
      "id": "6f9bee46"
    },
    {
      "cell_type": "code",
      "source": [
        "#!g1.1\n"
      ],
      "outputs": [],
      "execution_count": null,
      "metadata": {
        "cellId": "gck4f9ag4gzljk8vg7yj",
        "id": "a8599d74"
      },
      "id": "a8599d74"
    },
    {
      "cell_type": "code",
      "source": [
        "#!g1.1\n"
      ],
      "outputs": [],
      "execution_count": null,
      "metadata": {
        "cellId": "tecp3gzyuymggi8fojs9bl",
        "id": "ee310f7c"
      },
      "id": "ee310f7c"
    }
  ],
  "metadata": {
    "notebookPath": "Jup_main.ipynb",
    "colab": {
      "gpuType": "V100",
      "machine_shape": "hm",
      "provenance": []
    },
    "accelerator": "GPU",
    "gpuClass": "standard",
    "microsoft": {
      "ms_spell_check": {
        "ms_spell_check_language": "en"
      }
    },
    "kernel_info": {
      "name": "python3"
    },
    "language_info": {
      "name": "python",
      "version": "3.8.5",
      "mimetype": "text/x-python",
      "codemirror_mode": {
        "name": "ipython",
        "version": 3
      },
      "pygments_lexer": "ipython3",
      "nbconvert_exporter": "python",
      "file_extension": ".py"
    },
    "notebookId": "2eff6be8-f7ea-4468-a171-8075d4bcb638",
    "kernelspec": {
      "name": "python3",
      "language": "python",
      "display_name": "Python 3 (ipykernel)"
    },
    "nteract": {
      "version": "nteract-front-end@1.0.0"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 5
}