{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "4ce6f328",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "import sys\n",
    "sys.path.append(\"../../src\")\n",
    "import feature.scrna_dataset as scrna_dataset\n",
    "import model.sdes as sdes\n",
    "import model.generate as generate\n",
    "import model.scrna_ae as scrna_ae\n",
    "import model.util as model_util\n",
    "import analysis.fid as fid\n",
    "import torch\n",
    "import numpy as np\n",
    "import scipy.stats\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.font_manager as font_manager\n",
    "import os\n",
    "import h5py\n",
    "import json\n",
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5f452f18",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plotting defaults\n",
    "font_list = font_manager.findSystemFonts(fontpaths=[\"/home/anon/modules/fonts\"])\n",
    "for font in font_list:\n",
    "    font_manager.fontManager.addfont(font)\n",
    "plot_params = {\n",
    "    \"figure.titlesize\": 22,\n",
    "    \"axes.titlesize\": 22,\n",
    "    \"axes.labelsize\": 20,\n",
    "    \"legend.fontsize\": 18,\n",
    "    \"font.size\": 13,\n",
    "    \"xtick.labelsize\": 16,\n",
    "    \"ytick.labelsize\": 16,\n",
    "    \"font.family\": \"Roboto\",\n",
    "    \"font.weight\": \"bold\",\n",
    "    \"svg.fonttype\": \"none\"\n",
    "}\n",
    "plt.rcParams.update(plot_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "df22d86f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define device\n",
    "if torch.cuda.is_available():\n",
    "    DEVICE = \"cuda\"\n",
    "else:\n",
    "    DEVICE = \"cpu\""
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6dfb808b",
   "metadata": {},
   "source": [
    "### Define constants and paths"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "2cb7c2d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "models_base_path = \"/data/anon/branched_diffusion/models/trained_models/\"\n",
    "\n",
    "branched_model_path = os.path.join(models_base_path, \"scrna_covid_flu_continuous_branched_9classes_latent_d200/3/last_ckpt.pth\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "28376e47",
   "metadata": {},
   "outputs": [],
   "source": [
    "def import_classes_branch_points(json_path):\n",
    "    with open(json_path, \"r\") as f:\n",
    "        d = json.load(f)\n",
    "        return d[\"classes\"], \\\n",
    "            [(tuple(trip[0]), trip[1], trip[2]) for trip in d[\"branches\"]]\n",
    "    \n",
    "classes, branch_defs = import_classes_branch_points(\n",
    "    \"/home/anon/branched_diffusion/data/config/classes_branch_points/scrna_covid_flu/redset.json\"\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "2e29ad43",
   "metadata": {},
   "outputs": [],
   "source": [
    "latent_space = True\n",
    "latent_dim = 200"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "1c8f4db2",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_file = \"/data/anon/branched_diffusion/data/scrna/covid_flu/processed/covid_flu_processed_reduced_genes.h5\"\n",
    "autoencoder_path = \"/data/anon/branched_diffusion/models/trained_models/scrna_vaes/covid_flu/covid_flu_processed_reduced_genes_ldvae_d%d/\" % latent_dim"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "0364c0a2",
   "metadata": {},
   "outputs": [],
   "source": [
    "out_path = \"/home/anon/branched_diffusion/figures/scrna_covid_flu_efficiency\"\n",
    "\n",
    "os.makedirs(out_path, exist_ok=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c77b6452",
   "metadata": {},
   "source": [
    "### Create data loader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "98285ad2",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[rank: 0] Global seed set to 0\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[34mINFO    \u001b[0m File                                                                                                      \n",
      "         \u001b[35m/data/anon/branched_diffusion/models/trained_models/scrna_vaes/covid_flu/covid_flu_proc\u001b[0m\n",
      "         \u001b[35messed_reduced_genes_ldvae_d200/\u001b[0m\u001b[95mmodel.pt\u001b[0m already downloaded                                                \n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "No GPU/TPU found, falling back to CPU. (Set TF_CPP_MIN_LOG_LEVEL=0 and rerun for more info.)\n",
      "/home/anon/miniconda3/envs/scanpy/lib/python3.9/site-packages/scvi/data/fields/_layer_field.py:91: UserWarning: adata.X does not contain unnormalized count data. Are you sure this is what you want?\n",
      "  warnings.warn(\n"
     ]
    }
   ],
   "source": [
    "dataset = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=(autoencoder_path if latent_space else None))\n",
    "\n",
    "# Limit classes\n",
    "inds = np.isin(dataset.cell_cluster, classes)\n",
    "dataset.data = dataset.data[inds]\n",
    "dataset.cell_cluster = dataset.cell_cluster[inds]\n",
    "\n",
    "data_loader = torch.utils.data.DataLoader(dataset, batch_size=128, shuffle=True, num_workers=0)\n",
    "input_shape = next(iter(data_loader))[0].shape[1:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "5a99a083",
   "metadata": {},
   "outputs": [],
   "source": [
    "sde = sdes.VariancePreservingSDE(0.1, 20, input_shape)\n",
    "t_limit = 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "51fbc829",
   "metadata": {},
   "outputs": [],
   "source": [
    "# TODO: this is currently rather inefficient; a decision-tree-style structure\n",
    "# would be better\n",
    "\n",
    "def class_time_to_branch(c, t, branch_defs):\n",
    "    \"\"\"\n",
    "    Given a class and a time (both scalars), return the\n",
    "    corresponding branch index.\n",
    "    \"\"\"\n",
    "    for i, branch_def in enumerate(branch_defs):\n",
    "        if c in branch_def[0] and t >= branch_def[1] and t <= branch_def[2]:\n",
    "            return i\n",
    "    raise ValueError(\"Undefined class and time\")\n",
    "        \n",
    "def class_time_to_branch_tensor(c, t, branch_defs):\n",
    "    \"\"\"\n",
    "    Given tensors of classes and a times, return the\n",
    "    corresponding branch indices as a tensor.\n",
    "    \"\"\"\n",
    "    return torch.tensor([\n",
    "        class_time_to_branch(c_i, t_i, branch_defs) for c_i, t_i in zip(c, t)\n",
    "    ], device=DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c35fc1e8",
   "metadata": {},
   "source": [
    "### Import models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "40e98368",
   "metadata": {},
   "outputs": [],
   "source": [
    "branched_model = model_util.load_model(\n",
    "    scrna_ae.MultitaskResNet, branched_model_path\n",
    ").to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5f55c8b",
   "metadata": {},
   "source": [
    "### Generating multiple classes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "c1e50cd6",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n",
      "Total time taken: 275s\n",
      "Average time taken: 275.81 +/- 0.02\n"
     ]
    }
   ],
   "source": [
    "# Sample each class individually without taking advantage of branches\n",
    "num_linear_trials = 10\n",
    "linear_times = []\n",
    "for _ in range(num_linear_trials):\n",
    "    time_a = time.time()\n",
    "    for class_to_sample in classes:\n",
    "        print(\"Sampling class: %s\" % class_to_sample)\n",
    "        generate.generate_continuous_branched_samples(\n",
    "            branched_model, sde, class_to_sample,\n",
    "            lambda c, t: class_time_to_branch_tensor(c, t, branch_defs),\n",
    "            sampler=\"pc\", t_limit=t_limit, num_steps=1000\n",
    "        )\n",
    "    time_b = time.time()\n",
    "    time_taken = time_b - time_a\n",
    "    linear_times.append(time_taken)\n",
    "    print(\"Total time taken: %ds\" % time_taken)\n",
    "print(\"Average time taken: %.2f +/- %.2f\" % (np.mean(linear_times), np.std(linear_times) / np.sqrt(len(linear_times))))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "0de8f9d2",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Sampling branch 1/17\n",
      "Sampling branch 2/17\n",
      "Sampling branch 3/17\n",
      "Sampling branch 4/17\n",
      "Sampling branch 5/17\n",
      "Sampling branch 6/17\n",
      "Sampling branch 7/17\n",
      "Sampling branch 8/17\n",
      "Sampling branch 9/17\n",
      "Sampling branch 10/17\n",
      "Sampling branch 11/17\n",
      "Sampling branch 12/17\n",
      "Sampling branch 13/17\n",
      "Sampling branch 14/17\n",
      "Sampling branch 15/17\n",
      "Sampling branch 16/17\n",
      "Sampling branch 17/17\n",
      "Total time taken: 132s\n",
      "Average time taken: 132.37 +/- 0.01\n"
     ]
    }
   ],
   "source": [
    "# Sample each object by taking advantage of branches\n",
    "num_branched_trials = 10\n",
    "branched_times = []\n",
    "\n",
    "# Sort the branches by starting time point (in reverse order), and generate along those\n",
    "# branches, caching results; this guarantees that we will always find a cached batch\n",
    "# (other than the first one)\n",
    "cache = {}\n",
    "sorted_branch_defs = sorted(branch_defs, key=(lambda t: -t[1]))\n",
    "\n",
    "for _ in range(num_branched_trials):\n",
    "    time_a = time.time()\n",
    "    # First branch\n",
    "    print(\"Sampling branch 1/%d\" % len(sorted_branch_defs))\n",
    "    branch_def = sorted_branch_defs[0]\n",
    "    samples = generate.generate_continuous_branched_samples(\n",
    "        # Specify arbitrary class\n",
    "        branched_model, sde, branch_def[0][0],\n",
    "        lambda c, t: class_time_to_branch_tensor(c, t, branch_defs),\n",
    "        sampler=\"pc\", t_limit=branch_def[2], t_start=branch_def[1],\n",
    "        num_steps=int(1000 * (branch_def[2] - branch_def[1]))\n",
    "    )\n",
    "    for class_i in branch_def[0]:\n",
    "        cache[class_i] = (branch_def[1], samples)\n",
    "\n",
    "    for i, branch_def in enumerate(sorted_branch_defs[1:]):\n",
    "        print(\"Sampling branch %d/%d\" % (i + 2, len(sorted_branch_defs)))\n",
    "        cached_time, cached_samples = cache[branch_def[0][0]]\n",
    "        assert cached_time == branch_def[2]\n",
    "        samples = generate.generate_continuous_branched_samples(\n",
    "            branched_model, sde, branch_def[0][0],\n",
    "            lambda c, t: class_time_to_branch_tensor(c, t, branch_defs),\n",
    "            sampler=\"pc\", t_limit=branch_def[2], t_start=branch_def[1],\n",
    "            num_steps=int(1000 * (branch_def[2] - branch_def[1])),\n",
    "            initial_samples=cached_samples\n",
    "        )\n",
    "        for class_i in branch_def[0]:\n",
    "            cache[class_i] = (branch_def[1], samples)\n",
    "\n",
    "    time_b = time.time()\n",
    "    time_taken = time_b - time_a\n",
    "    branched_times.append(time_taken)\n",
    "    print(\"Total time taken: %ds\" % time_taken)\n",
    "print(\"Average time taken: %.2f +/- %.2f\" % (np.mean(branched_times), np.std(branched_times) / np.sqrt(len(branched_times))))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "38c99f3d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(8, 8))\n",
    "labels = [\"Linear\", \"Branched\"]\n",
    "times = [np.mean(linear_times), np.mean(branched_times)]\n",
    "errors = [np.std(linear_times) / np.sqrt(num_linear_trials), np.std(branched_times) / np.sqrt(num_branched_trials)]\n",
    "ax.bar(labels, times, color=\"mediumorchid\")\n",
    "ax.errorbar(labels, times, yerr=errors, fmt=\"none\", color=\"darkmagenta\")\n",
    "ax.set_xlabel(\"Time to generate one batch of each class\")\n",
    "ax.set_ylabel(\"Seconds\")\n",
    "plt.show()\n",
    "fig.savefig(\n",
    "    os.path.join(out_path, \"scrna_covid_flu_efficiency.svg\"),\n",
    "    format=\"svg\"\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
