{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "22b019d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7a23d1bf",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import json\n",
    "\n",
    "import sys\n",
    "\n",
    "sys.path.append(\"../\")\n",
    "\n",
    "##################################################################\n",
    "os.environ[\"TOKENIZERS_PARALLELISM\"] = \"false\"\n",
    "# os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"0,1,2,3,4,5,6,7\"\n",
    "##################################################################\n",
    "\n",
    "import logging\n",
    "from src.utils import logging_utils\n",
    "from src.utils import env_utils\n",
    "\n",
    "logger = logging.getLogger(__name__)\n",
    "\n",
    "logging.basicConfig(\n",
    "    level=logging.DEBUG,\n",
    "    format=logging_utils.DEFAULT_FORMAT,\n",
    "    datefmt=logging_utils.DEFAULT_DATEFMT,\n",
    "    stream=sys.stdout,\n",
    ")\n",
    "\n",
    "import torch\n",
    "import transformers\n",
    "\n",
    "logger.info(f\"{torch.__version__=}, {torch.version.cuda=}\")\n",
    "logger.info(\n",
    "    f\"{torch.cuda.is_available()=}, {torch.cuda.device_count()=}, {torch.cuda.get_device_name()=}\"\n",
    ")\n",
    "logger.info(f\"{transformers.__version__=}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c63a2f16",
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.utils.training_utils import get_device_map\n",
    "\n",
    "# model_key = \"meta-llama/Llama-3.2-3B\"\n",
    "# model_key = \"meta-llama/Llama-3.1-8B\"\n",
    "# model_key = \"meta-llama/Llama-3.1-70B-Instruct\"\n",
    "model_key = \"meta-llama/Llama-3.3-70B-Instruct\"\n",
    "# model_key = \"meta-llama/Llama-3.1-405B-Instruct\"\n",
    "\n",
    "# model_key = \"google/gemma-2-9b-it\"\n",
    "# model_key = \"google/gemma-3-12b-it\"\n",
    "# model_key = \"google/gemma-2-27b-it\"\n",
    "\n",
    "# model_key = \"deepseek-ai/DeepSeek-R1-Distill-Llama-8B\"\n",
    "\n",
    "# model_key = \"allenai/OLMo-2-1124-7B-Instruct\"\n",
    "# model_key = \"allenai/OLMo-7B-0424-hf\"\n",
    "\n",
    "# model_key = \"Qwen/Qwen2-7B\"\n",
    "# model_key = \"Qwen/Qwen2.5-14B-Instruct\"\n",
    "# model_key = \"Qwen/Qwen2.5-32B-Instruct\"\n",
    "# model_key = \"Qwen/Qwen2.5-72B-Instruct\"\n",
    "\n",
    "# model_key = \"Qwen/Qwen3-1.7B\"\n",
    "# model_key = \"Qwen/Qwen3-4B\"\n",
    "# model_key = \"Qwen/Qwen3-8B\"\n",
    "# model_key = \"Qwen/Qwen3-14B\"\n",
    "# model_key = \"Qwen/Qwen3-32B\"\n",
    "\n",
    "# device_map = get_device_map(model_key, 30, n_gpus=8)\n",
    "# device_map"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "38c23c68",
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.models import ModelandTokenizer\n",
    "\n",
    "# from transformers import BitsAndBytesConfig\n",
    "\n",
    "mt = ModelandTokenizer(\n",
    "    model_key=model_key,\n",
    "    torch_dtype=torch.bfloat16,\n",
    "    # device_map=device_map,\n",
    "    device_map=\"auto\",\n",
    "    # quantization_config = BitsAndBytesConfig(\n",
    "    #     # load_in_4bit=True\n",
    "    #     load_in_8bit=True\n",
    "    # )\n",
    "    attn_implementation=\"eager\",\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f17329b3",
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.selection.data import SelectOneTask, SelectOrderTask\n",
    "\n",
    "#################################################################################\n",
    "# TASK_CLS = SelectOrderTask\n",
    "# prompt_template_idx = 1\n",
    "TASK_CLS = SelectOneTask\n",
    "prompt_template_idx = 2\n",
    "N_DISTRACTORS = 5\n",
    "OPTION_STYLE = \"single_line\"\n",
    "#################################################################################\n",
    "\n",
    "select_task = TASK_CLS.load(\n",
    "    path=os.path.join(\n",
    "        env_utils.DEFAULT_DATA_DIR, \n",
    "        \"selection\", \n",
    "        # \"profession.json\"\n",
    "        # \"nationality.json\"\n",
    "        \"objects.json\"\n",
    "    )\n",
    ")\n",
    "\n",
    "print(select_task)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "acc5e354",
   "metadata": {},
   "outputs": [],
   "source": [
    "sample = select_task.get_random_sample(\n",
    "    mt = mt,\n",
    "    option_style=OPTION_STYLE,\n",
    "    prompt_template_idx=prompt_template_idx,\n",
    "    obj_idx=2,\n",
    "    # category=\"actor\",\n",
    "    # category=\"Brazil\"\n",
    "    category=\"fruit\",\n",
    "    filter_by_lm_prediction=False,\n",
    ")\n",
    "\n",
    "print(sample)\n",
    "print(sample.prompt())\n",
    "\n",
    "from src.selection.utils import verify_correct_option\n",
    "# sample.prompt_template = select_prof.prompt_templates[3]\n",
    "print(f'\"{sample.prompt()}\"', \">>\", sample.obj)\n",
    "\n",
    "verify_correct_option(\n",
    "    mt=mt,\n",
    "    target=sample.obj,\n",
    "    options=sample.options,\n",
    "    input=sample.prompt()\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "38750af9",
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.tokens import prepare_input, TokenizerOutput, find_token_range\n",
    "from src.selection.functional import verify_head_patterns\n",
    "\n",
    "sample_tokenized = prepare_input(\n",
    "    prompts=sample.prompt(),\n",
    "    tokenizer=mt.tokenizer,\n",
    ")\n",
    "def locate_with_delim(prompt, option):\n",
    "    st = prompt.index(option)\n",
    "    return prompt[st : st + len(option) + 1]\n",
    "\n",
    "attn_pattern = verify_head_patterns(  # noqa\n",
    "    prompt=sample.prompt(),\n",
    "    tokenized_prompt=sample_tokenized,\n",
    "    options=(\n",
    "        [\n",
    "            locate_with_delim(sample.prompt(), opt)\n",
    "            for opt in sample.options\n",
    "        ]\n",
    "    ),\n",
    "    # options=sample.options,\n",
    "    pivot=sample.subj,\n",
    "    mt=mt,\n",
    "    heads=[(35, 19)],\n",
    "    # heads=[(35, 19)],\n",
    "    # generate_full_answer=True,\n",
    "    query_index=-1,\n",
    "    ablate_possible_ans_info_from_options=True,\n",
    "    bare_prompt_template=\"Option: {}\",\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "20081bc5",
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.selection.functional import get_patches_to_verify_independent_enrichment\n",
    "from src.functional import patch_with_baukit, interpret_logits\n",
    "from src.selection.utils import get_first_token_id\n",
    "\n",
    "patches = get_patches_to_verify_independent_enrichment(\n",
    "    prompt = sample.prompt(),\n",
    "    options=(\n",
    "        [\n",
    "            locate_with_delim(sample.prompt(), opt)\n",
    "            for opt in sample.options\n",
    "        ]\n",
    "    ),\n",
    "    pivot=sample.subj,\n",
    "    mt=mt,\n",
    "    tokenized_prompt=sample_tokenized,\n",
    ")\n",
    "\n",
    "patched_out = patch_with_baukit(\n",
    "    mt=mt,\n",
    "    inputs=sample_tokenized,\n",
    "    patches=patches,\n",
    ")\n",
    "logits = patched_out.logits[:, -1, :]\n",
    "interpret_logits(\n",
    "    logits=logits,\n",
    "    tokenizer=mt.tokenizer,\n",
    "    interested_tokens=[get_first_token_id(opt, mt.tokenizer) for opt in sample.options],\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f6e78ecd",
   "metadata": {},
   "outputs": [],
   "source": [
    "from matplotlib import pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "# optimized_path = os.path.join(\n",
    "#     env_utils.DEFAULT_RESULTS_DIR,\n",
    "#     \"selection/optimized_heads\",\n",
    "#     mt.name.split(\"/\")[-1],\n",
    "#     f\"{select_task.task_name}.npz\"\n",
    "# )\n",
    "\n",
    "optimized_path = os.path.join(\n",
    "    env_utils.DEFAULT_RESULTS_DIR,\n",
    "    \"selection/optimized_heads\",\n",
    "    model_key.split(\"/\")[-1],\n",
    "    # \"distinct_options\",\n",
    "    f\"{select_task.task_name}\",\n",
    "    \"epoch_10.npz\"\n",
    ")\n",
    "\n",
    "optimization_results = np.load(optimized_path, allow_pickle=True)\n",
    "plt.plot(optimization_results[\"losses\"])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cded9f5b",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure(figsize=(20, 10))\n",
    "\n",
    "optimal_head_mask = torch.tensor(optimization_results[\"optimal_mask\"]).to(torch.float32)\n",
    "optimal_head_mask[50:, :] = 0.0\n",
    "\n",
    "plt.imshow(\n",
    "    optimal_head_mask.T.numpy(),\n",
    "    cmap=\"Blues\",\n",
    "    aspect=\"auto\",\n",
    "    vmin=0,\n",
    "    vmax=1,\n",
    ")\n",
    "\n",
    "heads_selected = torch.nonzero(optimal_head_mask > 0.5, as_tuple=False).tolist()\n",
    "heads_selected = [\n",
    "    (layer_idx, head_idx) for layer_idx, head_idx in heads_selected\n",
    "]\n",
    "print(len(heads_selected))\n",
    "\n",
    "# HEADS = heads_selected\n",
    "\n",
    "# (35, 19) in HEADS, (35, 19) in heads_selected"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8580aa60",
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.selection.data import CounterFactualSamplePair\n",
    "from src.functional import free_gpu_cache\n",
    "from src.selection.data import get_counterfactual_samples_interface\n",
    "import random\n",
    "\n",
    "validation_samples_save_path = os.path.join(\n",
    "    env_utils.DEFAULT_RESULTS_DIR,\n",
    "    \"selection\",\n",
    "    \"samples\",\n",
    "    \"validation\",\n",
    "    mt.name.split(\"/\")[-1],\n",
    "    select_task.task_name,\n",
    "    \"objects\",\n",
    "    \"ques_before\"\n",
    ")\n",
    "\n",
    "os.makedirs(validation_samples_save_path, exist_ok=True)\n",
    "\n",
    "\n",
    "free_gpu_cache()\n",
    "validation_set = []\n",
    "validation_limit = 1024\n",
    "\n",
    "counterfactual_sampler = get_counterfactual_samples_interface[select_task.task_name]\n",
    "\n",
    "while len(validation_set) < validation_limit:\n",
    "    print(f\"sample {len(validation_set)+1} / {validation_limit}\")\n",
    "    patch, clean = counterfactual_sampler(\n",
    "        mt=mt,\n",
    "        task=select_task,\n",
    "        filter_by_lm_prediction=True,\n",
    "        prompt_template_idx=2,\n",
    "        option_style=OPTION_STYLE,\n",
    "        n_distractors=random.choice(range(2, 6)),\n",
    "    )\n",
    "    validation_set.append((clean, patch))\n",
    "    cf_pair = CounterFactualSamplePair(\n",
    "        patch_sample=patch,\n",
    "        clean_sample=clean,\n",
    "    )\n",
    "    cf_pair.detensorize()\n",
    "    with open(\n",
    "        os.path.join(validation_samples_save_path, f\"{len(validation_set):05d}.json\"),\n",
    "        \"w\",\n",
    "    ) as f:\n",
    "        json.dump(cf_pair.to_dict(), f, indent=2)\n",
    "\n",
    "len(validation_set)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7dee93b9",
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.selection.data import CounterFactualSamplePair\n",
    "import random\n",
    "\n",
    "validation_set = []\n",
    "validation_limit = 1024\n",
    "\n",
    "validation_samples_load_path = os.path.join(\n",
    "    env_utils.DEFAULT_RESULTS_DIR,\n",
    "    \"selection\",\n",
    "    \"samples\",\n",
    "    \"validation\",\n",
    "    mt.name.split(\"/\")[-1],\n",
    "    select_task.task_name,\n",
    "    \"objects\",\n",
    "    \"ques_before\"\n",
    ")\n",
    "\n",
    "sample_files = [\n",
    "    os.path.join(validation_samples_load_path, f)\n",
    "    for f in os.listdir(validation_samples_load_path)\n",
    "    if f.endswith(\".json\")\n",
    "]\n",
    "logger.info(f\"Found {len(sample_files)} sample files\")\n",
    "\n",
    "random.shuffle(sample_files)\n",
    "sample_files = sample_files[:validation_limit]\n",
    "for sample_file in sample_files:\n",
    "    with open(sample_file, \"r\") as f:\n",
    "        cf_pair_data = json.load(f)\n",
    "    cf_pair = CounterFactualSamplePair.from_dict(cf_pair_data)\n",
    "    validation_set.append((cf_pair.clean_sample, cf_pair.patch_sample))\n",
    "\n",
    "len(validation_set)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ac007bf6",
   "metadata": {},
   "outputs": [],
   "source": [
    "sample, patch_sample = validation_set[0]\n",
    "print(sample.prompt(), \">>\", mt.tokenizer.decode(sample.ans_token_id))\n",
    "print(patch_sample.prompt(), \">>\", mt.tokenizer.decode(patch_sample.ans_token_id))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "01e6f753",
   "metadata": {},
   "outputs": [],
   "source": [
    "from tqdm.auto import tqdm\n",
    "\n",
    "ind_enrich_results = []\n",
    "for i, (sample, patch_sample) in tqdm(enumerate(validation_set)):\n",
    "    logger.info(f\"Processing test sample: {i+1}/{len(validation_set)}\")\n",
    "    print(sample)\n",
    "    print(sample.prompt())\n",
    "    sample_tokenized = prepare_input(\n",
    "        prompts=sample.prompt(),\n",
    "        tokenizer=mt.tokenizer,\n",
    "    )\n",
    "    interested_tokens = [\n",
    "        get_first_token_id(opt, mt.tokenizer) for opt in sample.options\n",
    "    ]\n",
    "    # clean run\n",
    "    clean_output = patch_with_baukit(\n",
    "        mt=mt,\n",
    "        inputs=sample_tokenized,\n",
    "        patches=[],\n",
    "    )\n",
    "    clean_logits = clean_output.logits[:, -1, :]\n",
    "    clean_pred, clean_track = interpret_logits(\n",
    "        logits=clean_logits,\n",
    "        tokenizer=mt.tokenizer,\n",
    "        interested_tokens=interested_tokens,\n",
    "    )\n",
    "    logger.info(f\"clean pred={[str(pred) for pred in clean_pred]}\")\n",
    "    logger.info(f\"clean track={clean_track}\")\n",
    "    logger.info(\"-\"*75)\n",
    "\n",
    "    # get the patches\n",
    "    patches = get_patches_to_verify_independent_enrichment(\n",
    "        prompt = sample.prompt(),\n",
    "        options=(\n",
    "            [\n",
    "                locate_with_delim(sample.prompt(), opt)\n",
    "                for opt in sample.options\n",
    "            ]\n",
    "        ),\n",
    "        # options=sample.options,\n",
    "        # options=[locate_with_delim(sample.prompt(), sample.obj)],\n",
    "        # options=[sample.obj],\n",
    "        # options=sample.options,\n",
    "        pivot=sample.subj,\n",
    "        mt=mt,\n",
    "        tokenized_prompt=sample_tokenized,\n",
    "        bare_prompt_template=\"Option: {}\"\n",
    "    )\n",
    "\n",
    "    # patched run\n",
    "    patched_out = patch_with_baukit(\n",
    "        mt=mt,\n",
    "        inputs=sample_tokenized,\n",
    "        patches=patches,\n",
    "    )\n",
    "    int_logits = patched_out.logits[:, -1, :]\n",
    "    int_pred, int_track = interpret_logits(\n",
    "        logits=int_logits,\n",
    "        tokenizer=mt.tokenizer,\n",
    "        interested_tokens=[get_first_token_id(opt, mt.tokenizer) for opt in sample.options],\n",
    "    )\n",
    "    logger.info(f\"int pred={[str(pred) for pred in int_pred]}\")\n",
    "    logger.info(f\"int track={int_track}\")\n",
    "\n",
    "    ind_enrich_results.append({\n",
    "        \"sample\": sample,\n",
    "        \"interested_tokens\": interested_tokens,\n",
    "        \"clean\": {\n",
    "            \"pred\": clean_pred,\n",
    "            \"track\": clean_track,\n",
    "        },\n",
    "        \"int\": {\n",
    "            \"pred\": int_pred,\n",
    "            \"track\": int_track,\n",
    "        },\n",
    "    })\n",
    "    logger.info(\"=\" * 100)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3be0a99b",
   "metadata": {},
   "outputs": [],
   "source": [
    "len(ind_enrich_results)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "36d73e9a",
   "metadata": {},
   "outputs": [],
   "source": [
    "passed_cases = []\n",
    "failed_cases = []\n",
    "\n",
    "for result in ind_enrich_results:\n",
    "    sample = result[\"sample\"]\n",
    "    int_track = result[\"int\"][\"track\"]\n",
    "    if int_track[list(int_track.keys())[0]][1].token_id == sample.ans_token_id:\n",
    "        passed_cases.append(result)\n",
    "    else:\n",
    "        failed_cases.append(result)\n",
    "\n",
    "accuracy = len(passed_cases) / len(ind_enrich_results)\n",
    "logger.info(f\"Independent enrichment accuracy: {accuracy*100:.2f}%\")\n",
    "\n",
    "failed_positions = {}\n",
    "for result in failed_cases:\n",
    "    sample = result[\"sample\"]\n",
    "    obj_idx = sample.obj_idx\n",
    "    if obj_idx not in failed_positions:\n",
    "        failed_positions[obj_idx] = 0\n",
    "    failed_positions[obj_idx] += 1\n",
    "\n",
    "from matplotlib import pyplot as plt\n",
    "plt.bar(failed_positions.keys(), failed_positions.values())\n",
    "plt.xlabel(\"Object Index\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "52d785ed",
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.tokens import prepare_input, TokenizerOutput, find_token_range\n",
    "from src.selection.functional import verify_head_patterns\n",
    "\n",
    "sample_idx = 12\n",
    "# sample = passed_cases[sample_idx][\"sample\"]\n",
    "sample = failed_cases[sample_idx][\"sample\"]\n",
    "sample_tokenized = prepare_input(\n",
    "    prompts=sample.prompt(),\n",
    "    tokenizer=mt.tokenizer,\n",
    ")\n",
    "def locate_with_delim(prompt, option):\n",
    "    st = prompt.index(option)\n",
    "    return prompt[st : st + len(option) + 1]\n",
    "\n",
    "attn_pattern = verify_head_patterns(  # noqa\n",
    "    prompt=sample.prompt(),\n",
    "    tokenized_prompt=sample_tokenized,\n",
    "    # options=(\n",
    "    #     [\n",
    "    #         locate_with_delim(sample.prompt(), opt)\n",
    "    #         for opt in sample.options\n",
    "    #     ]\n",
    "    # ),\n",
    "    # options=[locate_with_delim(sample.prompt(), sample.obj)], #! Less destructive. But I think it convays the same point\n",
    "    options=[f\"{sample.obj}\"], #! Without delim. the LM can store some info there.\n",
    "    pivot=sample.subj,\n",
    "    mt=mt,\n",
    "    # heads=[(35, 19)],\n",
    "    heads=heads_selected,\n",
    "    # generate_full_answer=True,\n",
    "    query_index=-1,\n",
    "    ablate_possible_ans_info_from_options=True,\n",
    "    bare_prompt_template=\"Option: {}\",\n",
    "    # bare_prompt_template=\" A {}\", #! Not bad, a bit inconsistent. Probably `A Pear` is less common. Not a problem when the option is a person.\n",
    "    # bare_prompt_template=\" {}\", #! Breaks attn pattern and LM performance. Maybe the LM deals with first token differently. \n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c082d9d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "passed_cases[sample_idx][\"int\"][\"pred\"], passed_cases[sample_idx][\"clean\"][\"pred\"] "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "71fc906f",
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.selection.functional import get_patches_to_verify_independent_enrichment\n",
    "from src.functional import patch_with_baukit, patch_with_nnsight\n",
    "\n",
    "\n",
    "patches = get_patches_to_verify_independent_enrichment(\n",
    "    prompt = sample.prompt(),\n",
    "    # options=[\"Options: Folder, Birch, Bat, Chain, Accordion, Air fryer.\\n\"],\n",
    "    options=[\"Birch,\"],\n",
    "    pivot=sample.subj,\n",
    "    mt=mt,\n",
    "    tokenized_prompt=sample_tokenized,\n",
    ")\n",
    "\n",
    "patched_run = patch_with_baukit(\n",
    "    mt=mt,\n",
    "    inputs=sample_tokenized,\n",
    "    patches=patches,\n",
    ")\n",
    "\n",
    "patched_logits = patched_run.logits[0, -1, :]\n",
    "interpret_logits(\n",
    "    logits=patched_logits,\n",
    "    tokenizer=mt.tokenizer,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3477c761",
   "metadata": {},
   "outputs": [],
   "source": [
    "patches"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "54091b4d",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "connection",
   "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.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
