{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "99d811ee-0b10-45a2-b0b4-9eb77d03cc12",
   "metadata": {},
   "source": [
    "# ODE NHP Test\n",
    "\n",
    "Test ODE-GRU network with NHP loss on synthetic data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "5d15cc90-4432-414b-be82-2b68989faad8",
   "metadata": {},
   "outputs": [],
   "source": [
    "import math\n",
    "import numpy as np\n",
    "import torch\n",
    "import hotpp\n",
    "from hotpp.data import PaddedBatch\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "DEVICE = \"cuda\" if torch.cuda.is_available() else \"cpu\""
   ]
  },
  {
   "cell_type": "markdown",
   "id": "17f21a8e-d879-44c1-ac33-fce6c7d5d3d0",
   "metadata": {},
   "source": [
    "# Dataset\n",
    "\n",
    "Synthetic dataset with a predefined distribution of time delta."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "a6c94e50-838b-4ccd-8c09-9a9f4711ae9a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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2+lSAm3x4sXavHmzrrjiCh09elRr167VDTfcoPfee0+TJk0a8fO4NqAAOL/Vq1frvffeG9X4LtLjiiuuUGNjozo7O/XCCy9o5cqVev311wkp46ypqUn33XefXn31VeXn5ztdTqAtXbo09d9z587V4sWLdckll+j555+3NfTp2oAybdo0ZWdnq6WlZdDxlpYWzZgxw6GqAPeorKzUf/7nf2rXrl26+OKLnS4nsPLy8vT5z39ekrRgwQK9/fbbevLJJ/XjH//Y4cqCpaGhQa2trbr22mtTxwzD0K5du/TUU08pFospOzvbwQqDa/Lkybr88su1f/9+W9e5dg5KXl6eFixYoJ07d6aOmaapnTt3Mr6LQLMsS5WVlXrxxRf12muv6dJLL3W6JJzGNE3FYjGnywicr371q3r33XfV2NiYeixcuFB33nmnGhsbCScO6u7u1scff6yLLrrI1nWu7aBIUlVVlVauXKmFCxdq0aJFWr9+vaLRqFatWuV0aYHT3d09KP0ePHhQjY2Nmjp1qmbPnu1gZcGzevVqbdmyRb/+9a81adIkNTc3S5KKi4tVUFDgcHXBUl1draVLl2r27Nnq6urSli1bVFdXp1deecXp0gJn0qRJZ83DmjBhgi644ALmZ42zb3/721q2bJkuueQSffbZZ6qpqVF2drZuv/12W8/j6oCyfPlytbW1ac2aNWpubta8efO0Y8eOsybOIvP27Nmjm266KfV9VVWVJGnlypXavHmzQ1UF0zPPPCNJuvHGGwcd/9nPfqa/+Iu/GP+CAqy1tVUrVqzQ0aNHVVxcrLlz5+qVV17RzTff7HRpgGM+/fRT3X777Tp27JguvPBCfelLX9Jbb72lCy+80NbzuHofFAAAEEyunYMCAACCi4ACAABch4ACAABch4ACAABch4ACAABch4ACAABch4ACAABch4ACAABch4ACAABch4ACAABch4ACAABch4ACAABc5/8DFqBbltEvSRAAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class DTDistribution:\n",
    "    def __init__(self, label=0):\n",
    "        if label == 0:\n",
    "            self.means = torch.tensor([1, 3]).float()\n",
    "            self.stds = torch.tensor([0.2, 1])\n",
    "            self.probs = torch.tensor([0.7, 0.3])\n",
    "        elif label == 1:\n",
    "            self.means = torch.tensor([2]).float()\n",
    "            self.stds = torch.tensor([0.5])\n",
    "            self.probs = torch.tensor([1.0])\n",
    "        else:\n",
    "            raise NotImplementedError(label)\n",
    "\n",
    "    @property\n",
    "    def entropy(self):\n",
    "        n = 10000\n",
    "        s = self.sample(10000)\n",
    "        return -self.log_pdf(s).mean()\n",
    "\n",
    "    @property\n",
    "    def expectation(self):\n",
    "        return (self.means * self.probs).sum()\n",
    "\n",
    "    def log_pdf(self, dt):\n",
    "        nc = len(self.means)\n",
    "        shape = [nc] + [1] * dt.ndim\n",
    "        means = self.means.reshape(*shape)\n",
    "        stds = self.stds.reshape(*shape)\n",
    "        probs = self.probs.reshape(*shape)\n",
    "        log_pdfs = -0.5 * (2 * math.pi * stds ** 2).log() - (means - dt) ** 2 / 2 / stds ** 2\n",
    "        return torch.logsumexp(log_pdfs + probs.log(), 0)\n",
    "\n",
    "    def sample(self, size):\n",
    "        components = torch.multinomial(self.probs, size, replacement=True)  # S.\n",
    "        sample = (torch.randn(size, 1) * self.stds + self.means).take_along_dim(components[:, None], 1).squeeze(1)\n",
    "        return sample.clip(min=0)\n",
    "\n",
    "    def plot(self, sample_size=20, ax=None):\n",
    "        if ax is None:\n",
    "            ax = plt.gca()\n",
    "        x_min = 0\n",
    "        x_max = self.means.max().item() + self.stds.max().item() * 2\n",
    "        ax.set_xlim(x_min, x_max)\n",
    "        if sample_size:\n",
    "            sample = self.sample(sample_size)\n",
    "            for x in sample:\n",
    "                ax.axvline(x, 0, 1, linewidth=0.5)\n",
    "\n",
    "        xs = torch.linspace(x_min, x_max, 1000)\n",
    "        pdfs = self.log_pdf(xs).exp()\n",
    "        # Fix filling.\n",
    "        xs[-1] = xs[-2]\n",
    "        pdfs[-1] = 0\n",
    "        ax.fill(xs, pdfs, alpha=0.5)\n",
    "\n",
    "DTDistribution().plot()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0b2e9b7d-db89-47ec-86e7-bdc5b836acfe",
   "metadata": {},
   "source": [
    "**Create batch**\n",
    "\n",
    "Each sequences is composed of two timestamps: 0 and sampled. All labels are identical."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "3bb36262-6d3b-4d3e-87ad-143727d5ac24",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([20000, 2])\n"
     ]
    }
   ],
   "source": [
    "sample_size = 10000\n",
    "num_labels = 2\n",
    "\n",
    "time_deltas = torch.cat([DTDistribution(label).sample(sample_size).reshape(sample_size)\n",
    "                         for label in range(num_labels)])  # (B).\n",
    "time_deltas = torch.stack([torch.zeros_like(time_deltas), time_deltas], 1)  # (B, 2).\n",
    "\n",
    "labels = torch.cat([torch.full([sample_size], label, dtype=torch.long)\n",
    "                    for label in range(num_labels)])  # (B).\n",
    "labels = labels[:, None].repeat(1, 2)  # (B, 2).\n",
    "\n",
    "batch = PaddedBatch(\n",
    "    {\n",
    "        \"timestamps\": time_deltas.cumsum(1),\n",
    "        \"labels\": labels\n",
    "    },\n",
    "    torch.full([len(time_deltas)], 2, dtype=torch.long)\n",
    ")\n",
    "print(batch.shape)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c9ca3299-b5cc-4619-bc51-cbec20485827",
   "metadata": {},
   "source": [
    "# Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "d31f3965-9ebe-4a96-8eda-ef5cf26335c9",
   "metadata": {},
   "outputs": [],
   "source": [
    "module = hotpp.modules.NextItemModule(\n",
    "    seq_encoder=hotpp.nn.RnnEncoder(\n",
    "        embeddings={\"labels\": {\"in\": num_labels, \"out\": 16}},\n",
    "        rnn_partial=lambda dim: hotpp.nn.ODEGRU(dim, 16, num_diff_layers=3),\n",
    "    ),\n",
    "    head_partial=lambda idim, odim: hotpp.nn.Head(idim, odim, hidden_dims=[16]),\n",
    "    loss=hotpp.losses.NHPLoss(num_labels, thinning_params={\"max_delta\": 5, \"max_steps\": 10000}),\n",
    "    optimizer_partial=lambda p: torch.optim.Adam(p, lr=0.001)\n",
    ").to(DEVICE).eval()\n",
    "\n",
    "batch = batch.to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9d6c931c-fb6a-4a42-98b1-237204a66efc",
   "metadata": {},
   "source": [
    "# Train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "cf01141a-998f-4ef7-b46f-d40bfa43b8fc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Mean entropy (minimum expected loss): 0.765\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "You are trying to `self.log()` but the `self.trainer` reference is not registered on the model yet. This is most likely because the model hasn't been passed to the `Trainer`\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Step 0, Loss 2.32, Grad 1.09\n",
      "Step 200, Loss 1.00, Grad 0.23\n",
      "Step 400, Loss 0.88, Grad 0.10\n",
      "Step 600, Loss 0.79, Grad 0.15\n",
      "Step 800, Loss 0.81, Grad 0.24\n"
     ]
    }
   ],
   "source": [
    "entropy = np.mean([DTDistribution(label).entropy.item() for label in range(num_labels)])\n",
    "print(f\"Mean entropy (minimum expected loss): {entropy:.3f}\")\n",
    "\n",
    "module.train()\n",
    "optimizer = module.configure_optimizers()\n",
    "for step in range(1000):\n",
    "    loss = module.training_step((batch, None), None)\n",
    "    optimizer.zero_grad()\n",
    "    loss.backward()\n",
    "    grad_norm = torch.nn.utils.clip_grad_norm_(module.parameters(), 1)\n",
    "    if step % 200 == 0:\n",
    "        print(f\"Step {step}, Loss {loss.item():.2f}, Grad {grad_norm.item():.2f}\")\n",
    "    optimizer.step()\n",
    "module.eval()\n",
    "pass"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e680c6c9-4ac4-4540-8a32-ddf8f49b8292",
   "metadata": {},
   "source": [
    "# Visualize results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "aacc01cb-a14f-4cd4-8db9-2f87431c48ef",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Label 0\n",
      "Expected time delta: 1.600000023841858\n",
      "Predicted time delta: 1.602463722229004\n",
      "Label 1\n",
      "Expected time delta: 2.0\n",
      "Predicted time delta: 2.038818597793579\n"
     ]
    }
   ],
   "source": [
    "label_states = []\n",
    "for label in range(num_labels):\n",
    "    i = label * sample_size\n",
    "    prefix = PaddedBatch({\n",
    "        \"timestamps\": batch.payload[\"timestamps\"][i:i + 1, :1],\n",
    "        \"labels\": batch.payload[\"labels\"][i:i + 1, :1]\n",
    "    }, torch.ones_like(batch.seq_lens[:1]))\n",
    "    assert prefix.payload[\"labels\"].item() == label\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        outputs = module(prefix)\n",
    "        label_states.append(module.encode(prefix)[1])\n",
    "    print(f\"Label {label}\")\n",
    "    print(\"Expected time delta:\", DTDistribution(label).expectation.item())\n",
    "    print(\"Predicted time delta:\", outputs.payload[\"timestamps\"].item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "efa2e149-8c2b-41c1-b1ff-0752809210f7",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Density:\n",
    "    \"\"\"Simple intensity-to-density mapper.\"\"\"\n",
    "    def __init__(self, intensity, sample_size=100):\n",
    "        self.i = intensity\n",
    "        self.s = torch.linspace(0, 1, sample_size)\n",
    "\n",
    "    def __call__(self, deltas):\n",
    "        means = self.i(deltas[..., None] * self.s.to(deltas.device)).mean(-1)  # (...).\n",
    "        integrals = means * deltas\n",
    "        log_values = self.i(deltas).log() - integrals\n",
    "        values = log_values.exp()\n",
    "        return values\n",
    "\n",
    "class IntensityFN:\n",
    "    \"\"\"Intensity function wrapper.\"\"\"\n",
    "    def __init__(self, deltas, intensities):\n",
    "        self.deltas = deltas.cpu().numpy()\n",
    "        self.intensities = intensities.cpu().numpy()\n",
    "\n",
    "    def __call__(self, deltas):\n",
    "        values = np.interp(deltas.cpu().numpy(), self.deltas, self.intensities)\n",
    "        values = torch.tensor(values, device=deltas.device, dtype=deltas.dtype)\n",
    "        return values\n",
    "\n",
    "def get_density(deltas, intensities):\n",
    "    \"\"\"Convert intensity to density.\"\"\"\n",
    "    return Density(IntensityFN(deltas, intensities))(deltas)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "5f6ebac7-fa86-4cf7-9354-3c1376269c27",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiEAAAGzCAYAAAD5UcdSAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/TGe4hAAAACXBIWXMAAA9hAAAPYQGoP6dpAABL90lEQVR4nO3dd3QU9f7/8edm0zstJEBoht67gBQBQUSuoNguV8F2LejVy88C36uA1+vF3lFRBFSuoiBF6UhTVEBAkCYC0gktQDZ1k+zO74+BxEjAJOzubJLX45w9zExmd9472bN5MfMpNsMwDERERER8LMDqAkRERKRiUggRERERSyiEiIiIiCUUQkRERMQSCiEiIiJiCYUQERERsYRCiIiIiFhCIUREREQsoRAiIiIillAIEZE/tW/fPmw2Gy+99JLHXnPlypXYbDZWrlzpsdcUkbJFIUSknJo6dSo2m43169dbXcolSU9PZ+zYsVx99dVUrlwZm83G1KlTrS5LRDxAIURE/NrJkyf597//zY4dO2jVqpXV5YiIBwVaXYCIyMUkJCSQnJxMfHw869evp0OHDlaXJCIeoishIhVYTk4OY8aMoV27dsTExBAREUG3bt1YsWLFBZ/z6quvUqdOHcLCwujRowdbt249b59ffvmFIUOGULlyZUJDQ2nfvj1ffvllqWoMCQkhPj6+VM8VEf+mKyEiFZjD4WDSpEnceuut3HPPPaSlpfHBBx/Qr18/1q1bR+vWrQvt/9FHH5GWlsaIESPIzs7m9ddfp1evXmzZsoXq1asDsG3bNrp27UrNmjUZNWoUERERfP755wwaNIgvvviCwYMHW/BORcQfKYSIVGCVKlVi3759BAcH52+75557aNy4MW+++SYffPBBof13797Nrl27qFmzJgBXX301nTp14vnnn+eVV14B4OGHH6Z27dr8+OOPhISEAPDAAw9wxRVX8MQTTyiEiEg+3Y4RqcDsdnt+AHG73Zw6dYq8vDzat2/Pxo0bz9t/0KBB+QEEoGPHjnTq1IkFCxYAcOrUKZYvX85NN91EWloaJ0+e5OTJk6SkpNCvXz927drF4cOHffPmRMTvKYSIVHAffvghLVu2JDQ0lCpVqlCtWjXmz59Pamrqefs2aNDgvG0NGzZk3759gHmlxDAMnnrqKapVq1boMXbsWACOHz/u1fcjImWHbseIVGDTpk1j+PDhDBo0iMcee4y4uDjsdjvjx49nz549JX49t9sNwKOPPkq/fv2K3CcpKemSahaR8kMhRKQCmzlzJvXr12fWrFnYbLb87eeuWvzRrl27ztv266+/UrduXQDq168PQFBQEH369PF8wSJSruh2jEgFZrfbATAMI3/b2rVr+eGHH4rcf86cOYXadKxbt461a9fSv39/AOLi4ujZsycTJ04kOTn5vOefOHHCk+WLSBmnKyEi5dzkyZNZtGjRedsffvhhrr32WmbNmsXgwYMZMGAAe/fu5d1336Vp06akp6ef95ykpCSuuOIK7r//fpxOJ6+99hpVqlTh8ccfz99nwoQJXHHFFbRo0YJ77rmH+vXrc+zYMX744QcOHTrE5s2bS/we3nrrLc6cOcORI0cA+Oqrrzh06BAADz30EDExMSV+TRGxnkKISDn3zjvvFLl9+PDhDB8+nKNHjzJx4kQWL15M06ZNmTZtGjNmzChyYrnbb7+dgIAAXnvtNY4fP07Hjh156623SEhIyN+nadOmrF+/nqeffpqpU6eSkpJCXFwcbdq0YcyYMaV6Dy+99BL79+/PX581axazZs0C4G9/+5tCiEgZZTN+fx1WRERExEfUJkREREQsoRAiIiIillAIEREREUsohIiIiIglFEJERETEEgohIiIiYgmfjxPidrs5cuQIUVFRhYaJFhEREf9lGAZpaWnUqFGDgADPXMPweQg5cuQIiYmJvj6siIiIeMDBgwepVauWR17L5yEkKioKMN9EdHS0rw8vIiIipeBwOEhMTMz/O+4JPg8h527BREdHK4SIiIiUMZ5sSqGGqSIiImIJhRARERGxhEKIiIiIWMLnbUJERKRiMAyDvLw8XC6X1aVIMdjtdgIDA306fIZCiIiIeFxOTg7JyclkZmZaXYqUQHh4OAkJCQQHB/vkeAohIiLiUW63m71792K326lRowbBwcEanNLPGYZBTk4OJ06cYO/evTRo0MBjA5JdjEKIiIh4VE5ODm63m8TERMLDw60uR4opLCyMoKAg9u/fT05ODqGhoV4/phqmioiIV/jif9LiWb7+nekTIiIiIpZQCBERERFLlCiEjBs3DpvNVujRuHFjb9UmIiIif7Bv3z5sNhubNm2yupRLVuIrIc2aNSM5OTn/sXr1am/UJSIi4nPDhw9n0KBBxd7fZrMxZ84c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zeoJUwEphIiIiHVGj4Yvv4SQEJgzRz1hKhiFEBERscbUqfDCC+byBx9A586WliO+pxAiIiK+9+238Pe/m8tPPmnekpEKRyFERER867ffYPBgyM2FIUPg6aetrkgsohAiIiK+k5oKAwdCSgq0awcffqieMBWYfvMiIuIbeXlwyy2wfTvUqAFz50J4uNVViYUUQkRExDcefRQWLYKwMLNHTM2aVlckFlMIERER73v3XXj9dXP5o4/MWzFS4SmEiIiIdy1ZAg8+aC4/84zZGFUEhRAREfGmbdvgxhvB5TInpPvXv6yuSPyIQoiIiHjHsWMwYAA4HNC9O7z3HthsVlclfkQhREREPC8ry5wVd/9+SEqCWbPModlFfkchREREPMvthjvugDVroFIlmDcPqlSxuirxQwohIiLiWWPHwmefQWCgeQWkUSOrKxI/pRAiIiKe89FH8J//mMvvvQc9e1pajvg3hRAREfGMb76Bu+82l0ePNm/JiFyEQoiIiFy63bsLT0p37mqIyEUohIiIyKU5eRKuuQZOnYKOHc1bMpqUTopBnxIRESm97GyzK+6uXVCnjjkpXViY1VVJGaEQIiIipeN2m6OgfvcdxMTAggUQH291VVKGKISIiEjpjBoFM2ZAUBDMng1Nm1pdkZQxCiEiIlJy77wDL75oLk+eDFdeaW09UiYphIiISMnMn194Vty//c3aeqTMUggREZHi27ABbr7ZbA9y552aFVcuiUKIiIgUz/79cO21kJEBV10F776rWXHlkiiEiIjInztzxhwL5OhRaNECZs40G6SKXAKFEBERubicHLjhBti+HWrUMNuEREdbXZWUAwohIiJyYW43DB8Oy5dDZKQZQBITra5KygmFEBERubDHH4dPP4XAQJg1C1q3troiKUcUQkREpGivvgovv2wuT55sNkYV8SCFEBEROd/06TBypLn83HNw223W1iPlkkKIiIgUtmIFDBtmLj/0kHlLRsQLFEJERKTA5s3mrLg5OTBkiHlLRmOBiJcohIiIiGn/fujfHxwO6N4dPv4Y7Harq5JyTCFERETg1CkzgCQnQ7NmMGcOhIZaXZWUcwohIiIVXVYW/OUvsGMH1KoFixZBpUpWVyUVgEKIiEhFlpdnTkj33XcQG2sGkFq1rK5KKgiFEBGRisrthrvvhq++Mm+9fPmleStGxEdKFELGjx9P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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for label, state in enumerate(label_states):\n",
    "    inter_times = torch.linspace(0, 5, 100)\n",
    "    inter_times = PaddedBatch(inter_times[None, None], torch.tensor([1])).to(DEVICE)\n",
    "    with torch.no_grad():\n",
    "        inter = module._loss.interpolator(state, inter_times).payload  # (B, L, S, D).\n",
    "        intensity = module._loss.intensity(inter)[0, 0, :, label]\n",
    "        density = get_density(inter_times.payload[0, 0], intensity)\n",
    "\n",
    "    plt.title(f\"Label {label}\")\n",
    "    DTDistribution(label).plot(sample_size=None)\n",
    "    plt.axhline(0, color=\"k\", linewidth=0.5)\n",
    "    plt.plot(inter_times.payload[0, 0].cpu(), intensity.cpu(), label=\"Intensity\", c=\"red\")\n",
    "    plt.plot(inter_times.payload[0, 0].cpu(), density.cpu(), label=\"Density\", c=\"blue\")\n",
    "    plt.legend()\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e9afe28a-61ef-4730-818c-1edf07fe6705",
   "metadata": {},
   "source": [
    "# Conclusion\n",
    "* Don't use batchnorm in the models' head"
   ]
  }
 ],
 "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.12.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
