{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ad59dd92",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "import HALL_lib"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8bf87ecc",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.rcParams['text.usetex'] = True\n",
    "plt.rcParams['text.latex.preamble'] = r'''\n",
    "\\usepackage{mathtools}\n",
    "% more packages here\n",
    "'''"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9b600274",
   "metadata": {},
   "outputs": [],
   "source": [
    "# ============== NEWDATASET ==============\n",
    "\n",
    "# cache_dir = \"cache-newdataset\"\n",
    "# maxResponsesPerPrompt = 500\n",
    "\n",
    "# model_names = {\n",
    "#     0 : \"Llama-8B\",\n",
    "#     1 : \"Phi-14B\",\n",
    "#     2 : \"Mistral-7B\",\n",
    "#     3 : \"Solar-11B\",\n",
    "#     4 : \"Gemma-9B\",\n",
    "#     5 : \"Qwen-15B\",\n",
    "#     6 : \"DeepSeek-7B\",\n",
    "#     7 : \"Yi-9B\"\n",
    "# }\n",
    "\n",
    "# ============== OLDDATASET ==============\n",
    "\n",
    "# cache_dir = \"cache-olddataset-20-22\"\n",
    "# maxResponsesPerPrompt=100\n",
    "\n",
    "# model_names = {\n",
    "#     0: 'Mistral-7B',\n",
    "#     1: 'Gemma-9B',\n",
    "#     2: 'Solar-11B',\n",
    "#     3: 'Phi-14B',\n",
    "#     4: 'Qwen-32B',\n",
    "#     5: 'Gemma-27B',\n",
    "#     6: 'Qwen-15B'\n",
    "# }\n",
    "\n",
    "# best_reg_lambda = 1.9\n",
    "\n",
    "# ============== OLDDATASET ==============\n",
    "\n",
    "embedding_label=\"response_embeddings\"\n",
    "cache_dir = \"cache-icml\"\n",
    "maxResponsesPerPrompt=150\n",
    "\n",
    "model_names = {\n",
    "    0: 'Mistral-7B',\n",
    "    1: 'Gemma-9B',\n",
    "    2: 'Solar-11B',\n",
    "    3: 'Phi-14B',\n",
    "    4: 'Qwen-15B',\n",
    "    5: 'Gemma-27B',\n",
    "    6: 'Qwen-32B',\n",
    "    7: \"DeepSeek-7B\",\n",
    "    8: \"Llama-8B\",\n",
    "    9: \"Yi-9B\"\n",
    "}\n",
    "\n",
    "best_reg_lambda = 1.2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a810ae8c",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset_fn = \"../llm-hallucinations/dataset_icml.parquet\"\n",
    "\n",
    "df = HALL_lib.loadParquet(dataset_fn, unifyYears=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d7f99912",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_order, model_rank = HALL_lib.build_model_size_order(model_names)\n",
    "\n",
    "false_premise = {pid : fp for pid, fp in df[['prompt_id', \"false_premise\"]].value_counts().keys()} if \"false_premise\" in df.columns else None"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "47a3b737",
   "metadata": {},
   "outputs": [],
   "source": [
    "lambda_values = [\n",
    "    1e-4, 3e-4,\n",
    "    1e-3, 3e-3,\n",
    "    1e-2, 3e-2,\n",
    "    1e-1, 2e-1, 3e-1, 5e-1, 7e-1,\n",
    "    1e+0, 1.4e+0, 2e+0, 3e+0, 5e+0,\n",
    "    1e+1, 3e+1,\n",
    "    1e+2, 3e+2,\n",
    "]\n",
    "\n",
    "results_lambda = HALL_lib.run_full_lambda_sensitivity_study(\n",
    "    df=df,\n",
    "    model_ids=model_names.keys(),\n",
    "    prompt_ids_by_model=[\n",
    "        [x for x in df[df[\"model_id\"] == mid][\"prompt_id\"].unique()]\n",
    "        for mid in model_names\n",
    "    ],\n",
    "    lambda_values=lambda_values,\n",
    "    test_fraction=0.2,\n",
    "    n_splits=5,\n",
    "    random_state=42,\n",
    "    use_cache=True,\n",
    "    cache_dir=cache_dir + \"/LP-lambda-sensitivity\",\n",
    "    overwrite_cache=False,\n",
    "    logskip=True,\n",
    "    embedding_label=embedding_label\n",
    ")\n",
    "\n",
    "# The regularisation parameter \\lambda was varied over a logarithmically spaced grid ranging from 10^{-6} to 10^{-1}, covering both weakly and strongly regularised regimes. This range was found sufficient to capture the transition from unstable Fisher directions to over-regularised, mean-difference–dominated projections."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b1484c68",
   "metadata": {},
   "outputs": [],
   "source": [
    "results_lambda_fine = HALL_lib.run_full_lambda_sensitivity_study(\n",
    "    df=df,\n",
    "    model_ids=model_names.keys(),\n",
    "    prompt_ids_by_model=[\n",
    "        [x for x in df[df[\"model_id\"] == mid][\"prompt_id\"].unique()]\n",
    "        for mid in model_names\n",
    "    ],\n",
    "    lambda_values=np.arange(0.5, 4.5, .1),\n",
    "    test_fraction=0.2,\n",
    "    n_splits=5,\n",
    "    random_state=42,\n",
    "    use_cache=True,\n",
    "    cache_dir=cache_dir + \"/LP-lambda-sensitivity-fine\",\n",
    "    overwrite_cache=False,\n",
    "    logskip=True,\n",
    "    embedding_label=embedding_label\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "47c65595",
   "metadata": {},
   "source": [
    "## PLOT"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6af4b9d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "def aggregate_metric_over_lambda(\n",
    "    df,\n",
    "    metric=\"f1\",\n",
    "    score_metric=\"accuracy\",\n",
    "    agg_prompts=True,\n",
    "    agg_models=False,\n",
    "):\n",
    "    \"\"\"\n",
    "    Aggregate metrics over prompts, test splits and repetitions\n",
    "    as a function of the Fisher regularisation parameter lambda.\n",
    "\n",
    "    Returns one row per (model_id, lambda_reg) unless aggregation is requested.\n",
    "\n",
    "    Parameters\n",
    "    ----------\n",
    "    df : pd.DataFrame\n",
    "        Output of lambda sensitivity experiments\n",
    "    metric : str\n",
    "        Primary metric to aggregate (e.g. 'f1')\n",
    "    score_metric : str\n",
    "        Secondary metric (e.g. 'accuracy')\n",
    "    agg_prompts : bool\n",
    "        If True, aggregate over prompts\n",
    "    agg_models : bool\n",
    "        If True, aggregate over models as well\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    agg_df : pd.DataFrame\n",
    "        Columns:\n",
    "        - model_id (optional)\n",
    "        - lambda_reg\n",
    "        - metric_mean\n",
    "        - metric_std\n",
    "        - score_mean\n",
    "        - score_std\n",
    "        - n_runs\n",
    "    \"\"\"\n",
    "\n",
    "    group_cols = [\"lambda_reg\"]\n",
    "\n",
    "    if not agg_models:\n",
    "        group_cols.insert(0, \"model_id\")\n",
    "    if not agg_prompts:\n",
    "        group_cols.insert(1, \"prompt_id\")\n",
    "\n",
    "    agg_df = (\n",
    "        df\n",
    "        .groupby(group_cols)\n",
    "        .agg(\n",
    "            metric_mean=(metric, \"mean\"),\n",
    "            metric_std=(metric, \"std\"),\n",
    "            score_mean=(score_metric, \"mean\"),\n",
    "            score_std=(score_metric, \"std\"),\n",
    "            n_runs=(metric, \"count\"),\n",
    "        )\n",
    "        .reset_index()\n",
    "        .sort_values(\"lambda_reg\")\n",
    "    )\n",
    "\n",
    "    return agg_df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c1338b04",
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_lambda_sensitivity(\n",
    "    agg_df,\n",
    "    model_names,\n",
    "    metric=\"f1\",\n",
    "    metric_label='metric',\n",
    "    ratio=(3, 2),\n",
    "    scale=3,\n",
    "    best_reg_lambda=None,\n",
    "):\n",
    "    fig, ax = plt.subplots(figsize=[scale * x for x in ratio])\n",
    "\n",
    "    for mid, name in model_names.items():\n",
    "        df_m = agg_df[agg_df[\"model_id\"] == mid]\n",
    "        if df_m.empty:\n",
    "            continue\n",
    "\n",
    "        x = np.log10(df_m[\"lambda_reg\"].values)\n",
    "\n",
    "        ax.plot(\n",
    "            x,\n",
    "            df_m[f\"{metric_label}_mean\"],\n",
    "            marker=\"o\",\n",
    "            label=name\n",
    "        )\n",
    "\n",
    "        ax.fill_between(\n",
    "            x,\n",
    "            df_m[f\"{metric_label}_mean\"] - df_m[f\"{metric_label}_std\"],\n",
    "            df_m[f\"{metric_label}_mean\"] + df_m[f\"{metric_label}_std\"],\n",
    "            alpha=0.2\n",
    "        )\n",
    "\n",
    "    if best_reg_lambda is not None:\n",
    "        ax.axvline(np.log10(best_reg_lambda), ls='--', color='k', label='Best parameter')\n",
    "\n",
    "    ax.set_xlabel(\"$\\\\log_{10}(\\\\lambda)$\", size=16)\n",
    "    ax.set_ylabel(metric.capitalize(), size=16)\n",
    "    ax.grid(True, alpha=0.4)\n",
    "\n",
    "    ax.tick_params(axis='both', labelsize=16)\n",
    "    ax.legend(ncols=3, fontsize=16)\n",
    "    # ax.set_title(f\"{metric.upper()} vs Fisher regularisation\")\n",
    "\n",
    "    return fig, ax\n",
    "\n",
    "# ===== ===== ===== ===== ===== ===== ===== =====\n",
    "\n",
    "agg_lambda = aggregate_metric_over_lambda(\n",
    "    results_lambda,\n",
    "    metric=\"f1\",\n",
    "    score_metric=\"accuracy\",\n",
    "    agg_prompts=True,\n",
    "    agg_models=False\n",
    ")\n",
    "\n",
    "fig, ax = plot_lambda_sensitivity(\n",
    "    agg_lambda,\n",
    "    model_names=model_names,\n",
    "    metric=\"f1\",\n",
    "    best_reg_lambda=best_reg_lambda\n",
    ")\n",
    "\n",
    "fig.savefig(\"img/LP_lambda_sensitivity.pdf\", bbox_inches='tight')\n",
    "\n",
    "fig, ax = plot_lambda_sensitivity(\n",
    "    agg_lambda,\n",
    "    model_names=model_names,\n",
    "    metric=\"accuracy\",\n",
    "    metric_label=\"score\",\n",
    "    best_reg_lambda=best_reg_lambda\n",
    ")\n",
    "\n",
    "fig.savefig(\"img/LP_lambda_sensitivity_acc.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "daa6aacd",
   "metadata": {},
   "outputs": [],
   "source": [
    "agg_lambda_global = aggregate_metric_over_lambda(\n",
    "    results_lambda,\n",
    "    metric=\"f1\",\n",
    "    agg_prompts=True,\n",
    "    agg_models=True,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "34ccf7e8",
   "metadata": {},
   "outputs": [],
   "source": [
    "agg_lambda_fine = aggregate_metric_over_lambda(\n",
    "    results_lambda_fine,\n",
    "    metric=\"f1\",\n",
    "    score_metric=\"accuracy\",\n",
    "    agg_prompts=True,\n",
    "    agg_models=False,\n",
    ")\n",
    "fig, ax = plot_lambda_sensitivity(\n",
    "    agg_lambda_fine,\n",
    "    model_names=model_names,\n",
    "    metric=\"f1\",\n",
    "    best_reg_lambda=best_reg_lambda,\n",
    ")\n",
    "fig, ax = plot_lambda_sensitivity(\n",
    "    agg_lambda_fine,\n",
    "    model_names=model_names,\n",
    "    metric=\"accuracy\",\n",
    "    metric_label=\"score\",\n",
    "    best_reg_lambda=best_reg_lambda,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d2d902da",
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_lambda_sensitivity_2d(\n",
    "    agg_df,\n",
    "    model_names,\n",
    "    metric=\"f1\",\n",
    "    metric_label=\"metric\",\n",
    "    ratio=(3, 2),\n",
    "    scale=3,\n",
    "    best_reg_lambda=None,\n",
    "):\n",
    "    \"\"\"\n",
    "    2D sensitivity plot:\n",
    "      x-axis: metric mean\n",
    "      y-axis: metric std\n",
    "\n",
    "    Each model is a trajectory over lambda.\n",
    "    The best lambda is highlighted with a star (nearest if needed).\n",
    "    \"\"\"\n",
    "\n",
    "    fig, ax = plt.subplots(figsize=[scale * x for x in ratio])\n",
    "\n",
    "    for mid, name in model_names.items():\n",
    "        df_m = agg_df[agg_df[\"model_id\"] == mid]\n",
    "        if df_m.empty:\n",
    "            continue\n",
    "\n",
    "        # ---- sort by lambda (important for line continuity)\n",
    "        df_m = df_m.sort_values(\"lambda_reg\")\n",
    "\n",
    "        x = df_m[f\"{metric_label}_mean\"].values\n",
    "        y = df_m[f\"{metric_label}_std\"].values\n",
    "\n",
    "        ax.plot(\n",
    "            x,\n",
    "            y,\n",
    "            marker=\"o\",\n",
    "            label=name,\n",
    "            alpha=0.9\n",
    "        )\n",
    "\n",
    "        # ---- highlight best lambda (nearest if not exact)\n",
    "        if best_reg_lambda is not None:\n",
    "            idx = np.argmin(np.abs(df_m[\"lambda_reg\"].values - best_reg_lambda))\n",
    "\n",
    "            ax.plot(\n",
    "                x[idx],\n",
    "                y[idx],\n",
    "                marker=\"*\",\n",
    "                markersize=14,\n",
    "                color=ax.lines[-1].get_color(),\n",
    "                zorder=5\n",
    "            )\n",
    "\n",
    "    ax.set_xlabel(f\"Mean {metric.upper()}\")\n",
    "    ax.set_ylabel(f\"Std {metric.upper()}\")\n",
    "    ax.grid(True, alpha=0.4)\n",
    "    ax.legend()\n",
    "    ax.set_title(f\"{metric.upper()} mean-variance trade-off vs Fisher regularisation\")\n",
    "\n",
    "    return fig, ax\n",
    "\n",
    "# ===== ===== ===== ===== ===== ===== ===== =====\n",
    "\n",
    "fig, ax = plot_lambda_sensitivity_2d(\n",
    "    agg_df=agg_lambda,\n",
    "    model_names=model_names,\n",
    "    metric=\"f1\",\n",
    "    metric_label=\"metric\",\n",
    "    best_reg_lambda=best_reg_lambda,\n",
    "    scale=3\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "96bbb025",
   "metadata": {},
   "outputs": [],
   "source": [
    "agg_lambda.loc[agg_lambda.groupby('model_id')['metric_mean'].idxmax()]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "43d40b14",
   "metadata": {},
   "outputs": [],
   "source": [
    "agg_lambda"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "29c38fb6",
   "metadata": {},
   "source": [
    "## Choose a single λ that minimises the average relative regret with respect to the best achievable λ for each (model, prompt) task.\n",
    "\n",
    "Let:\n",
    " - $s_{m,p}(\\lambda)$ be the score (e.g. F1) for model $m$, prompt $p$, and regularisation $\\lambda$\n",
    " - $s^*_{m,p} = \\max_{\\lambda} s_{m,p}(\\lambda)$\n",
    "\n",
    "Define the relative loss (regret):\n",
    "$$\n",
    "    \\ell_{m,p}(\\lambda)\n",
    "    \\;=\\;\n",
    "    \\frac{s^*_{m,p} - s_{m,p}(\\lambda)}{s^*_{m,p}}\n",
    "$$\n",
    "\n",
    "Then define the average relative loss:\n",
    "$$\n",
    "    L(\\lambda) \\;=\\; \\mathbb{E}_{m,p}[\\ell_{m,p}(\\lambda)]\n",
    "$$\n",
    "\n",
    "Your chosen $\\lambda$ is:\n",
    "$$\n",
    "    \\lambda^\\star = \\arg\\min_{\\lambda} L(\\lambda)\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "50e9e8a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "def aggregate_over_runs(\n",
    "    df,\n",
    "    metric=\"f1\"\n",
    "):\n",
    "    \"\"\"\n",
    "    Average metric over runs (splits, iterations, etc.)\n",
    "    \"\"\"\n",
    "    return (\n",
    "        df\n",
    "        .groupby([\"model_id\", \"prompt_id\", \"lambda_reg\"])[metric]\n",
    "        .mean()\n",
    "        .reset_index(name=metric)\n",
    "    )\n",
    "\n",
    "def compute_best_scores(df_agg, metric=\"f1\"):\n",
    "    \"\"\"\n",
    "    Compute s*_{m,p} = max_lambda s(m,p,lambda)\n",
    "    \"\"\"\n",
    "    return (\n",
    "        df_agg\n",
    "        .groupby([\"model_id\", \"prompt_id\"])[metric]\n",
    "        .max()\n",
    "        .reset_index(name=\"best_score\")\n",
    "    )\n",
    "\n",
    "def compute_relative_loss(df_agg, df_best, metric=\"f1\"):\n",
    "    \"\"\"\n",
    "    Add relative loss column\n",
    "    \"\"\"\n",
    "    df = df_agg.merge(\n",
    "        df_best,\n",
    "        on=[\"model_id\", \"prompt_id\"],\n",
    "        how=\"left\"\n",
    "    )\n",
    "\n",
    "    df[\"relative_loss\"] = (\n",
    "        (df[\"best_score\"] - df[metric]) / df[\"best_score\"]\n",
    "    )\n",
    "\n",
    "    return df\n",
    "\n",
    "def aggregate_relative_loss(df_rel):\n",
    "    \"\"\"\n",
    "    Compute L(lambda)\n",
    "    \"\"\"\n",
    "    return (\n",
    "        df_rel\n",
    "        .groupby(\"lambda_reg\")[\"relative_loss\"]\n",
    "        .agg([\"mean\", \"std\", \"max\"])\n",
    "        .reset_index()\n",
    "        .rename(columns={\n",
    "            \"mean\": \"rel_loss_mean\",\n",
    "            \"std\": \"rel_loss_std\",\n",
    "            \"max\": \"rel_loss_max\"\n",
    "        })\n",
    "    )\n",
    "\n",
    "def select_lambda_min_regret(df_lambda_loss):\n",
    "    idx = df_lambda_loss[\"rel_loss_mean\"].idxmin()\n",
    "    return df_lambda_loss.loc[idx]\n",
    "\n",
    "# ===== ===== ===== ===== ===== ===== ===== =====\n",
    "\n",
    "df_agg = aggregate_over_runs(results_lambda_fine, metric=\"accuracy\")\n",
    "df_best = compute_best_scores(df_agg, metric=\"accuracy\")\n",
    "df_rel  = compute_relative_loss(df_agg, df_best, metric=\"accuracy\")\n",
    "df_loss = aggregate_relative_loss(df_rel)\n",
    "\n",
    "lambda_star = select_lambda_min_regret(df_loss)\n",
    "\n",
    "print(\"lambda_star (acc)\", lambda_star, '\\n')\n",
    "\n",
    "df_agg = aggregate_over_runs(results_lambda_fine, metric=\"f1\")\n",
    "df_best = compute_best_scores(df_agg, metric=\"f1\")\n",
    "df_rel  = compute_relative_loss(df_agg, df_best, metric=\"f1\")\n",
    "df_loss = aggregate_relative_loss(df_rel)\n",
    "\n",
    "lambda_star = select_lambda_min_regret(df_loss)\n",
    "\n",
    "print(\"lambda_star (f1)\", lambda_star, '\\n')\n",
    "\n",
    "lambda_star_worst = (\n",
    "    df_loss\n",
    "    .sort_values(\"rel_loss_max\")\n",
    "    .iloc[0]\n",
    ")\n",
    "\n",
    "print(\"lambda_star_worst\", lambda_star_worst)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "592895d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_average_best_lambda(df_agg, metric=\"f1\"):\n",
    "    best_per_task = (\n",
    "        df_agg\n",
    "        .sort_values(metric, ascending=False)\n",
    "        .groupby([\"model_id\", \"prompt_id\"])\n",
    "        .first()\n",
    "        .reset_index()\n",
    "    )\n",
    "\n",
    "    return {\n",
    "        \"mean_best_lambda\": best_per_task[\"lambda_reg\"].mean(),\n",
    "        \"median_best_lambda\": best_per_task[\"lambda_reg\"].median()\n",
    "    }\n",
    "\n",
    "# ===== ===== ===== ===== ===== ===== ===== =====\n",
    "\n",
    "avg_best = compute_average_best_lambda(df_agg)\n",
    "\n",
    "print(\"Min-regret lambda:\", lambda_star[\"lambda_reg\"])\n",
    "print(\"Average best lambda:\", avg_best)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2621fd62",
   "metadata": {},
   "outputs": [],
   "source": [
    "fig, ax = plt.subplots(figsize=(6, 4))\n",
    "\n",
    "ax.plot(\n",
    "    df_loss[\"lambda_reg\"],\n",
    "    df_loss[\"rel_loss_mean\"],\n",
    "    marker=\"o\"\n",
    ")\n",
    "\n",
    "ax.fill_between(\n",
    "    df_loss[\"lambda_reg\"],\n",
    "    df_loss[\"rel_loss_mean\"] - df_loss[\"rel_loss_std\"],\n",
    "    df_loss[\"rel_loss_mean\"] + df_loss[\"rel_loss_std\"],\n",
    "    alpha=0.2\n",
    ")\n",
    "\n",
    "ax.axvline(\n",
    "    lambda_star[\"lambda_reg\"],\n",
    "    color=\"red\",\n",
    "    linestyle=\"--\",\n",
    "    label=r\"$\\lambda^\\star$ (min regret)\"\n",
    ")\n",
    "\n",
    "ax.set_xscale(\"log\")\n",
    "ax.set_xlabel(r\"Regularisation $\\lambda$\")\n",
    "ax.set_ylabel(\"Relative performance loss\")\n",
    "ax.grid(True, alpha=0.4)\n",
    "ax.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1805409e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "df7b1932",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f4aa3527",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a4b3872e",
   "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.14.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
