{
 "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 matplotlib.pyplot as plt\n",
    "import matplotlib.font_manager as font_manager\n",
    "import os\n",
    "import h5py"
   ]
  },
  {
   "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": "c770c048",
   "metadata": {},
   "outputs": [],
   "source": [
    "latent_space = False\n",
    "latent_dim = 200"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "d5c1a69f",
   "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\n",
    "\n",
    "models_base_path = \"/data/anon/branched_diffusion/models/trained_models/scrna_covid_flu_continuous_class_extension\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "2cb7c2d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "branched_before_model_path = os.path.join(models_base_path, \"scrna_covid_flu_continuous_branched_2classes/1/last_ckpt.pth\")\n",
    "branched_after_model_path = os.path.join(models_base_path, \"extension/1/last_ckpt.pth\")\n",
    "label_guided_before_model_path = os.path.join(models_base_path, \"scrna_covid_flu_continuous_labelguided_2classes/1/last_ckpt.pth\")\n",
    "label_guided_afterone_model_path = os.path.join(models_base_path, \"extension/2/last_ckpt.pth\")\n",
    "label_guided_afterall_model_path = os.path.join(models_base_path, \"extension/4/last_ckpt.pth\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "93300a94",
   "metadata": {},
   "outputs": [],
   "source": [
    "num_trials = 5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "7ee483d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define the branches\n",
    "classes_01 = [0, 1]\n",
    "branch_defs_01 = [((0, 1), 0.5795795795795796, 1), ((0,), 0, 0.5795795795795796), ((1,), 0, 0.5795795795795796)]\n",
    "\n",
    "classes_012 = [0, 1, 5]\n",
    "branch_defs_012 = [((0, 1, 5), 6.786786786786787e-01, 1), ((0, 1), 0.5795795795795796, 0.6786786786786787), ((5,), 0, 0.6786786786786787), ((0,), 0, 0.5795795795795796), ((1,), 0, 0.5795795795795796)]\n",
    "\n",
    "classes_2 = [5]\n",
    "branch_defs_2 = [((5,), 0, 0.6786786786786787)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "0364c0a2",
   "metadata": {},
   "outputs": [],
   "source": [
    "out_path = \"/home/anon/branched_diffusion/figures/scrna_covid_flu_class_extension_2\"\n",
    "\n",
    "os.makedirs(out_path, exist_ok=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c77b6452",
   "metadata": {},
   "source": [
    "### Create data loaders"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "98285ad2",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset_01 = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=(autoencoder_path if latent_space else None))\n",
    "dataset_012 = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=(autoencoder_path if latent_space else None))\n",
    "dataset_2 = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=(autoencoder_path if latent_space else None))\n",
    "\n",
    "# Limit classes\n",
    "inds_01 = np.isin(dataset_01.cell_cluster, classes_01)\n",
    "dataset_01.data = dataset_01.data[inds_01]\n",
    "dataset_01.cell_cluster = dataset_01.cell_cluster[inds_01]\n",
    "inds_012 = np.isin(dataset_012.cell_cluster, classes_012)\n",
    "dataset_012.data = dataset_012.data[inds_012]\n",
    "dataset_012.cell_cluster = dataset_012.cell_cluster[inds_012]\n",
    "inds_2 = np.isin(dataset_2.cell_cluster, classes_2)\n",
    "dataset_2.data = dataset_2.data[inds_2]\n",
    "dataset_2.cell_cluster = dataset_2.cell_cluster[inds_2]\n",
    "\n",
    "data_loader_01 = torch.utils.data.DataLoader(dataset_01, batch_size=128, shuffle=True, num_workers=0)\n",
    "data_loader_012 = torch.utils.data.DataLoader(dataset_012, batch_size=128, shuffle=True, num_workers=0)\n",
    "data_loader_2 = torch.utils.data.DataLoader(dataset_2, batch_size=128, shuffle=True, num_workers=0)\n",
    "input_shape = next(iter(data_loader_01))[0].shape[1:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "5be5f5ec",
   "metadata": {},
   "outputs": [],
   "source": [
    "sde = sdes.VariancePreservingSDE(0.1, 5, input_shape)\n",
    "t_limit = 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "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)\n",
    "\n",
    "def class_to_class_index_tensor(c, classes):\n",
    "    \"\"\"\n",
    "    Given a tensor of classes, return the corresponding class indices\n",
    "    as a tensor.\n",
    "    \"\"\"\n",
    "    return torch.argmax(\n",
    "        (c[:, None] == torch.tensor(classes, device=c.device)).int(), dim=1\n",
    "    ).to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c35fc1e8",
   "metadata": {},
   "source": [
    "### Import models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "40e98368",
   "metadata": {},
   "outputs": [],
   "source": [
    "branched_before_model = model_util.load_model(\n",
    "    scrna_ae.MultitaskResNet, branched_before_model_path\n",
    ").to(DEVICE)\n",
    "branched_after_model = model_util.load_model(\n",
    "    scrna_ae.MultitaskResNet, branched_after_model_path\n",
    ").to(DEVICE)\n",
    "\n",
    "label_guided_before_model = model_util.load_model(\n",
    "    scrna_ae.LabelGuidedResNet, label_guided_before_model_path\n",
    ").to(DEVICE)\n",
    "label_guided_afterone_model = model_util.load_model(\n",
    "    scrna_ae.LabelGuidedResNet, label_guided_afterone_model_path\n",
    ").to(DEVICE)\n",
    "label_guided_afterall_model = model_util.load_model(\n",
    "    scrna_ae.LabelGuidedResNet, label_guided_afterall_model_path\n",
    ").to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0b38e89f",
   "metadata": {},
   "source": [
    "### Introducing a new digit class"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3738e63d",
   "metadata": {},
   "source": [
    "**Fine-tune new branch on branched model**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "4701b0f9",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.58it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.61it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.61it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.61it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.61it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.61it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.61it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    }
   ],
   "source": [
    "# Samples before fine-tuning\n",
    "all_branched_before_samples = []\n",
    "for i in range(num_trials):\n",
    "    branched_before_samples = {}\n",
    "    for class_to_sample in classes_01:\n",
    "        print(\"Sampling class: %s\" % class_to_sample)\n",
    "        samples = generate.generate_continuous_branched_samples(\n",
    "            branched_before_model, sde, class_to_sample,\n",
    "            lambda c, t: class_time_to_branch_tensor(c, t, branch_defs_01),\n",
    "            sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "        ).cpu().numpy()\n",
    "        branched_before_samples[class_to_sample] = samples\n",
    "    all_branched_before_samples.append(branched_before_samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "f5283fb3",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.59it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.59it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.59it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.59it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.60it/s]\n"
     ]
    }
   ],
   "source": [
    "# Samples after fine-tuning\n",
    "all_branched_after_samples = []\n",
    "for i in range(num_trials):\n",
    "    branched_after_samples = {}\n",
    "    for class_to_sample in classes_012:\n",
    "        print(\"Sampling class: %s\" % class_to_sample)\n",
    "        samples = generate.generate_continuous_branched_samples(\n",
    "            branched_after_model, sde, class_to_sample,\n",
    "            lambda c, t: class_time_to_branch_tensor(c, t, branch_defs_012),\n",
    "            sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "        ).cpu().numpy()\n",
    "        branched_after_samples[class_to_sample] = samples\n",
    "    all_branched_after_samples.append(branched_after_samples)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fe9b5ca7",
   "metadata": {},
   "source": [
    "#### Train additional digit on label-guided (linear) model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "a7395b41",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.76it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.76it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.76it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.76it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.76it/s]\n"
     ]
    }
   ],
   "source": [
    "all_label_guided_before_samples = []\n",
    "for i in range(num_trials):\n",
    "    label_guided_before_samples = {}\n",
    "    for class_to_sample in classes_012:\n",
    "        print(\"Sampling class: %s\" % class_to_sample)\n",
    "        samples = generate.generate_continuous_label_guided_samples(\n",
    "            label_guided_before_model, sde, class_to_sample,\n",
    "            lambda c: class_to_class_index_tensor(c, classes_012),\n",
    "            sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "        ).cpu().numpy()\n",
    "        label_guided_before_samples[class_to_sample] = samples\n",
    "    all_label_guided_before_samples.append(label_guided_before_samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "f5d9ec71",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.76it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.77it/s]\n"
     ]
    }
   ],
   "source": [
    "all_label_guided_afterone_samples = []\n",
    "for i in range(num_trials):\n",
    "    label_guided_afterone_samples = {}\n",
    "    for class_to_sample in classes_012:\n",
    "        print(\"Sampling class: %s\" % class_to_sample)\n",
    "        samples = generate.generate_continuous_label_guided_samples(\n",
    "            label_guided_afterone_model, sde, class_to_sample,\n",
    "            lambda c: class_to_class_index_tensor(c, classes_012),\n",
    "            sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "        ).cpu().numpy()\n",
    "        label_guided_afterone_samples[class_to_sample] = samples\n",
    "    all_label_guided_afterone_samples.append(label_guided_afterone_samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "8db51bf0",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.78it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.78it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.76it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.78it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.78it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████| 500/500 [00:56<00:00,  8.77it/s]\n"
     ]
    }
   ],
   "source": [
    "all_label_guided_afterall_samples = []\n",
    "for i in range(num_trials):\n",
    "    label_guided_afterall_samples = {}\n",
    "    for class_to_sample in classes_012:\n",
    "        print(\"Sampling class: %s\" % class_to_sample)\n",
    "        samples = generate.generate_continuous_label_guided_samples(\n",
    "            label_guided_afterall_model, sde, class_to_sample,\n",
    "            lambda c: class_to_class_index_tensor(c, classes_012),\n",
    "            sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "        ).cpu().numpy()\n",
    "        label_guided_afterall_samples[class_to_sample] = samples\n",
    "    all_label_guided_afterall_samples.append(label_guided_afterall_samples)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e5f431ba",
   "metadata": {},
   "source": [
    "**Compare FIDs**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "c2b292e5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 5\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 5\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 5\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 5\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 5\n"
     ]
    }
   ],
   "source": [
    "# Sample objects from the original dataset\n",
    "all_true_samples = []\n",
    "for _ in range(num_trials):\n",
    "    true_samples = {}\n",
    "    for class_to_sample in classes_012:\n",
    "        print(\"Sampling class: %s\" % class_to_sample)\n",
    "        inds = np.where(dataset_012.cell_cluster == class_to_sample)[0]\n",
    "        sample_inds = np.random.choice(inds, size=1000, replace=False)\n",
    "        true_samples[class_to_sample] = dataset_012.data[sample_inds]\n",
    "    all_true_samples.append(true_samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "988af0c2",
   "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": [
    "if not latent_space:\n",
    "    dataset_with_ae = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=autoencoder_path)\n",
    "\n",
    "def compute_fid(gen_samples, true_samples, latent=True):\n",
    "    if latent_space:\n",
    "        if latent:\n",
    "            return fid.compute_fid(\n",
    "                gen_samples,\n",
    "                dataset_01.encode_batch(torch.tensor(true_samples, device=DEVICE)).cpu().numpy()\n",
    "            )\n",
    "        else:\n",
    "            return fid.compute_fid(\n",
    "                dataset_01.decode_batch(torch.tensor(gen_samples, device=DEVICE)).cpu().numpy(),\n",
    "                true_samples\n",
    "            )\n",
    "    else:\n",
    "        gen_samples[gen_samples < 0] = 0  # Generated values should never be above 0\n",
    "        if latent:\n",
    "            return fid.compute_fid(\n",
    "                dataset_with_ae.encode_batch(torch.tensor(gen_samples, device=DEVICE)).cpu().numpy(),\n",
    "                dataset_with_ae.encode_batch(torch.tensor(true_samples, device=DEVICE)).cpu().numpy()\n",
    "            )\n",
    "        else:\n",
    "            return fid.compute_fid(\n",
    "                dataset_with_ae.decode_batch(dataset_with_ae.encode_batch(torch.tensor(gen_samples, device=DEVICE))).cpu().numpy(),\n",
    "                dataset_with_ae.decode_batch(dataset_with_ae.encode_batch(torch.tensor(true_samples, device=DEVICE))).cpu().numpy()\n",
    "            )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "e9cb6fa5",
   "metadata": {},
   "outputs": [],
   "source": [
    "all_branched_before_fids = []\n",
    "all_branched_after_fids = []\n",
    "all_label_guided_before_fids = []\n",
    "all_label_guided_afterone_fids = []\n",
    "all_label_guided_afterall_fids = []\n",
    "\n",
    "latent = True\n",
    "\n",
    "for i in range(num_trials):\n",
    "\n",
    "    branched_before_fids = {}\n",
    "    branched_after_fids = {}\n",
    "    label_guided_before_fids = {}\n",
    "    label_guided_afterone_fids = {}\n",
    "    label_guided_afterall_fids = {}\n",
    "\n",
    "    for c in branched_before_samples.keys():\n",
    "        branched_before_fids[c] = compute_fid(all_branched_before_samples[i][c], all_true_samples[i][c], latent)\n",
    "    for c in branched_after_samples.keys():\n",
    "        branched_after_fids[c] = compute_fid(all_branched_after_samples[i][c], all_true_samples[i][c], latent)\n",
    "    for c in label_guided_before_samples.keys():\n",
    "        label_guided_before_fids[c] = compute_fid(all_label_guided_before_samples[i][c], all_true_samples[i][c], latent)\n",
    "    for c in label_guided_afterone_samples.keys():\n",
    "        label_guided_afterone_fids[c] = compute_fid(all_label_guided_afterone_samples[i][c], all_true_samples[i][c], latent)\n",
    "    for c in label_guided_afterall_samples.keys():\n",
    "        label_guided_afterall_fids[c] = compute_fid(all_label_guided_afterall_samples[i][c], all_true_samples[i][c], latent)\n",
    "        \n",
    "    all_branched_before_fids.append(branched_before_fids)\n",
    "    all_branched_after_fids.append(branched_after_fids)\n",
    "    all_label_guided_before_fids.append(label_guided_before_fids)\n",
    "    all_label_guided_afterone_fids.append(label_guided_afterone_fids)\n",
    "    all_label_guided_afterall_fids.append(label_guided_afterall_fids)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "ed64d2b6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x360 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(ncols=2, sharey=True, figsize=(10, 5))\n",
    "\n",
    "bar_width = 0.3\n",
    "x = np.arange(len(all_branched_after_fids[0]))\n",
    "\n",
    "branched_before_vals = [np.mean([all_branched_before_fids[i][c] for i in range(num_trials)]) for c in classes_01] + [0]\n",
    "branched_before_errs = [np.std([all_branched_before_fids[i][c] for i in range(num_trials)]) / np.sqrt(num_trials) for c in classes_01] + [0]\n",
    "branched_after_vals = [np.mean([all_branched_after_fids[i][c] for i in range(num_trials)]) for c in classes_012]\n",
    "branched_after_errs = [np.std([all_branched_after_fids[i][c] for i in range(num_trials)]) / np.sqrt(num_trials) for c in classes_012]\n",
    "ax[0].bar(\n",
    "    x, branched_before_vals, yerr=branched_before_errs, width=bar_width, label=\"Before training on new class\", color=\"royalblue\"\n",
    ")\n",
    "ax[0].bar(\n",
    "    x + bar_width, branched_after_vals, yerr=branched_after_errs, width=bar_width, label=\"After training on new class\", color=\"slateblue\"\n",
    ")\n",
    "ax[0].set_title(\"Branched model\")\n",
    "\n",
    "label_guided_before_vals = [np.mean([all_label_guided_before_fids[i][c] for i in range(num_trials)]) for c in classes_01] + [0]\n",
    "label_guided_before_errs = [np.std([all_label_guided_before_fids[i][c] for i in range(num_trials)]) / np.sqrt(num_trials) for c in classes_01] + [0]\n",
    "label_guided_afterone_vals = [np.mean([all_label_guided_afterone_fids[i][c] for i in range(num_trials)]) for c in classes_012]\n",
    "label_guided_afterone_errs = [np.std([all_label_guided_afterone_fids[i][c] for i in range(num_trials)]) / np.sqrt(num_trials) for c in classes_012]\n",
    "label_guided_afterall_vals = [np.mean([all_label_guided_afterall_fids[i][c] for i in range(num_trials)]) for c in classes_012]\n",
    "label_guided_afterall_errs = [np.std([all_label_guided_afterall_fids[i][c] for i in range(num_trials)]) / np.sqrt(num_trials) for c in classes_012]\n",
    "\n",
    "ax[1].bar(\n",
    "    x, label_guided_before_vals, yerr=label_guided_before_errs, width=bar_width, label=\"Before training on new class\", color=\"darkorange\"\n",
    ")\n",
    "ax[1].bar(\n",
    "    x + bar_width, label_guided_afterone_vals, yerr=label_guided_afterone_errs, width=bar_width, label=\"After training on new class\", color=\"goldenrod\"\n",
    ")\n",
    "ax[1].bar(\n",
    "    x + (2 * bar_width), label_guided_afterall_vals, yerr=label_guided_afterall_errs, width=bar_width, label=\"After training on all classes\", color=\"firebrick\"\n",
    ")\n",
    "ax[1].set_title(\"Label-guided (linear) model\")\n",
    "for i in range(2):\n",
    "    ax[i].set_xticks(x + bar_width, labels=[c for c in classes_012])\n",
    "    ax[i].set_xlabel(\"Class\")\n",
    "    ax[i].legend()\n",
    "ax[0].set_ylabel(\"FID\")\n",
    "fig.suptitle(\"Fréchet inception distance between true and generated objects\")\n",
    "fig.savefig(\n",
    "    os.path.join(out_path, \"scrna_class_extension_fid.svg\"),\n",
    "    format=\"svg\"\n",
    ")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "2b681b47",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[17.837523056458895, 20.32130264684904, 0]\n",
      "[17.753223231183103, 20.81473603574572, 23.841423951100403]\n",
      "[20.864595807786934, 22.5730320142479, 0]\n",
      "[25.54976399408663, 27.357055498774304, 32.224359363520854]\n",
      "[18.581977572542094, 19.39291484138446, 27.391991357464594]\n"
     ]
    }
   ],
   "source": [
    "print(branched_before_vals)\n",
    "print(branched_after_vals)\n",
    "print(label_guided_before_vals)\n",
    "print(label_guided_afterone_vals)\n",
    "print(label_guided_afterall_vals)"
   ]
  }
 ],
 "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
}
