{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import argparse\n",
    "import json\n",
    "from typing import List, Tuple\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib.colors import PowerNorm, LinearSegmentedColormap  # NEW\n",
    "from matplotlib import cm  # NEW\n",
    "from matplotlib.colors import TwoSlopeNorm  # NEW\n",
    "import re\n",
    "import os\n",
    "from matplotlib.transforms import ScaledTranslation\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def load_jsonl(path: str) -> pd.DataFrame:\n",
    "    df = pd.read_json(path, lines=True)\n",
    "    assert {\"gt\", \"top_ids\", \"top_probs\"}.issubset(df.columns), \\\n",
    "        \"Input must have keys: gt, top_ids, top_probs\"\n",
    "    # Convert top_probs to float if it's a string\n",
    "    if df[\"top_probs\"].dtype == object:\n",
    "        df[\"top_probs\"] = df[\"top_probs\"].apply(lambda x: [float(p) for p in x])\n",
    "    return df\n",
    "\n",
    "def print_last_k_lists(acc: np.ndarray, labels: List[str], support: np.ndarray, thresholds=(0.10, 0.05, 0.01)):\n",
    "    \"\"\"\n",
    "    For each bin (row), find the largest rank k with correctness > threshold.\n",
    "    Prints three lists (one per threshold), ordered from highest top1 bin (e.g., 0.9–1.0) to lowest.\n",
    "    If no rank exceeds the threshold, returns 0 for that row.\n",
    "    \"\"\"\n",
    "    def right_edge(label: str) -> float:\n",
    "        # Strip any appended \" (n=...)\" part\n",
    "        base = label.split(\"(\")[0].strip()\n",
    "        # Accept en-dash or hyphen\n",
    "        sep = \"–\" if \"–\" in base else \"-\"\n",
    "        parts = [p.strip() for p in base.split(sep)]\n",
    "        try:\n",
    "            return float(parts[-1])\n",
    "        except Exception:\n",
    "            nums = re.findall(r\"[-+]?\\d*\\.\\d+|\\d+\", base)\n",
    "            return float(nums[-1]) if nums else np.nan\n",
    "\n",
    "    rights = np.array([right_edge(s) for s in labels])\n",
    "    # Order rows by descending right edge so first is ~0.9–1.0\n",
    "    order = np.argsort(rights)[::-1]\n",
    "\n",
    "    for thr in thresholds:\n",
    "        result = []\n",
    "        for b in order:\n",
    "            row = acc[b, :]\n",
    "            mask = (~np.isnan(row)) & (row > thr)\n",
    "            idxs = np.flatnonzero(mask)\n",
    "            if support[b] < 10:  # NEW: require at least 100 samples in this row\n",
    "                result.append(1)\n",
    "                continue\n",
    "            result.append(max(1, int(idxs[-1] + 1) if idxs.size > 0 else 0))  # CHANGED: enforce minimum of 1\n",
    "        print(f\"last_top_k_where_c_gt_{thr:.2f} = [{','.join(map(str, result))}]\")\n",
    "\n",
    "def make_bins_top1(\n",
    "    top1: np.ndarray,\n",
    "    bins: int = 10,\n",
    "    quantile: bool = False,\n",
    ") -> Tuple[np.ndarray, List[str], np.ndarray]:\n",
    "    if quantile:\n",
    "        cat = pd.qcut(top1, q=bins, duplicates=\"drop\")\n",
    "        codes = cat.cat.codes.to_numpy()\n",
    "        intervals = cat.cat.categories\n",
    "        labels = [f\"{iv.left:.2f}–{iv.right:.2f}\" for iv in intervals]\n",
    "        edges = np.array([iv.left for iv in intervals] + [intervals[-1].right])\n",
    "        return codes, labels, edges\n",
    "    else:\n",
    "        edges = np.linspace(0.0, 1.0, bins + 1)\n",
    "        codes = np.digitize(top1, edges, right=True) - 1\n",
    "        codes = np.clip(codes, 0, bins - 1)\n",
    "        labels = [f\"({edges[i]:.1f},{edges[i+1]:.1f}]\" for i in range(bins)]\n",
    "        return codes, labels, edges\n",
    "\n",
    "\n",
    "def aggregate_rank_by_bin_conditional(\n",
    "    df: pd.DataFrame,\n",
    "    K: int,\n",
    "    bins: int = 10,\n",
    "    quantile_bins: bool = False,\n",
    "):\n",
    "    \"\"\"\n",
    "    Computes:\n",
    "      acc[b, r]      = \\bar c_{m,r}  = P(gold at rank r | bin b)\n",
    "      mu_cond[b, r]  = \\tilde p_{m,r}= E[p^(r) | bin b, gold at rank r]   <-- conditional mean\n",
    "      std_cond[b, r] = std of p^(r) over the same conditional subset\n",
    "      (also returns unconditional means for reference)\n",
    "      C_topK[b]      = sum_r mu_cond[b,r] * acc[b,r]  (bin accuracy restricted to gold ∈ top-K)\n",
    "    \"\"\"\n",
    "    top_ids = np.stack(df[\"top_ids\"].to_numpy())\n",
    "    top_probs = np.stack(df[\"top_probs\"].to_numpy())\n",
    "    assert top_ids.shape == top_probs.shape, \"top_ids/top_probs shape mismatch\"\n",
    "\n",
    "    if K > top_probs.shape[1]:\n",
    "        raise ValueError(f\"K={K} > available top list length {top_probs.shape[1]}\")\n",
    "\n",
    "    top_ids = top_ids[:, :K]\n",
    "    top_probs = top_probs[:, :K]\n",
    "    top1 = top_probs[:, 0]\n",
    "    gt = df[\"gt\"].to_numpy()\n",
    "\n",
    "    codes, labels, _ = make_bins_top1(top1, bins=bins, quantile=quantile_bins)\n",
    "    B = len(labels)\n",
    "\n",
    "    # rank-wise correctness (same as your current acc)\n",
    "    acc = np.full((B, K), np.nan)          # \\bar c_{m,r}\n",
    "    support = np.zeros(B, dtype=int)       # N_m\n",
    "\n",
    "    # NEW: conditional means (tilde p), and also keep unconditional for reference\n",
    "    mu_cond = np.full((B, K), np.nan)      # \\tilde p_{m,r}\n",
    "    std_cond = np.full((B, K), np.nan)\n",
    "    mu_uncond = np.full((B, K), np.nan)    # \\bar p_{m,r}\n",
    "    std_uncond = np.full((B, K), np.nan)\n",
    "\n",
    "    # Optional: counts per (bin, rank) where gold is at rank r\n",
    "    N_mr = np.zeros((B, K), dtype=int)\n",
    "\n",
    "    for b in range(B):\n",
    "        idx = (codes == b)\n",
    "        n = int(idx.sum())\n",
    "        support[b] = n\n",
    "        if n == 0:\n",
    "            continue\n",
    "\n",
    "        probs_b = top_probs[idx, :]      # [n, K]\n",
    "        ids_b = top_ids[idx, :]          # [n, K]\n",
    "        gt_b = gt[idx]                   # [n]\n",
    "\n",
    "        # Unconditional means (what you had before)\n",
    "        mu_uncond[b, :] = probs_b.mean(axis=0)\n",
    "        std_uncond[b, :] = probs_b.std(axis=0, ddof=0)\n",
    "\n",
    "        # Rank-wise correctness and counts\n",
    "        # mask[t, r] == True iff gold is at rank r for example t\n",
    "        mask = (ids_b == gt_b[:, None])  # [n, K] bool\n",
    "        counts = mask.sum(axis=0)        # [K]\n",
    "        N_mr[b, :] = counts\n",
    "        acc[b, :] = counts / n\n",
    "\n",
    "        # CONDITIONAL means: average p^(r) ONLY over examples where gold is at rank r\n",
    "        for r in range(K):\n",
    "            if counts[r] > 0:\n",
    "                vals = probs_b[mask[:, r], r]   # select p^(r) where gold-at-r\n",
    "                mu_cond[b, r] = vals.mean()\n",
    "                std_cond[b, r] = vals.std(ddof=0)\n",
    "            else:\n",
    "                # leave as NaN; contribution to sums will be zero because acc[b,r]=0\n",
    "                pass\n",
    "\n",
    "    # Bin accuracy restricted to top-K (sanity check / proxy):\n",
    "    # C_topK[b] = sum_r \\tilde p_{m,r} * \\bar c_{m,r}\n",
    "    C_topK = np.nansum(mu_cond * acc, axis=1)  # shape [B]\n",
    "\n",
    "    return (\n",
    "        acc,            # \\bar c_{m,r}\n",
    "        mu_cond,        # \\tilde p_{m,r}  (use this for conditioned calculations)\n",
    "        std_cond,\n",
    "        support,\n",
    "        labels,\n",
    "        # extras (optional, but handy to inspect)\n",
    "        mu_uncond,      # \\bar p_{m,r}    (your previous \"mu\")\n",
    "        std_uncond,\n",
    "        N_mr,\n",
    "        C_topK\n",
    "    )\n",
    "\n",
    "def plot_heatmap(\n",
    "    acc: np.ndarray,\n",
    "    mu: np.ndarray,\n",
    "    support: np.ndarray,\n",
    "    labels: List[str],\n",
    "    K: int,\n",
    "    title: str = \"\",\n",
    "    save_path: str = None,\n",
    "):\n",
    "    acc_plot = acc[:, :K]\n",
    "    mu_plot = mu[:, :K]\n",
    "    prod_plot = mu_plot * acc_plot\n",
    "\n",
    "    B = acc_plot.shape[0]\n",
    "    fig_w = 9.0   # a bit wider to fit 3-line annotations\n",
    "    fig_h = 6.0\n",
    "    fig, ax = plt.subplots(figsize=(fig_w, fig_h))\n",
    "\n",
    "    # Color = mu * c\n",
    "    prod = mu * acc  # elementwise product; NaNs propagate if any input is NaN\n",
    "    norm = PowerNorm(gamma=0.5, vmin=0.0, vmax=1.0)  # NEW: more contrast below 0.1\n",
    "    viridis = cm.get_cmap(\"viridis\")  # NEW\n",
    "    mid = viridis(0.5)                # NEW\n",
    "    high = viridis(1.0)               # NEW\n",
    "    new_cmap = LinearSegmentedColormap.from_list(\"powderblue_to_viridis\", [\"#E0F2FE\", mid, high])  # CHANGED\n",
    "\n",
    "    #im = ax.imshow(prod, origin=\"lower\", aspect=\"auto\", norm=norm, cmap=new_cmap)  # CHANGED\n",
    "    im = ax.imshow(acc_plot, origin=\"lower\", aspect=\"auto\", norm=norm, cmap=new_cmap)\n",
    "    cbar = fig.colorbar(im, ax=ax, pad=0.01)\n",
    "    #cbar.set_label(\"Color: μ × c (PowerNorm γ=0.5)\", rotation=90)  # CHANGED\n",
    "    cbar.set_label(\"Correctness (c)\", rotation=90)\n",
    "\n",
    "    ax.set_xlabel(\"Rank\")\n",
    "    ax.set_ylabel(\"Confidence bin\")\n",
    "    ax.set_xticks(np.arange(K))\n",
    "    ax.set_xticklabels([str(r) for r in range(1, K + 1)])\n",
    "\n",
    "    f = [i / sum(support) for i in support]\n",
    "    yticklabels = [f\"{labels[b]}  {f[b]*100:.2f}%\" for b in range(B)]\n",
    "    ax.set_yticks(np.arange(B))\n",
    "    ax.set_yticklabels(yticklabels)\n",
    "    ax.set_title(title)\n",
    "\n",
    "    # Per-cell text: mu=..., c=..., mu*c=... (all to 4 dp)\n",
    "    for b in range(B):\n",
    "        for r in range(K):\n",
    "            m = mu_plot[b, r]\n",
    "            c = acc_plot[b, r]\n",
    "            if np.isnan(m) or np.isnan(c):\n",
    "                continue\n",
    "            mc = m * c\n",
    "            txt = (\n",
    "                f\" \\n\"\n",
    "    f\"$\\hat p$={m:.3f}\\n\"\n",
    "    f\"$\\hat c$={c:.3f}\\n\"\n",
    ")\n",
    "            ax.text(r, b, txt, ha=\"center\", va=\"center\", fontsize=8)\n",
    "\n",
    "    plt.tight_layout()\n",
    "    if save_path:\n",
    "        plt.savefig(save_path, dpi=220, bbox_inches=\"tight\")\n",
    "        print(f\"Saved heatmap to {save_path}\")\n",
    "    else:\n",
    "        plt.show()\n",
    "model_name=\"qwen2.5-32b-ins\"\n",
    "dataset_name=\"gsm8k\"\n",
    "trace_path = os.getenv(\"TOPK_TRACES_PATH\", \"topk_traces.jsonl\")\n",
    "df = load_jsonl(trace_path)\n",
    "acc, mu_cond, std_cond, support, labels, mu_uncond, std_uncond, N_mr, C_topK = aggregate_rank_by_bin_conditional(\n",
    "    df, K=20, bins=10, quantile_bins=False\n",
    ")\n",
    "#print_last_k_lists(acc, labels, support, thresholds=(0.1, 0.05, 0.01))\n",
    "print_last_k_lists(acc, labels, support, thresholds=(0.01,0.03,0.05,0.07,0.09,0.1))\n",
    "frequency = [i/sum(support) for i in support]\n",
    "print(frequency)\n",
    "cal_dir = os.getenv(\"CALIBRATION_DIR\", \".\")\n",
    "plot_heatmap(\n",
    "    acc, mu_uncond, support, labels, K=10,\n",
    "    save_path=f\"{cal_dir}/{model_name}-{dataset_name}.pdf\"\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "-1.1650,0.5376\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 350x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Unweighted fit loss (log-space): SSE=76.7528, MSE=0.495179, RMSE=0.70369\n"
     ]
    }
   ],
   "source": [
    "# x,y > 0 required for log\n",
    "x = mu_uncond.ravel()\n",
    "y = acc.ravel()\n",
    "m = (x > 0) & (y > 0) & np.isfinite(x) & np.isfinite(y)\n",
    "x, y = x[m], y[m]\n",
    "\n",
    "# fit in log space: log y = A + B log x\n",
    "X = np.log(x)\n",
    "Y = np.log(y)\n",
    "# frequency[b] = support[b] / sum(support)\n",
    "freq = np.asarray([i / np.sum(support) for i in support], dtype=float)   # shape [B]\n",
    "Bbins, Kcols = acc.shape\n",
    "w_full = np.repeat(freq[:, None], Kcols, axis=1).ravel()                  # shape [B*K]\n",
    "w = w_full[m]  \n",
    "B, A = np.polyfit(X, Y, 1, w=np.sqrt(w))\n",
    "B, A = np.polyfit(X, Y, 1)   # slope B, intercept A\n",
    "print(f\"{A:.4f},{B:.4f}\")\n",
    "# line in data space\n",
    "xx = np.logspace(np.log10(x.min()), np.log10(x.max()), 200)\n",
    "yy = np.exp(A) * xx**B\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(3.5, 3))\n",
    "ax.scatter(x, y, s=5)\n",
    "ax.set_xscale(\"log\"); ax.set_yscale(\"log\")\n",
    "#ax.set_xlim(1e-5, 1)\n",
    "#ax.set_ylim(1e-5, 1)\n",
    "#ax.set_xticks([1e-5, 1e-4, 1e-3, 1e-2, 1e-1, 1e0])\n",
    "#ax.set_yticks([1e-5, 1e-4, 1e-3, 1e-2, 1e-1, 1e0])\n",
    "ax.plot(xx, yy, color='red', linestyle='solid', linewidth=1)  # Changed to black dotted lineax.set_xscale(\"log\"); ax.set_yscale(\"log\")\n",
    "\n",
    "cal_x_min = max(x.min(), 1e-12)\n",
    "cal_x_max = min(x.max(), 1.0)\n",
    "xx_cal = np.logspace(np.log10(cal_x_min), np.log10(cal_x_max), 200)\n",
    "#ax.plot(xx_cal, xx_cal, linestyle='--', linewidth=1, label=r'perfect: $\\hat{c}=\\hat{p}$')\n",
    "\n",
    "ax.set_xlabel(r\"Average probability $\\hat{p}$\")\n",
    "ax.set_ylabel(r\"Average correctness $\\hat{c}$\")\n",
    "ax.text(0.02, 0.97,\n",
    "        rf\"$\\log\\hat{{c}} = A + B\\log\\hat{{p}}$\",\n",
    "        transform=ax.transAxes, ha=\"left\", va=\"top\",\n",
    "        fontsize=10, bbox=dict(boxstyle=\"round\", facecolor=\"white\", alpha=0.8, lw=0))\n",
    "ax.text(0.02, 0.87,\n",
    "        rf\"$A={A:.3f}$, $B={B:.3f}$\",\n",
    "        transform=ax.transAxes, ha=\"left\", va=\"top\",\n",
    "        fontsize=10, bbox=dict(boxstyle=\"round\", facecolor=\"white\", alpha=0.8, lw=0))\n",
    "plt.savefig(f'/calibration_grids/{model_name}-{dataset_name}-scatter.pdf', dpi=300, bbox_inches='tight')\n",
    "#ax.set_aspect('equal', adjustable='box')\n",
    "plt.tight_layout(); plt.show()\n",
    "yhat = A + B * X\n",
    "resid = Y - yhat\n",
    "w_eff = w  # effective weights\n",
    "\n",
    "#SSE = np.sum(w_eff * resid**2)              # weighted sum of squared errors\n",
    "#MSE = SSE / np.sum(w_eff)                   # weighted mean squared error\n",
    "#RMSE = np.sqrt(MSE)\n",
    "#print(f\"Weighted fit loss (log-space): SSE={SSE:.6g}, MSE={MSE:.6g}, RMSE={RMSE:.6g}\")\n",
    "SSE_unw = np.sum(resid**2)\n",
    "MSE_unw = SSE_unw / resid.size\n",
    "RMSE_unw = np.sqrt(MSE_unw)\n",
    "print(f\"Unweighted fit loss (log-space): SSE={SSE_unw:.6g}, MSE={MSE_unw:.6g}, RMSE={RMSE_unw:.6g}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_bins_top1(\n",
    "    top1: np.ndarray,\n",
    "    bins: int = 10,\n",
    "    quantile: bool = False,\n",
    ") -> Tuple[np.ndarray, List[str], np.ndarray]:\n",
    "    if quantile:\n",
    "        cat = pd.qcut(top1, q=bins, duplicates=\"drop\")\n",
    "        codes = cat.cat.codes.to_numpy()\n",
    "        intervals = cat.cat.categories\n",
    "        labels = [f\"{iv.left:.1f}–{iv.right:.1f}\" for iv in intervals]\n",
    "        edges = np.array([iv.left for iv in intervals] + [intervals[-1].right])\n",
    "        return codes, labels, edges\n",
    "    else:\n",
    "        edges = np.linspace(0.0, 1.0, bins + 1)  # with bins=5 => 0.0,0.2,...,1.0\n",
    "        codes = np.digitize(top1, edges, right=True) - 1\n",
    "        codes = np.clip(codes, 0, bins - 1)\n",
    "        labels = [f\"{edges[i]:.1f}–{edges[i+1]:.1f}\" for i in range(bins)]\n",
    "        return codes, labels, edges\n",
    "def plot_heatmap(\n",
    "    acc: np.ndarray,\n",
    "    mu: np.ndarray,\n",
    "    support: np.ndarray,\n",
    "    labels: List[str],\n",
    "    K: int,\n",
    "    title: str = \"\",\n",
    "    save_path: str = None,\n",
    "):\n",
    "    acc_plot = acc[:, :K]\n",
    "    mu_plot = mu[:, :K]\n",
    "    prod_plot = mu_plot * acc_plot\n",
    "\n",
    "    B = acc_plot.shape[0]\n",
    "    fig_w = 5.5   # a bit wider to fit 3-line annotations\n",
    "    fig_h = 3.5\n",
    "    fig, ax = plt.subplots(figsize=(fig_w, fig_h))\n",
    "\n",
    "    # Color = mu * c\n",
    "    prod = mu * acc  # elementwise product; NaNs propagate if any input is NaN\n",
    "    norm = PowerNorm(gamma=0.5, vmin=0.0, vmax=1.0)  # NEW: more contrast below 0.1\n",
    "    viridis = cm.get_cmap(\"viridis\")  # NEW\n",
    "    mid = viridis(0.5)                # NEW\n",
    "    high = viridis(1.0)               # NEW\n",
    "    new_cmap = LinearSegmentedColormap.from_list(\"powderblue_to_viridis\", [\"#E0F2FE\", mid, high])  # CHANGED\n",
    "\n",
    "    #im = ax.imshow(prod, origin=\"lower\", aspect=\"auto\", norm=norm, cmap=new_cmap)  # CHANGED\n",
    "    im = ax.imshow(acc_plot, origin=\"lower\", aspect=\"auto\", norm=norm, cmap=new_cmap)\n",
    "    cbar = fig.colorbar(im, ax=ax, pad=0.01)\n",
    "    #cbar.set_label(\"Color: μ × c (PowerNorm γ=0.5)\", rotation=90)  # CHANGED\n",
    "    cbar.set_label(\"Correctness\", rotation=90)\n",
    "    cbar.set_ticks([0.01, 0.10, 0.20, 0.40, 0.60, 0.80, 1.00])\n",
    "    #cbar.set_ticks([0.01, 0.10, 0.3, 0.50, 0.70, 1.00])\n",
    "\n",
    "    ax.set_xlabel(\"Rank\")\n",
    "    ax.set_ylabel(\"Confidence bin\")\n",
    "    ax.set_xticks(np.arange(K))\n",
    "    ax.set_xticklabels([str(r) for r in range(1, K + 1)])\n",
    "\n",
    "    f = [i / sum(support) for i in support]\n",
    "    yticklabels = [f\"{labels[b]}  {f[b]*100:.2f}%\" for b in range(B)]\n",
    "    ax.set_yticks(np.arange(B))\n",
    "    ax.set_yticklabels(yticklabels)\n",
    "    ax.set_title(title)\n",
    "\n",
    "    # Per-cell text: mu=..., c=..., mu*c=... (all to 4 dp)\n",
    "    for b in range(B):\n",
    "        for r in range(K):\n",
    "            m = mu_plot[b, r]\n",
    "            c = acc_plot[b, r]\n",
    "            if np.isnan(m) or np.isnan(c):\n",
    "                continue\n",
    "            mc = m * c\n",
    "            txt = (\n",
    "                f\" \\n\"\n",
    "    f\"p̂={m:.3f}\\n\"\n",
    "    f\"ĉ={c:.3f}\\n\"\n",
    ")\n",
    "            ax.text(r, b, txt, ha=\"center\", va=\"center\", fontsize=9)\n",
    "\n",
    "    plt.tight_layout()\n",
    "    if save_path:\n",
    "        plt.savefig(save_path, dpi=220, bbox_inches=\"tight\")\n",
    "        print(f\"Saved heatmap to {save_path}\")\n",
    "    else:\n",
    "        plt.show()\n",
    "\n",
    "acc, mu_cond, std_cond, support, labels, mu_uncond, std_uncond, N_mr, C_topK = aggregate_rank_by_bin_conditional(\n",
    "    df, K=5, bins=5, quantile_bins=False\n",
    ")\n",
    "\n",
    "plot_heatmap(\n",
    "    acc, mu_cond, support, labels, K=5,\n",
    "    save_path='/heatmap.pdf'\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "torchtune",
   "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.10.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
