{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "b3960d83-1962-4f28-bdbc-4ebf6caf3927",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2025-05-06 17:32:05.721526: I external/local_xla/xla/tsl/cuda/cudart_stub.cc:32] Could not find cuda drivers on your machine, GPU will not be used.\n",
      "2025-05-06 17:32:05.731640: I external/local_xla/xla/tsl/cuda/cudart_stub.cc:32] Could not find cuda drivers on your machine, GPU will not be used.\n",
      "2025-05-06 17:32:05.764308: E external/local_xla/xla/stream_executor/cuda/cuda_fft.cc:467] Unable to register cuFFT factory: Attempting to register factory for plugin cuFFT when one has already been registered\n",
      "WARNING: All log messages before absl::InitializeLog() is called are written to STDERR\n",
      "E0000 00:00:1746545525.798441   57101 cuda_dnn.cc:8579] Unable to register cuDNN factory: Attempting to register factory for plugin cuDNN when one has already been registered\n",
      "E0000 00:00:1746545525.808256   57101 cuda_blas.cc:1407] Unable to register cuBLAS factory: Attempting to register factory for plugin cuBLAS when one has already been registered\n",
      "W0000 00:00:1746545525.833243   57101 computation_placer.cc:177] computation placer already registered. Please check linkage and avoid linking the same target more than once.\n",
      "W0000 00:00:1746545525.833272   57101 computation_placer.cc:177] computation placer already registered. Please check linkage and avoid linking the same target more than once.\n",
      "W0000 00:00:1746545525.833275   57101 computation_placer.cc:177] computation placer already registered. Please check linkage and avoid linking the same target more than once.\n",
      "W0000 00:00:1746545525.833277   57101 computation_placer.cc:177] computation placer already registered. Please check linkage and avoid linking the same target more than once.\n",
      "2025-05-06 17:32:05.841161: I tensorflow/core/platform/cpu_feature_guard.cc:210] This TensorFlow binary is optimized to use available CPU instructions in performance-critical operations.\n",
      "To enable the following instructions: AVX2 FMA, in other operations, rebuild TensorFlow with the appropriate compiler flags.\n"
     ]
    }
   ],
   "source": [
    "import sys; sys.path.append(\"../src/\")\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import kernel\n",
    "from tqdm import tqdm\n",
    "from itertools import chain\n",
    "import pandas as pd\n",
    "import data\n",
    "import pandas as pd"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "2f0bce76-f643-423b-8c1c-3189fcd1b448",
   "metadata": {},
   "outputs": [],
   "source": [
    "thresholds = pd.read_csv(\"../results/mnist_0_thresholds.csv\", index_col=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "dd5a3c39-9a34-4a4b-8e4c-ee415c5341ac",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>log ARL</th>\n",
       "      <th>threshold</th>\n",
       "      <th>d</th>\n",
       "      <th>data</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>3.00</td>\n",
       "      <td>1.210154</td>\n",
       "      <td>784</td>\n",
       "      <td>MNIST 0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>3.25</td>\n",
       "      <td>1.231540</td>\n",
       "      <td>784</td>\n",
       "      <td>MNIST 0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>3.50</td>\n",
       "      <td>1.249811</td>\n",
       "      <td>784</td>\n",
       "      <td>MNIST 0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>3.75</td>\n",
       "      <td>1.264903</td>\n",
       "      <td>784</td>\n",
       "      <td>MNIST 0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>4.00</td>\n",
       "      <td>1.280195</td>\n",
       "      <td>784</td>\n",
       "      <td>MNIST 0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>4.25</td>\n",
       "      <td>1.294077</td>\n",
       "      <td>784</td>\n",
       "      <td>MNIST 0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>4.50</td>\n",
       "      <td>1.309564</td>\n",
       "      <td>784</td>\n",
       "      <td>MNIST 0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>4.75</td>\n",
       "      <td>1.329522</td>\n",
       "      <td>784</td>\n",
       "      <td>MNIST 0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>5.00</td>\n",
       "      <td>1.344752</td>\n",
       "      <td>784</td>\n",
       "      <td>MNIST 0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   log ARL  threshold    d     data\n",
       "0     3.00   1.210154  784  MNIST 0\n",
       "1     3.25   1.231540  784  MNIST 0\n",
       "2     3.50   1.249811  784  MNIST 0\n",
       "3     3.75   1.264903  784  MNIST 0\n",
       "4     4.00   1.280195  784  MNIST 0\n",
       "5     4.25   1.294077  784  MNIST 0\n",
       "6     4.50   1.309564  784  MNIST 0\n",
       "7     4.75   1.329522  784  MNIST 0\n",
       "8     5.00   1.344752  784  MNIST 0"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "thresholds"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "65c85922-6f76-4f74-842c-056d729c0c34",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "bd160d24-7d86-46fa-82dc-1bf2866efd7f",
   "metadata": {},
   "outputs": [],
   "source": [
    "rng = np.random.default_rng(1234)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "1dcf5db1-5f51-4fe1-a772-e8a1c8a47895",
   "metadata": {},
   "outputs": [],
   "source": [
    "d=784"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "4aec4fec-9a67-4eb6-b0b6-9a0d54aa8aa4",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_pre_change = 2**6\n",
    "mnist0 = data.MNIST(n_pre_change, digit=0)\n",
    "mnist1 = data.MNIST(2**8, digit=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "db910d84-52b2-41ce-aa54-e0f676a82c94",
   "metadata": {},
   "outputs": [],
   "source": [
    "gamma = 1.6319519213908345e-07 # from ARL computation\n",
    "gauss = kernel.Gauss(gamma=gamma)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "8419aa02-dc35-4cda-9b11-06bbc38b1f90",
   "metadata": {},
   "outputs": [],
   "source": [
    "target_arls = [100,1000,10000]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "f2ff411b-8292-4686-b9a6-c12f102ca4e8",
   "metadata": {},
   "outputs": [],
   "source": [
    "num_omegas = 1000"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "14466b77-9944-4e8a-be6a-58bacdbdb02d",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 21%|████████▊                                 | 21/100 [00:30<01:56,  1.47s/it]\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[11], line 6\u001b[0m\n\u001b[1;32m      4\u001b[0m cd \u001b[38;5;241m=\u001b[39m kernel\u001b[38;5;241m.\u001b[39mStreamingRFFMMD(gauss,d\u001b[38;5;241m=\u001b[39md,num_omegas\u001b[38;5;241m=\u001b[39mnum_omegas)\n\u001b[1;32m      5\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m i, elem \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28menumerate\u001b[39m(np\u001b[38;5;241m.\u001b[39mconcatenate((mnist0\u001b[38;5;241m.\u001b[39mdraw(),mnist1\u001b[38;5;241m.\u001b[39mdraw()))):\n\u001b[0;32m----> 6\u001b[0m     \u001b[43mcd\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43minsert\u001b[49m\u001b[43m(\u001b[49m\u001b[43melem\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m      7\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m i \u001b[38;5;241m>\u001b[39m\u001b[38;5;241m=\u001b[39m n_pre_change \u001b[38;5;129;01mand\u001b[39;00m (np\u001b[38;5;241m.\u001b[39marray(cd\u001b[38;5;241m.\u001b[39mnormalized_mmd()) \u001b[38;5;241m>\u001b[39m\u001b[38;5;241m=\u001b[39m \u001b[38;5;241m1.210154\u001b[39m)\u001b[38;5;241m.\u001b[39many():\n\u001b[1;32m      8\u001b[0m         results \u001b[38;5;241m=\u001b[39m pd\u001b[38;5;241m.\u001b[39mconcat((results,pd\u001b[38;5;241m.\u001b[39mDataFrame({\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mnum_omegas\u001b[39m\u001b[38;5;124m\"\u001b[39m : [num_omegas],\n\u001b[1;32m      9\u001b[0m                                                        \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mtarget_arl\u001b[39m\u001b[38;5;124m\"\u001b[39m : [\u001b[38;5;241m100\u001b[39m],\n\u001b[1;32m     10\u001b[0m                                                        \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mdelay\u001b[39m\u001b[38;5;124m\"\u001b[39m      : [i\u001b[38;5;241m-\u001b[39mn_pre_change]})))\n",
      "File \u001b[0;32m~/Documents/rff-mmd-change-detection/code/notebooks/../src/kernel.py:80\u001b[0m, in \u001b[0;36mStreamingRFFMMD.insert\u001b[0;34m(self, element)\u001b[0m\n\u001b[1;32m     78\u001b[0m element \u001b[38;5;241m=\u001b[39m element\u001b[38;5;241m.\u001b[39mreshape(\u001b[38;5;241m1\u001b[39m,\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m)\n\u001b[1;32m     79\u001b[0m tmp \u001b[38;5;241m=\u001b[39m element \u001b[38;5;241m@\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39momegas\u001b[38;5;241m.\u001b[39mT\n\u001b[0;32m---> 80\u001b[0m z \u001b[38;5;241m=\u001b[39m \u001b[38;5;241m1\u001b[39m\u001b[38;5;241m/\u001b[39m\u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msqrt\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mnum_omegas\u001b[49m\u001b[43m)\u001b[49m \u001b[38;5;241m*\u001b[39m np\u001b[38;5;241m.\u001b[39mconcatenate((np\u001b[38;5;241m.\u001b[39mcos(tmp), np\u001b[38;5;241m.\u001b[39msin(tmp)),axis\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m1\u001b[39m)\n\u001b[1;32m     81\u001b[0m W \u001b[38;5;241m=\u001b[39m StreamingRFFMMD\u001b[38;5;241m.\u001b[39mWindow(z\u001b[38;5;241m=\u001b[39mz,c\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m1\u001b[39m)\n\u001b[1;32m     82\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mwindows\u001b[38;5;241m.\u001b[39mappend(W)\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "results = pd.DataFrame()\n",
    "runs=100\n",
    "for _ in tqdm(range(runs)):\n",
    "    cd = kernel.StreamingRFFMMD(gauss,d=d,num_omegas=num_omegas)\n",
    "    for i, elem in enumerate(np.concatenate((mnist0.draw(),mnist1.draw()))):\n",
    "        cd.insert(elem)\n",
    "        if i >= n_pre_change and (np.array(cd.normalized_mmd()) >= 1.210154).any():\n",
    "            results = pd.concat((results,pd.DataFrame({\"num_omegas\" : [num_omegas],\n",
    "                                                           \"target_arl\" : [100],\n",
    "                                                           \"delay\"      : [i-n_pre_change]})))\n",
    "            break\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "34b52fd9-d853-4963-862a-9ff2f6cc40ca",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>num_omegas</th>\n",
       "      <th>target_arl</th>\n",
       "      <th>delay</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>3</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1000</td>\n",
       "      <td>100</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   num_omegas  target_arl  delay\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      3\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      1\n",
       "0        1000         100      1\n",
       "0        1000         100      2\n",
       "0        1000         100      2\n",
       "0        1000         100      2"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "6e54c0a3-36b4-472a-80c6-a73a948ef58e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th>delay</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>num_omegas</th>\n",
       "      <th>target_arl</th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th rowspan=\"3\" valign=\"top\">1000</th>\n",
       "      <th>100</th>\n",
       "      <td>2969.272727</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1000</th>\n",
       "      <td>2983.818182</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>10000</th>\n",
       "      <td>2907.800000</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                             delay\n",
       "num_omegas target_arl             \n",
       "1000       100         2969.272727\n",
       "           1000        2983.818182\n",
       "           10000       2907.800000"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "results.groupby([\"num_omegas\", \"target_arl\"]).mean()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "eb248311-40bc-4dd1-86fb-f3e1f988f02f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[np.float64(20.00295508656015),\n",
       " np.float64(12.870418543852852),\n",
       " np.float64(4.4836206462003),\n",
       " np.float64(1.8567048818779173),\n",
       " np.float64(1.1082299581799395)]"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "cd.normalized_mmd()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "019ea2ce-c95e-4021-b55a-dd89fc1e6a2e",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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
}
