{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "c81572c3",
   "metadata": {},
   "source": [
    "Thank you for checking out the code! This code is for the Neurips 2025 camera ready version only, intending to showcase the method and the main result. The full repository will be released in the future on https://github.com/wei-bbc-lab/DVC.\n",
    "The environment file is included along with the package. However it is advised to install only the necessary packages as you go along.\n",
    "This code will attempt to save .pkl files for the model activations. Please make sure there is enough space (~10GB)\n",
    "\n",
    "DVC_experiment.ipynb = this file\n",
    "DVC.py = file for the DVC method\n",
    "environment.yml = environment file\n",
    "category_ids.pkl = file of stimuli_ids to make sure the stimuli is loaded in order"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "33c66169",
   "metadata": {},
   "source": [
    "### Get Monkey Recordings (from brainscore_vision)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6409a2a7",
   "metadata": {},
   "outputs": [],
   "source": [
    "import brainscore_vision\n",
    "from brainscore_vision.benchmark_helpers.neural_common import average_repetition\n",
    "\n",
    "\n",
    "import numpy as np\n",
    "\n",
    "from sklearn.decomposition import PCA\n",
    "from matplotlib.backends.backend_pdf import PdfPages\n",
    "\n",
    "%matplotlib inline\n",
    "from matplotlib import image\n",
    "import matplotlib.pyplot as plt\n",
    "import tqdm\n",
    "import pickle\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c6010bc2",
   "metadata": {},
   "outputs": [],
   "source": [
    "neural_data = brainscore_vision.load_dataset(\"MajajHong2015.public\")\n",
    "neural_data = neural_data.transpose('presentation', 'neuroid', 'time_bin')\n",
    "\n",
    "categories = set(neural_data['category_name'].values)\n",
    "num_categories = len(categories)\n",
    "stimulus_set = neural_data.attrs['stimulus_set']\n",
    "\n",
    "# This is important! The image_ids are stored so they are retrieved in the same order every time.\n",
    "with open('category_ids.pkl', 'rb') as f:\n",
    "    category_ids = pickle.load(f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f50d3308",
   "metadata": {},
   "outputs": [],
   "source": [
    "Chabo_recordings = {}\n",
    "for category in categories:\n",
    "    Chabo_recordings[category] = []\n",
    "    for i in category_ids[category]:\n",
    "        data = neural_data.sel(animal = 'Chabo', category_name = category, stimulus_id = i)\n",
    "        data = average_repetition(data)\n",
    "        data = data.squeeze('time_bin')\n",
    "        Chabo_recordings[category].append(data.values)\n",
    "    Chabo_recordings[category] = np.stack(Chabo_recordings[category], axis=0).squeeze()\n",
    "\n",
    "with open('Chabo_recordings.pkl', 'wb') as f:\n",
    "    pickle.dump(Chabo_recordings, f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fa581c9a",
   "metadata": {},
   "outputs": [],
   "source": [
    "Tito_recordings = {}\n",
    "for category in categories:\n",
    "    Tito_recordings[category] = []\n",
    "    for i in category_ids[category]:\n",
    "        data = neural_data.sel(animal = 'Tito', category_name = category, stimulus_id = i)\n",
    "        data = average_repetition(data)\n",
    "        data = data.squeeze('time_bin')\n",
    "        Tito_recordings[category].append(data.values)\n",
    "    Tito_recordings[category] = np.stack(Tito_recordings[category], axis=0).squeeze()\n",
    "\n",
    "with open('Tito_recordings.pkl', 'wb') as f:\n",
    "    pickle.dump(Tito_recordings, f)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "10fca884",
   "metadata": {},
   "source": [
    "### Get Model Activations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "608ffe0b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torchvision.models as models\n",
    "import torchvision.transforms as transforms\n",
    "import numpy as np\n",
    "import pickle\n",
    "from PIL import Image\n",
    "import tqdm\n",
    "\n",
    "torchvision_models = {\n",
    "    \"alexnet\": models.alexnet,\n",
    "    \"vgg11_bn\": models.vgg11_bn,\n",
    "    \"vgg13_bn\": models.vgg13_bn,\n",
    "    \"vgg16_bn\": models.vgg16_bn,\n",
    "    \"vgg19_bn\": models.vgg19_bn,\n",
    "    \"squeezenet1_0\": models.squeezenet1_0,\n",
    "    \"squeezenet1_1\": models.squeezenet1_1,\n",
    "    \"densenet121\": models.densenet121,\n",
    "    \"densenet169\": models.densenet169,\n",
    "    \"densenet201\": models.densenet201,\n",
    "    \"inception_v3\": models.inception_v3,\n",
    "    \"resnet18\": models.resnet18,\n",
    "    \"resnet34\": models.resnet34,\n",
    "    \"resnet50\": models.resnet50,\n",
    "    \"resnet101\": models.resnet101,\n",
    "    \"resnet152\": models.resnet152,\n",
    "    \"shufflenet_v2_x0_5\": models.shufflenet_v2_x0_5,\n",
    "    \"mobilenet_v2\": models.mobilenet_v2,\n",
    "    \"resnext50_32x4d\": models.resnext50_32x4d,\n",
    "    \"resnext101_32x8d\": models.resnext101_32x8d,\n",
    "    \"wide_resnet50_2\": models.wide_resnet50_2,\n",
    "    \"wide_resnet101_2\": models.wide_resnet101_2,\n",
    "    \"mnasnet0_5\": models.mnasnet0_5,\n",
    "    \"mnasnet1_0\": models.mnasnet1_0,\n",
    "    \"googlenet\": models.googlenet,\n",
    "    \"convnext_tiny\": models.convnext_tiny,\n",
    "    \"convnext_small\": models.convnext_small,\n",
    "    \"convnext_base\": models.convnext_base,\n",
    "    \"convnext_large\": models.convnext_large,\n",
    "    \"efficientnet_b0\": models.efficientnet_b0,\n",
    "    \"efficientnet_b4\": models.efficientnet_b4,\n",
    "    \"efficientnet_b7\": models.efficientnet_b7,\n",
    "    \"efficientnet_v2_s\": models.efficientnet_v2_s,\n",
    "    \"efficientnet_v2_m\": models.efficientnet_v2_m,\n",
    "    \"regnet_y_8gf\": models.regnet_y_8gf,\n",
    "    \"regnet_y_16gf\": models.regnet_y_16gf,\n",
    "    \"swin_b\": models.swin_b,\n",
    "    \"swin_v2_b\": models.swin_v2_b,\n",
    "    \"swin_v2_s\": models.swin_v2_s,\n",
    "    \"swin_v2_t\": models.swin_v2_t,\n",
    "    \"vit_b_16\": models.vit_b_16,\n",
    "    \"vit_b_32\": models.vit_b_32,\n",
    "    \"vit_l_16\": models.vit_l_16\n",
    "}\n",
    "\n",
    "transform = transforms.Compose([\n",
    "    transforms.Resize((224, 224)),\n",
    "    transforms.ToTensor(),\n",
    "    transforms.Normalize(mean=[0.485, 0.456, 0.406], std=[0.229, 0.224, 0.225])\n",
    "])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f3408cd6",
   "metadata": {},
   "outputs": [],
   "source": [
    "img_batches = {}\n",
    "\n",
    "\n",
    "for category in categories:\n",
    "    print(f\"Preloading images for category: {category}\")\n",
    "    data = category_ids[category]\n",
    "    imgs = []\n",
    "\n",
    "    for i in data:\n",
    "        local_path = stimulus_set.get_stimulus(i)\n",
    "        img = Image.open(local_path).convert('RGB')\n",
    "        img = transform(img)\n",
    "        imgs.append(img)\n",
    "\n",
    "    img_batch = torch.stack(imgs, dim=0)\n",
    "    img_batches[category] = img_batch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3344900f",
   "metadata": {},
   "outputs": [],
   "source": [
    "for model_name, model_fn in torchvision_models.items():\n",
    "    print(f\"Processing {model_name}...\")\n",
    "\n",
    "    model = model_fn(pretrained=True)\n",
    "    model.eval()\n",
    "\n",
    "    logit_activations = {}\n",
    "    prelogit_activations = {}\n",
    "\n",
    "    def get_activation(name):\n",
    "        def hook(model, input, output):\n",
    "            activations[name] = output.detach().cpu().numpy()\n",
    "        return hook\n",
    "\n",
    "    activations = {}\n",
    "    \n",
    "    if \"resnet\" in model_name or \"resnext\" in model_name or \"wide_resnet\" in model_name:\n",
    "        model.fc.register_forward_hook(get_activation(\"logits\"))\n",
    "        model.avgpool.register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"densenet\" in model_name:\n",
    "        model.classifier.register_forward_hook(get_activation(\"logits\"))\n",
    "        model.features.norm5.register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"mobilenet\" in model_name:\n",
    "        model.classifier[-1].register_forward_hook(get_activation(\"logits\"))\n",
    "        model.classifier[0].register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"shufflenet\" in model_name:\n",
    "        model.fc.register_forward_hook(get_activation(\"logits\"))\n",
    "        model.conv5.register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"mnasnet\" in model_name:\n",
    "        model.classifier[-1].register_forward_hook(get_activation(\"logits\"))\n",
    "        model.classifier[0].register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"squeezenet\" in model_name:\n",
    "        model.classifier[3].register_forward_hook(get_activation(\"logits\"))\n",
    "        model.features[-1].register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"vgg\" in model_name:\n",
    "        model.classifier[-1].register_forward_hook(get_activation(\"logits\"))\n",
    "        model.classifier[-3].register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"alexnet\" in model_name:\n",
    "        model.classifier[-1].register_forward_hook(get_activation(\"logits\"))\n",
    "        model.classifier[-3].register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"inception\" in model_name:\n",
    "        model.fc.register_forward_hook(get_activation(\"logits\"))\n",
    "        model.avgpool.register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"googlenet\" in model_name:\n",
    "        model.fc.register_forward_hook(get_activation(\"logits\"))\n",
    "        model.avgpool.register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"convnext\" in model_name:\n",
    "        model.classifier[-1].register_forward_hook(get_activation(\"logits\"))\n",
    "        model.avgpool.register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"efficientnet\" in model_name:\n",
    "        model.classifier[-1].register_forward_hook(get_activation(\"logits\"))\n",
    "        model.avgpool.register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"swin\" in model_name:\n",
    "        model.head.register_forward_hook(get_activation(\"logits\"))\n",
    "        model.avgpool.register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"regnet\" in model_name:\n",
    "        model.fc.register_forward_hook(get_activation(\"logits\"))\n",
    "        model.avgpool.register_forward_hook(get_activation(\"prelogits\"))\n",
    "    elif \"vit\" in model_name:\n",
    "        model.heads.head.register_forward_hook(get_activation(\"logits\"))\n",
    "        model.encoder.ln.register_forward_hook(get_activation(\"prelogits\"))\n",
    "\n",
    "    for category in categories:\n",
    "        print(f\"Processing category: {category}\")\n",
    "        data = category_ids[category]\n",
    "        logit_acts = []\n",
    "        prelogit_acts = []\n",
    "\n",
    "        with torch.no_grad():\n",
    "            _ = model(img_batches[category])\n",
    "\n",
    "        logit_activations[category] = activations[\"logits\"]  # already numpy\n",
    "\n",
    "        prelogit_tensor = activations[\"prelogits\"]\n",
    "        if prelogit_tensor.ndim > 2:\n",
    "            prelogit_tensor = prelogit_tensor.reshape(prelogit_tensor.shape[0], -1)\n",
    "\n",
    "        prelogit_activations[category] = prelogit_tensor\n",
    "\n",
    "    with open(f'{model_name}_logit_activations.pkl', 'wb') as f:\n",
    "        pickle.dump(logit_activations, f)\n",
    "\n",
    "    with open(f'{model_name}_prelogit_activations.pkl', 'wb') as f:\n",
    "        pickle.dump(prelogit_activations, f)\n",
    "\n",
    "    print(f\"Saved activations for {model_name}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "984768a6",
   "metadata": {},
   "outputs": [],
   "source": [
    "### Trend Analysis"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "369ce4e9",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_metadata = [\n",
    "    {\"name\": \"alexnet\", \"year\": 2012, \"num_params\": 61100000, \"top1_acc\": 56.522, \"top5_acc\": 79.066},\n",
    "    {\"name\": \"vgg11_bn\", \"year\": 2014, \"num_params\": 132900000, \"top1_acc\": 70.370, \"top5_acc\": 89.810},\n",
    "    {\"name\": \"vgg13_bn\", \"year\": 2014, \"num_params\": 133100000, \"top1_acc\": 71.586, \"top5_acc\": 90.374},\n",
    "    {\"name\": \"vgg16_bn\", \"year\": 2014, \"num_params\": 138400000, \"top1_acc\": 73.360, \"top5_acc\": 91.516},\n",
    "    {\"name\": \"vgg19_bn\", \"year\": 2014, \"num_params\": 143700000, \"top1_acc\": 74.218, \"top5_acc\": 91.842},\n",
    "    {\"name\": \"squeezenet1_0\", \"year\": 2016, \"num_params\": 1200000, \"top1_acc\": 58.092, \"top5_acc\": 80.420},\n",
    "    {\"name\": \"squeezenet1_1\", \"year\": 2016, \"num_params\": 1200000, \"top1_acc\": 58.178, \"top5_acc\": 80.624},\n",
    "    {\"name\": \"densenet121\", \"year\": 2017, \"num_params\": 8000000, \"top1_acc\": 74.434, \"top5_acc\": 91.972},\n",
    "    {\"name\": \"densenet169\", \"year\": 2017, \"num_params\": 14100000, \"top1_acc\": 75.600, \"top5_acc\": 92.806},\n",
    "    {\"name\": \"densenet201\", \"year\": 2017, \"num_params\": 20000000, \"top1_acc\": 76.896, \"top5_acc\": 93.370},\n",
    "    {\"name\": \"inception_v3\", \"year\": 2015, \"num_params\": 27200000, \"top1_acc\": 77.294, \"top5_acc\": 93.450},\n",
    "    {\"name\": \"resnet18\", \"year\": 2015, \"num_params\": 11700000, \"top1_acc\": 69.758, \"top5_acc\": 89.078},\n",
    "    {\"name\": \"resnet34\", \"year\": 2015, \"num_params\": 21800000, \"top1_acc\": 73.314, \"top5_acc\": 91.420},\n",
    "    {\"name\": \"resnet50\", \"year\": 2015, \"num_params\": 25600000, \"top1_acc\": 76.130, \"top5_acc\": 92.862},\n",
    "    {\"name\": \"resnet101\", \"year\": 2015, \"num_params\": 44500000, \"top1_acc\": 77.374, \"top5_acc\": 93.546},\n",
    "    {\"name\": \"resnet152\", \"year\": 2015, \"num_params\": 60200000, \"top1_acc\": 78.312, \"top5_acc\": 94.046},\n",
    "    {\"name\": \"shufflenet_v2_x0_5\", \"year\": 2018, \"num_params\": 1400000, \"top1_acc\": 60.552, \"top5_acc\": 81.746},\n",
    "    {\"name\": \"mobilenet_v2\", \"year\": 2018, \"num_params\": 3500000, \"top1_acc\": 71.878, \"top5_acc\": 90.286},\n",
    "    {\"name\": \"resnext50_32x4d\", \"year\": 2017, \"num_params\": 25000000, \"top1_acc\": 77.618, \"top5_acc\": 93.698},\n",
    "    {\"name\": \"resnext101_32x8d\", \"year\": 2017, \"num_params\": 88800000, \"top1_acc\": 79.312, \"top5_acc\": 94.526},\n",
    "    {\"name\": \"wide_resnet50_2\", \"year\": 2016, \"num_params\": 68900000, \"top1_acc\": 78.468, \"top5_acc\": 94.086},\n",
    "    {\"name\": \"wide_resnet101_2\", \"year\": 2016, \"num_params\": 126900000, \"top1_acc\": 78.848, \"top5_acc\": 94.284},\n",
    "    {\"name\": \"mnasnet0_5\", \"year\": 2018, \"num_params\": 2200000, \"top1_acc\": 67.734, \"top5_acc\": 87.490},\n",
    "    {\"name\": \"mnasnet1_0\", \"year\": 2018, \"num_params\": 4400000, \"top1_acc\": 73.456, \"top5_acc\": 91.510},\n",
    "    {\"name\": \"googlenet\", \"year\": 2014, \"num_params\": 6600000, \"top1_acc\": 69.778, \"top5_acc\": 89.530},\n",
    "    {\"name\": \"convnext_base\", \"year\": 2022, \"num_params\": 88600000, \"top1_acc\": 84.062, \"top5_acc\": 96.870},\n",
    "    {\"name\": \"convnext_tiny\", \"year\": 2022, \"num_params\": 28600000, \"top1_acc\": 82.520, \"top5_acc\": 96.146},\n",
    "    {\"name\": \"convnext_small\", \"year\": 2022, \"num_params\": 50200000, \"top1_acc\": 83.616, \"top5_acc\": 96.650},\n",
    "    {\"name\": \"convnext_large\", \"year\": 2022, \"num_params\": 197800000, \"top1_acc\": 84.414, \"top5_acc\": 96.976},\n",
    "    {\"name\": \"efficientnet_b0\", \"year\": 2019, \"num_params\": 5300000, \"top1_acc\": 77.692, \"top5_acc\": 93.532},\n",
    "    {\"name\": \"efficientnet_b4\", \"year\": 2019, \"num_params\": 19300000, \"top1_acc\": 83.384, \"top5_acc\": 96.594},\n",
    "    {\"name\": \"efficientnet_b7\", \"year\": 2019, \"num_params\": 66300000, \"top1_acc\": 84.122, \"top5_acc\": 96.908},\n",
    "    {\"name\": \"efficientnet_v2_s\", \"year\": 2021, \"num_params\": 21500000, \"top1_acc\": 84.228, \"top5_acc\": 96.878},\n",
    "    {\"name\": \"efficientnet_v2_m\", \"year\": 2021, \"num_params\": 54100000, \"top1_acc\": 85.112, \"top5_acc\": 97.156},\n",
    "    #{\"name\": \"efficientnet_v2_l\", \"year\": 2021, \"num_params\": 118500000, \"top1_acc\": 85.808, \"top5_acc\": 97.788},\n",
    "    {\"name\": \"regnet_y_8gf\", \"year\": 2020, \"num_params\": 39400000, \"top1_acc\": 82.828, \"top5_acc\": 96.330},\n",
    "    {\"name\": \"regnet_y_16gf\", \"year\": 2020, \"num_params\": 83600000, \"top1_acc\": 82.886, \"top5_acc\": 96.328},\n",
    "    {\"name\": \"swin_b\", \"year\": 2021, \"num_params\": 87800000, \"top1_acc\": 83.582, \"top5_acc\": 96.640},\n",
    "    {\"name\": \"swin_v2_b\", \"year\": 2022, \"num_params\": 87900000, \"top1_acc\": 84.112, \"top5_acc\": 96.864},\n",
    "    {\"name\": \"swin_v2_s\", \"year\": 2022, \"num_params\": 49700000, \"top1_acc\": 83.712, \"top5_acc\": 96.816},\n",
    "    {\"name\": \"swin_v2_t\", \"year\": 2022, \"num_params\": 28400000, \"top1_acc\": 82.072, \"top5_acc\": 96.132},\n",
    "    {\"name\": \"vit_b_16\", \"year\": 2021, \"num_params\": 86600000, \"top1_acc\": 81.072, \"top5_acc\": 95.318},\n",
    "    {\"name\": \"vit_b_32\", \"year\": 2021, \"num_params\": 88200000, \"top1_acc\": 75.912, \"top5_acc\": 92.466},\n",
    "    {\"name\": \"vit_l_16\", \"year\": 2021, \"num_params\": 304300000, \"top1_acc\": 79.662, \"top5_acc\": 94.638},\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "7271b159",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import adjustText as aT\n",
    "\n",
    "from DVC import *"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7367c9d3",
   "metadata": {},
   "outputs": [],
   "source": [
    "prelogit_activations = {}\n",
    "for entry in model_metadata:\n",
    "    model_name = entry[\"name\"]\n",
    "    filename = f\"{model_name}_prelogit_activations.pkl\"\n",
    "    if os.path.exists(filename):\n",
    "        with open(filename, \"rb\") as f:\n",
    "            prelogit_activations[model_name] = pickle.load(f)\n",
    "        print(f\"Loaded prelogit activation for {model_name}\")\n",
    "    else:\n",
    "        print(f\"File not found: {filename}\")\n",
    "\n",
    "with open('Chabo_recordings.pkl', 'rb') as f:\n",
    "    Chabo_recordings = pickle.load(f)\n",
    "\n",
    "with open('Tito_recordings.pkl', 'rb') as f:\n",
    "    Tito_recordings = pickle.load(f)\n",
    "\n",
    "categories =  {'Animals', 'Boats', 'Cars', 'Chairs', 'Faces', 'Fruits', 'Planes', 'Tables'}\n",
    "num_categories = len(categories)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a232a57a",
   "metadata": {},
   "outputs": [],
   "source": [
    "DVC_monkeys = {}\n",
    "for model_name, activations in prelogit_activations.items():\n",
    "    print(f\"Computing normalized DV correlation for {model_name}\")\n",
    "    DVC_matrix_chabo = calculate_normalized_dv_correlation(Chabo_recordings, activations, categories, dim=25)\n",
    "    DVC_chabo = np.nanmean(DVC_matrix_chabo)\n",
    "    DVC_matrix_tito = calculate_normalized_dv_correlation(Tito_recordings, activations, categories, dim=25)\n",
    "    DVC_tito = np.nanmean(DVC_matrix_tito)\n",
    "    DVC_monkeys[model_name] = np.mean([DVC_chabo, DVC_tito])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a2a8856b",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.scatter(DVC_monkeys.keys(), DVC_monkeys.values())"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "brain-score",
   "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.11.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
