{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Plotting Custom Metric Results\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import json\n",
    "import os\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from sae_bench.sae_bench_utils.formatting_utils import (\n",
    "    get_sparsity_penalty,\n",
    "    make_available_sae_df,\n",
    ")\n",
    "from sae_bench.sae_bench_utils.graphing_utils import (\n",
    "    plot_2var_graph,\n",
    "    plot_2var_graph_dict_size,\n",
    "    plot_3var_graph,\n",
    "    plot_correlation_heatmap,\n",
    "    plot_correlation_scatter,\n",
    "    plot_interactive_3var_graph,\n",
    "    plot_training_steps,\n",
    ")\n",
    "from sae_bench.sae_bench_utils.sae_selection_utils import select_saes_multiple_patterns"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Load data\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "eval_path = \"./evals/mdl\"\n",
    "image_path = os.path.join(eval_path, \"images\")\n",
    "results_path = os.path.join(eval_path, \"results\")\n",
    "\n",
    "if not os.path.exists(image_path):\n",
    "    os.makedirs(image_path)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sae_regex_patterns = [\n",
    "    r\"(sae_bench_pythia70m_sweep_topk_ctx128_0730).*\",\n",
    "    r\"(sae_bench_pythia70m_sweep_standard_ctx128_0712).*\",\n",
    "]\n",
    "sae_block_pattern = [\n",
    "    r\".*blocks\\.([4])\\.hook_resid_post__trainer_(1|2|5|6|9|10|17|18)$\",\n",
    "    r\".*blocks\\.([4])\\.hook_resid_post__trainer_(1|2|5|6|9|10|17|18)$\",\n",
    "]\n",
    "\n",
    "selected_saes_dict = select_saes_multiple_patterns(\n",
    "    sae_regex_patterns, sae_block_pattern\n",
    ")\n",
    "\n",
    "eval_results = {}\n",
    "for sae_release in selected_saes_dict:\n",
    "    for sae_id in selected_saes_dict[sae_release]:\n",
    "        filename = f\"{sae_release}_{sae_id}_eval_results.json\".replace(\"/\", \"_\")\n",
    "        filepath = os.path.join(results_path, filename)\n",
    "\n",
    "        if not os.path.exists(filepath):\n",
    "            print(f\"File {filepath} does not exist\")\n",
    "            continue\n",
    "\n",
    "        with open(filepath) as f:\n",
    "            single_sae_results = json.load(f)\n",
    "\n",
    "        eval_results[f\"{sae_release}_{sae_id}\"] = single_sae_results[\"eval_results\"][-1]\n",
    "        num_bins = eval_results[f\"{sae_release}_{sae_id}\"][\"num_bins\"]\n",
    "        print(num_bins)\n",
    "\n",
    "        print(single_sae_results[\"eval_results\"][-1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sae_regex_patterns = [\n",
    "    r\"(sae_bench_pythia70m_sweep_topk_ctx128_0730).*\",\n",
    "    r\"(sae_bench_pythia70m_sweep_standard_ctx128_0712).*\",\n",
    "]\n",
    "sae_block_pattern = [\n",
    "    r\".*blocks\\.([4])\\.hook_resid_post__trainer_(1|2|5|6|9|10|17|18)$\",\n",
    "    r\".*blocks\\.([4])\\.hook_resid_post__trainer_(1|2|5|6|9|10|17|18)$\",\n",
    "]\n",
    "\n",
    "selected_saes_dict = select_saes_multiple_patterns(\n",
    "    sae_regex_patterns, sae_block_pattern\n",
    ")\n",
    "\n",
    "eval_results = {}\n",
    "for sae_release in selected_saes_dict:\n",
    "    for sae_id in selected_saes_dict[sae_release]:\n",
    "        filename = f\"{sae_release}_{sae_id}_eval_results.json\".replace(\"/\", \"_\")\n",
    "        filepath = os.path.join(results_path, filename)\n",
    "\n",
    "        if not os.path.exists(filepath):\n",
    "            print(f\"File {filepath} does not exist\")\n",
    "            continue\n",
    "\n",
    "        with open(filepath) as f:\n",
    "            single_sae_results = json.load(f)\n",
    "\n",
    "        eval_results[f\"{sae_release}_{sae_id}\"] = single_sae_results[\"eval_results\"][-1]\n",
    "        values = single_sae_results[\"eval_results\"]\n",
    "        num_bins = [entry[\"num_bins\"] for entry in values]\n",
    "        mse_loss = [entry[\"mse_loss\"] for entry in values]\n",
    "\n",
    "        # Plotting the line for the current sae_id\n",
    "        plt.plot(num_bins, mse_loss, marker=\"o\", label=sae_id)\n",
    "\n",
    "# Customizing plot\n",
    "plt.xlabel(\"Number of Bins (num_bins)\")\n",
    "plt.ylabel(\"MSE Loss\")\n",
    "plt.title(\"MSE Loss vs Number of Bins for Each SAE ID\")\n",
    "# plt.legend()\n",
    "plt.grid(True)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sae_regex_patterns = [\n",
    "    r\"(sae_bench_pythia70m_sweep_topk_ctx128_0730).*\",\n",
    "    r\"(sae_bench_pythia70m_sweep_standard_ctx128_0712).*\",\n",
    "]\n",
    "sae_block_pattern = [\n",
    "    r\".*blocks\\.([4])\\.hook_resid_post__trainer_(1|2|5|6|9|10|17|18)$\",\n",
    "    r\".*blocks\\.([4])\\.hook_resid_post__trainer_(1|2|5|6|9|10|17|18)$\",\n",
    "]\n",
    "\n",
    "selected_saes_dict = select_saes_multiple_patterns(\n",
    "    sae_regex_patterns, sae_block_pattern\n",
    ")\n",
    "\n",
    "eval_results = {}\n",
    "for sae_release in selected_saes_dict:\n",
    "    for sae_id in selected_saes_dict[sae_release]:\n",
    "        filename = f\"{sae_release}_{sae_id}_eval_results.json\".replace(\"/\", \"_\")\n",
    "        filepath = os.path.join(results_path, filename)\n",
    "\n",
    "        if not os.path.exists(filepath):\n",
    "            print(f\"File {filepath} does not exist\")\n",
    "            continue\n",
    "\n",
    "        with open(filepath) as f:\n",
    "            single_sae_results = json.load(f)\n",
    "\n",
    "        eval_results[f\"{sae_release}_{sae_id}\"] = single_sae_results[\"eval_results\"][-1]\n",
    "        values = single_sae_results[\"eval_results\"]\n",
    "        num_bins = [entry[\"num_bins\"] for entry in values]\n",
    "        mse_loss = [entry[\"description_length\"] for entry in values]\n",
    "\n",
    "        # Plotting the line for the current sae_id\n",
    "        plt.plot(num_bins, mse_loss, marker=\"o\", label=sae_id)\n",
    "\n",
    "# Customizing plot\n",
    "plt.xlabel(\"Number of Bins (num_bins)\")\n",
    "plt.ylabel(\"Description Length\")\n",
    "plt.title(\"Description Length vs Number of Bins for Each SAE ID\")\n",
    "# plt.legend()\n",
    "plt.grid(True)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sae_names = list(eval_results.keys())\n",
    "\n",
    "print(eval_results.keys())\n",
    "print(\"\\nAvailable SAEs:\\n\", eval_results.keys())\n",
    "print(\"\\nAvailable custom metrics:\\n\", eval_results[sae_names[0]].keys())"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "In this cell, we find all of the sae_releases for the data file, and aggregate\n",
    "all of the data into `sae_data`. `sae_data` contains basic metrics like L0 and\n",
    "Loss Recovered, in addition to trainer parameters like dict size, sparsity\n",
    "penalty, SAE type, etc.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sae_data = {\"basic_eval_results\": {}, \"sae_config_dictionary_learning\": {}}\n",
    "\n",
    "for release_name in selected_saes_dict.keys():\n",
    "    sae_data_filename = f\"sae_bench_data/{release_name}_data.json\"\n",
    "\n",
    "    with open(sae_data_filename) as f:\n",
    "        sae_release_data = json.load(f)\n",
    "\n",
    "    sae_data[\"basic_eval_results\"].update(sae_release_data[\"basic_eval_results\"])\n",
    "    sae_data[\"sae_config_dictionary_learning\"].update(\n",
    "        sae_release_data[\"sae_config_dictionary_learning\"]\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(sae_data.keys())\n",
    "# print('\\nAvailable SAEs:\\n', sae_data[\"basic_eval_results\"].keys())\n",
    "\n",
    "first_sae_name = next(iter(sae_data[\"basic_eval_results\"]))\n",
    "print(first_sae_name)\n",
    "print(\n",
    "    \"\\nAvailable basic metrics:\\n\",\n",
    "    sae_data[\"basic_eval_results\"][first_sae_name].keys(),\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Gather all values in one dict for plotting\n",
    "plotting_results = eval_results\n",
    "\n",
    "sae_df = make_available_sae_df(for_printing=False)\n",
    "\n",
    "for sae_name in eval_results:\n",
    "    # sae_bench data is currently stored using the sae_name, not sae_id, as the key. So we need to do this hacky conversion\n",
    "    if \"sae_bench\" in sae_name:\n",
    "        sae_release = sae_name.split(\"_blocks\")[0]\n",
    "        sae_id = \"blocks\" + sae_name.split(\"_blocks\")[1]\n",
    "\n",
    "        sae_id_to_name_map = sae_df.saes_map[sae_release]\n",
    "        sae_data_name = sae_id_to_name_map[sae_id]\n",
    "\n",
    "    plotting_results[sae_name][\"l0\"] = sae_data[\"basic_eval_results\"][sae_data_name][\n",
    "        \"l0\"\n",
    "    ]\n",
    "    plotting_results[sae_name][\"sparsity_penalty\"] = get_sparsity_penalty(\n",
    "        sae_data[\"sae_config_dictionary_learning\"][sae_data_name]\n",
    "    )\n",
    "    plotting_results[sae_name][\"frac_recovered\"] = sae_data[\"basic_eval_results\"][\n",
    "        sae_data_name\n",
    "    ][\"frac_recovered\"]\n",
    "\n",
    "    # Add all trainer info\n",
    "    plotting_results[sae_name] = (\n",
    "        plotting_results[sae_name]\n",
    "        | sae_data[\"sae_config_dictionary_learning\"][sae_data_name][\"trainer\"]\n",
    "    )\n",
    "    plotting_results[sae_name][\"buffer\"] = sae_data[\"sae_config_dictionary_learning\"][\n",
    "        sae_data_name\n",
    "    ][\"buffer\"]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Plot custom metric above unsupervised metrics\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "k = 1\n",
    "custom_metric = f\"sae_top_{k}_test_accuracy\"\n",
    "custom_metric = \"description_length\"\n",
    "# custom_metric = \"mse_loss\"\n",
    "# custom_metric = \"llm_top_1_test_accuracy\"\n",
    "custom_metric_name = f\"description length, {num_bins} bins\"\n",
    "# custom_metric_name = f\"k={k}-Sparse Probe Accuracy\"\n",
    "title_3var = f\"L0 vs Loss Recovered vs {custom_metric_name}\"\n",
    "title_2var = f\"L0 vs {custom_metric_name}\"\n",
    "image_base_name = os.path.join(image_path, custom_metric)\n",
    "\n",
    "# plot_3var_graph(\n",
    "#     plotting_results,\n",
    "#     title_3var,\n",
    "#     custom_metric,\n",
    "#     colorbar_label=\"Custom Metric\",\n",
    "#     output_filename=f\"{image_base_name}_3var.png\",\n",
    "# )\n",
    "plot_2var_graph_dict_size(\n",
    "    plotting_results,\n",
    "    custom_metric,\n",
    "    title=title_2var,\n",
    "    output_filename=f\"{image_base_name}_2var.png\",\n",
    ")\n",
    "# plot_interactive_3var_graph(plotting_results, custom_metric)\n",
    "\n",
    "# At this point, if there's any additional .json files located alongside the ae.pt and eval_results.json\n",
    "# You can easily adapt them to be included in the plotting_results dictionary by using something similar to add_ae_config_results()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "k = 1\n",
    "custom_metric = f\"sae_top_{k}_test_accuracy\"\n",
    "custom_metric = \"description_length\"\n",
    "# custom_metric = \"mse_loss\"\n",
    "# custom_metric = \"llm_top_1_test_accuracy\"\n",
    "custom_metric_name = f\"description length, {num_bins} bins\"\n",
    "# custom_metric_name = f\"k={k}-Sparse Probe Accuracy\"\n",
    "title_3var = f\"L0 vs Loss Recovered vs {custom_metric_name}\"\n",
    "title_2var = f\"L0 vs {custom_metric_name}\"\n",
    "image_base_name = os.path.join(image_path, custom_metric)\n",
    "\n",
    "plot_3var_graph(\n",
    "    plotting_results,\n",
    "    title_3var,\n",
    "    custom_metric,\n",
    "    colorbar_label=\"Custom Metric\",\n",
    "    output_filename=f\"{image_base_name}_3var.png\",\n",
    ")\n",
    "plot_2var_graph(\n",
    "    plotting_results,\n",
    "    custom_metric,\n",
    "    title=title_2var,\n",
    "    output_filename=f\"{image_base_name}_2var.png\",\n",
    ")\n",
    "# plot_interactive_3var_graph(plotting_results, custom_metric)\n",
    "\n",
    "# At this point, if there's any additional .json files located alongside the ae.pt and eval_results.json\n",
    "# You can easily adapt them to be included in the plotting_results dictionary by using something similar to add_ae_config_results()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### ...with interactive hovering\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_interactive_3var_graph(\n",
    "    plotting_results,\n",
    "    custom_metric,\n",
    "    title=title_3var,\n",
    "    output_filename=f\"{image_base_name}_3var_interactive.html\",\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Plot metric over training checkpoints\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Note: We have SAE checkpoints at initialization (step 0), which does not fit on\n",
    "a log scale (log(0) = -inf). We visualize this with a cut in the graph."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_training_steps(\n",
    "    plotting_results,\n",
    "    custom_metric,\n",
    "    title=f\"Steps vs {custom_metric_name} Gemma Layer {layer}\",\n",
    "    output_filename=f\"{image_base_name}_steps_vs_diff.png\",\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This cell combines all of the above steps into a single function so we can plot results from multiple runs."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_results(\n",
    "    results_path: str,\n",
    "    filename: str,\n",
    "    custom_metric: str,\n",
    "    custom_metric_name: str,\n",
    "    layer: int,\n",
    "):\n",
    "    filepath = os.path.join(results_path, filename)\n",
    "\n",
    "    with open(filepath) as f:\n",
    "        eval_results = json.load(f)\n",
    "\n",
    "    sae_releases = eval_results[\"custom_eval_config\"][\"sae_releases\"]\n",
    "\n",
    "    sae_data = {\"basic_eval_results\": {}, \"sae_config_dictionary_learning\": {}}\n",
    "\n",
    "    for release_name in sae_releases:\n",
    "        sae_data_filename = f\"sae_bench_data/{release_name}_data.json\"\n",
    "\n",
    "        with open(sae_data_filename) as f:\n",
    "            sae_release_data = json.load(f)\n",
    "\n",
    "        sae_data[\"basic_eval_results\"].update(sae_release_data[\"basic_eval_results\"])\n",
    "        sae_data[\"sae_config_dictionary_learning\"].update(\n",
    "            sae_release_data[\"sae_config_dictionary_learning\"]\n",
    "        )\n",
    "\n",
    "    # Gather all values in one dict for plotting\n",
    "    plotting_results = eval_results\n",
    "\n",
    "    for sae_name in eval_results:\n",
    "        plotting_results[sae_name][\"l0\"] = sae_data[\"basic_eval_results\"][sae_name][\n",
    "            \"l0\"\n",
    "        ]\n",
    "        plotting_results[sae_name][\"sparsity_penalty\"] = get_sparsity_penalty(\n",
    "            sae_data[\"sae_config_dictionary_learning\"][sae_name]\n",
    "        )\n",
    "        plotting_results[sae_name][\"frac_recovered\"] = sae_data[\"basic_eval_results\"][\n",
    "            sae_name\n",
    "        ][\"frac_recovered\"]\n",
    "\n",
    "        # Add all trainer info\n",
    "        plotting_results[sae_name] = (\n",
    "            plotting_results[sae_name]\n",
    "            | sae_data[\"sae_config_dictionary_learning\"][sae_name][\"trainer\"]\n",
    "        )\n",
    "        plotting_results[sae_name][\"buffer\"] = sae_data[\n",
    "            \"sae_config_dictionary_learning\"\n",
    "        ][sae_name][\"buffer\"]\n",
    "\n",
    "    title_3var = f\"L0 vs Loss Recovered vs {custom_metric_name}\"\n",
    "    title_2var = f\"L0 vs {custom_metric_name}, Layer {layer}, Gemma-2-2B\"\n",
    "    image_base_name = os.path.join(image_path, custom_metric)\n",
    "\n",
    "    # plot_3var_graph(\n",
    "    #     plotting_results,\n",
    "    #     title_3var,\n",
    "    #     custom_metric,\n",
    "    #     colorbar_label=\"Custom Metric\",\n",
    "    #     output_filename=f\"{image_base_name}_3var.png\",\n",
    "    # )\n",
    "    plot_2var_graph(\n",
    "        plotting_results,\n",
    "        custom_metric,\n",
    "        title=title_2var,\n",
    "        output_filename=f\"{image_base_name}_2var.png\",\n",
    "        y_label=custom_metric_name,\n",
    "    )\n",
    "\n",
    "    if \"checkpoints\" in filename:\n",
    "        plot_training_steps(\n",
    "            plotting_results,\n",
    "            custom_metric,\n",
    "            y_label=custom_metric_name,\n",
    "            title=f\"Steps vs {custom_metric_name}\",\n",
    "            output_filename=f\"{image_base_name}_steps_vs_diff.png\",\n",
    "        )\n",
    "\n",
    "\n",
    "eval_path = \"./evals/sparse_probing\"\n",
    "eval_path = \"./evals/shift_and_tpp\"\n",
    "image_path = os.path.join(eval_path, \"images\")\n",
    "results_path = os.path.join(eval_path, \"results\")\n",
    "\n",
    "if not os.path.exists(image_path):\n",
    "    os.makedirs(image_path)\n",
    "\n",
    "\n",
    "k = 10\n",
    "\n",
    "if \"sparse_probing\" in eval_path:\n",
    "    custom_metric = f\"sae_top_{k}_test_accuracy\"\n",
    "    custom_metric_name = f\"k={k}-Sparse Probe Accuracy\"\n",
    "elif \"shift_and_tpp\" in eval_path:\n",
    "    custom_metric = f\"scr_metric_threshold_{k}\"\n",
    "    custom_metric_name = f\"SCR {k} latents\"\n",
    "else:\n",
    "    raise ValueError(\"Unknown eval path\")\n",
    "\n",
    "\n",
    "for layer in [3, 11, 19]:\n",
    "    filename = f\"gemma-2-2b_layer_{layer}_eval_results.json\"\n",
    "\n",
    "    if \"shift_and_tpp\" in eval_path:\n",
    "        filename = f\"gemma-2-2b_scr_layer_{layer}_eval_results.json\"\n",
    "\n",
    "    # filename = f\"gemma-2-2b_layer_{i}_with_checkpoints_eval_results.json\"\n",
    "\n",
    "    plot_results(results_path, filename, custom_metric, custom_metric_name, layer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Plot metric correlations\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# k=100\n",
    "# custom_metric = f'sae_top_{k}_test_accuracy'\n",
    "\n",
    "metric_keys = [\n",
    "    \"l0\",\n",
    "    \"frac_recovered\",\n",
    "    custom_metric,\n",
    "]\n",
    "\n",
    "plot_correlation_heatmap(plotting_results, metric_names=metric_keys, ae_names=None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Simple example usage:\n",
    "# plot_metric_scatter(plotting_results, metric_x=\"l0\", metric_y=\"frac_recovered\", title=\"L0 vs Fraction Recovered\")\n",
    "\n",
    "threshold_x = 50\n",
    "threshold_y = 100\n",
    "\n",
    "metric_x = f\"sae_top_{threshold_x}_test_accuracy\"\n",
    "metric_y = f\"sae_top_{threshold_y}_test_accuracy\"\n",
    "\n",
    "title = \"\"\n",
    "x_label = \"k=1 Sparse Probe Accuracy\"\n",
    "y_label = \"k=100 Sparse Probe Accuracy\"\n",
    "output_filename = os.path.join(\n",
    "    image_path,\n",
    "    f\"sparse_probing_result_correlation_for_thresholds_{threshold_y}_{threshold_y}.png\",\n",
    ")\n",
    "\n",
    "plot_correlation_scatter(\n",
    "    plotting_results,\n",
    "    metric_x=metric_x,\n",
    "    metric_y=metric_y,\n",
    "    title=title,\n",
    "    x_label=x_label,\n",
    "    y_label=y_label,\n",
    "    output_filename=output_filename,\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
