{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "gd-title",
   "metadata": {},
   "source": [
    "# Gradient Descent\n",
    "\n",
    "This notebook reproduces the one-step gradient-descent functional-residual example from the article. The CVXPY SDP construction lives in `paper_examples/gd.py`; the sparsification scan, closed-form multipliers, and plot generation are shown here.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "gd-imports",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-07T11:43:06.258139Z",
     "iopub.status.busy": "2026-05-07T11:43:06.257804Z",
     "iopub.status.idle": "2026-05-07T11:43:07.151324Z",
     "shell.execute_reply": "2026-05-07T11:43:07.150939Z"
    }
   },
   "outputs": [],
   "source": [
    "from pathlib import Path\n",
    "import itertools\n",
    "import math\n",
    "import os\n",
    "\n",
    "from IPython.display import Image, display\n",
    "import numpy as np\n",
    "\n",
    "repo_root = Path.cwd().resolve()\n",
    "os.environ.setdefault(\"MPLCONFIGDIR\", str(repo_root / \".cache\" / \"matplotlib\"))\n",
    "\n",
    "from matplotlib.lines import Line2D\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from paper_examples import gd\n",
    "from paper_examples.plots import PAPER_TEXT_WIDTH_IN, apply_plot_style, multiplier_palette, save_pdf_and_png, style_shared_legend\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "gd-parameters-md",
   "metadata": {},
   "source": [
    "We solve the dual SDP for one GD step on $\\mathcal F_{L,\\mu}$ with $L=1$, $\\mu=0.1$, and $f(x_0)-f_* \\le 1$.  The display grid runs from $0.05$ to $2.4$; the sparsification and fitted-curvature measurements use the classical certificate interval $\\gamma\\le 2/L$.  The convergence-rate variable $\\rho$ is returned separately; the plotted quantities are the six ordered interpolation multipliers on $\\{x_*,x_0,x_1\\}$.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "gd-parameters",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-07T11:43:07.153024Z",
     "iopub.status.busy": "2026-05-07T11:43:07.152917Z",
     "iopub.status.idle": "2026-05-07T11:43:07.156215Z",
     "shell.execute_reply": "2026-05-07T11:43:07.155841Z"
    }
   },
   "outputs": [],
   "source": [
    "L = 1.0\n",
    "mu = 0.1\n",
    "gamma_values = np.linspace(0.05, 2.4, 20)\n",
    "sparsification_gamma_values = np.unique(np.append(gamma_values[gamma_values <= 2.0 / L + 1e-12], 2.0 / L))\n",
    "tolerance = 1e-6\n",
    "solver = \"MOSEK\"\n",
    "figure_pdf = repo_root / \"figures\" / \"gd_functional_residual_combined.pdf\"\n",
    "\n",
    "labels, _, _ = gd.build_gd_sdp(L=L, mu=mu, gamma=float(gamma_values[0]))\n",
    "labels"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "gd-scan-md",
   "metadata": {},
   "source": [
    "The exhaustive scan is small enough to show directly.  For each subset of interpolation multipliers, we force those multipliers to zero, solve on the grid with $\\gamma\\le 2/L$, and keep the sparsest subset whose rates match the unrestricted SDP up to the chosen relative tolerance."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "gd-scan",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-07T11:43:07.157517Z",
     "iopub.status.busy": "2026-05-07T11:43:07.157436Z",
     "iopub.status.idle": "2026-05-07T11:43:10.434881Z",
     "shell.execute_reply": "2026-05-07T11:43:10.434543Z"
    }
   },
   "outputs": [],
   "source": [
    "def solve_grid(step_values=gamma_values, inactive_multipliers=()):\n",
    "    rates = []\n",
    "    multipliers = []\n",
    "    for gamma in step_values:\n",
    "        status, rho, dual = gd.solve_gd_dual(\n",
    "            L=L,\n",
    "            mu=mu,\n",
    "            gamma=float(gamma),\n",
    "            inactive_multipliers=inactive_multipliers,\n",
    "            solver=solver,\n",
    "        )\n",
    "        if rho is None or dual is None:\n",
    "            raise RuntimeError(f\"GD solve failed at gamma={gamma:g}: {status}\")\n",
    "        rates.append(rho)\n",
    "        multipliers.append(dual)\n",
    "    return np.array(rates), np.vstack(multipliers)\n",
    "\n",
    "\n",
    "raw_rates, raw_multipliers = solve_grid(step_values=sparsification_gamma_values)\n",
    "scale = np.maximum(np.abs(raw_rates), 1e-12)\n",
    "feasible_patterns = [((), 0.0)]\n",
    "\n",
    "for inactive_count in range(1, len(labels) + 1):\n",
    "    for inactive in itertools.combinations(range(len(labels)), inactive_count):\n",
    "        try:\n",
    "            candidate_rates, _ = solve_grid(step_values=sparsification_gamma_values, inactive_multipliers=inactive)\n",
    "        except RuntimeError:\n",
    "            continue\n",
    "        max_relative_error = float(np.max(np.abs(candidate_rates - raw_rates) / scale))\n",
    "        if max_relative_error <= tolerance:\n",
    "            feasible_patterns.append((inactive, max_relative_error))\n",
    "\n",
    "max_inactive = max(len(inactive) for inactive, _ in feasible_patterns)\n",
    "best_inactive, max_relative_error = min(\n",
    "    (item for item in feasible_patterns if len(item[0]) == max_inactive),\n",
    "    key=lambda item: (item[1], item[0]),\n",
    ")\n",
    "active_multipliers = tuple(idx for idx in range(len(labels)) if idx not in best_inactive)\n",
    "sparse_rates, sparse_multipliers = solve_grid(step_values=sparsification_gamma_values, inactive_multipliers=best_inactive)\n",
    "\n",
    "active_display = tuple(idx + 1 for idx in active_multipliers)\n",
    "inactive_display = tuple(idx + 1 for idx in best_inactive)\n",
    "assert active_display == (1, 2, 4)\n",
    "\n",
    "scan_summary = {\n",
    "    \"active multipliers\": active_display,\n",
    "    \"inactive multipliers\": inactive_display,\n",
    "    \"max relative error\": max_relative_error,\n",
    "}\n",
    "scan_summary\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "gd-closed-md",
   "metadata": {},
   "source": [
    "The exhaustive scan recovers the three active multipliers used by the closed form on $\\gamma\\le 2/L$.  The branch point is $\\gamma=2/(L+\\mu)$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "gd-closed",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-07T11:43:10.436435Z",
     "iopub.status.busy": "2026-05-07T11:43:10.436345Z",
     "iopub.status.idle": "2026-05-07T11:43:10.516809Z",
     "shell.execute_reply": "2026-05-07T11:43:10.516438Z"
    }
   },
   "outputs": [],
   "source": [
    "closed_form_gamma_values = sparsification_gamma_values\n",
    "closed_form_active = (0, 1, 3)\n",
    "closed_form_inactive = tuple(idx for idx in range(len(labels)) if idx not in closed_form_active)\n",
    "_, closed_form_sparse_multipliers = solve_grid(\n",
    "    step_values=closed_form_gamma_values,\n",
    "    inactive_multipliers=closed_form_inactive,\n",
    ")\n",
    "closed_form_grid = np.vstack([gd.closed_form_multipliers(float(gamma), L=L, mu=mu) for gamma in closed_form_gamma_values])\n",
    "closed_form_error = float(\n",
    "    np.max(np.abs(closed_form_sparse_multipliers[:, closed_form_active] - closed_form_grid[:, closed_form_active]))\n",
    ")\n",
    "assert math.isfinite(closed_form_error) and closed_form_error <= 5e-5\n",
    "\n",
    "closed_form_samples = []\n",
    "for gamma in (gamma_values[0], 2.0 / (L + mu), 2.0 / L):\n",
    "    closed = gd.closed_form_multipliers(float(gamma), L=L, mu=mu)\n",
    "    closed_form_samples.append(\n",
    "        {\n",
    "            \"gamma\": float(gamma),\n",
    "            \"rho\": gd.closed_form_rate(float(gamma), L=L, mu=mu),\n",
    "            \"lambda_1\": closed[0],\n",
    "            \"lambda_2\": closed[1],\n",
    "            \"lambda_4\": closed[3],\n",
    "        }\n",
    "    )\n",
    "\n",
    "closed_form_samples\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "gd-plot-md",
   "metadata": {},
   "source": [
    "The figure compares the raw unrestricted multipliers with the sparse certificate found by the exhaustive scan."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "gd-plot",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-07T11:43:10.518182Z",
     "iopub.status.busy": "2026-05-07T11:43:10.518112Z",
     "iopub.status.idle": "2026-05-07T11:43:13.885263Z",
     "shell.execute_reply": "2026-05-07T11:43:13.884735Z"
    }
   },
   "outputs": [],
   "source": [
    "apply_plot_style()\n",
    "colors = multiplier_palette(len(labels))\n",
    "plot_gamma_values = sparsification_gamma_values\n",
    "plot_raw_multipliers = raw_multipliers\n",
    "plot_sparse_multipliers = sparse_multipliers\n",
    "upper = 1.04 * float(max(np.max(plot_raw_multipliers), np.max(plot_sparse_multipliers)))\n",
    "fig, axes = plt.subplots(1, 2, figsize=(0.96 * PAPER_TEXT_WIDTH_IN, 2.1), sharey=False)\n",
    "\n",
    "for idx, color in enumerate(colors):\n",
    "    axes[0].plot(\n",
    "        plot_gamma_values,\n",
    "        plot_raw_multipliers[:, idx],\n",
    "        color=color,\n",
    "        label=rf\"$\\lambda_{{{idx + 1}}}$\",\n",
    "        alpha=1.0 if idx in active_multipliers else 0.78,\n",
    "        linewidth=1.4 if idx in active_multipliers else 1.0,\n",
    "    )\n",
    "    if idx in active_multipliers:\n",
    "        axes[1].plot(\n",
    "            plot_gamma_values,\n",
    "            plot_sparse_multipliers[:, idx],\n",
    "            color=color,\n",
    "            label=rf\"$\\lambda_{{{idx + 1}}}$\",\n",
    "            linewidth=1.4,\n",
    "        )\n",
    "\n",
    "for ax, title in zip(axes, [\"Raw interpolation multipliers\", \"Exhaustive sparsification\"]):\n",
    "    ax.set_title(title, pad=4)\n",
    "    ax.set_xlabel(r\"$\\gamma$\", labelpad=2)\n",
    "    ax.set_ylabel(r\"$\\lambda_i$\", labelpad=2)\n",
    "    ax.set_xlim(float(plot_gamma_values[0]), 2.0 / L)\n",
    "    ax.set_ylim(-0.02 * upper, upper)\n",
    "    ax.grid(axis=\"y\", color=\"#D1D5DB\", alpha=0.30, linewidth=0.45)\n",
    "    ax.grid(axis=\"x\", color=\"#E5E7EB\", alpha=0.18, linewidth=0.40)\n",
    "    ax.axhline(0.0, color=\"#9CA3AF\", linewidth=0.55, zorder=0)\n",
    "\n",
    "handles = [Line2D([0], [0], color=colors[idx], lw=1.6, label=rf\"$\\lambda_{{{idx + 1}}}$\") for idx in range(len(labels))]\n",
    "legend = axes[1].legend(\n",
    "    handles=handles,\n",
    "    loc=\"center left\",\n",
    "    bbox_to_anchor=(1.01, 0.5),\n",
    "    ncol=1,\n",
    "    frameon=True,\n",
    "    handlelength=1.35,\n",
    "    labelspacing=0.28,\n",
    "    borderpad=0.20,\n",
    "    handletextpad=0.35,\n",
    ")\n",
    "style_shared_legend(legend)\n",
    "fig.subplots_adjust(left=0.09, right=0.86, bottom=0.20, top=0.84, wspace=0.26)\n",
    "\n",
    "save_pdf_and_png(fig, figure_pdf)\n",
    "plt.close(fig)\n",
    "assert figure_pdf.exists()\n",
    "display(Image(filename=str(figure_pdf.with_suffix(\".png\"))))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "gd-curvature-md",
   "metadata": {},
   "source": [
    "Now enforce the three singleton slots found by sparsification and fix their weights to their exact closed forms in our candidate lemma search SDP. Using a trace objective to incentivize a small rank on the residual slack matrix, we can then identify the resulting inequalities as the corresponding smooth strongly-convex interpolation inequalities by examining the coefficients of the quadratic terms after subtracting the linear term $\\langle g_j,x_i-x_j\\rangle$. For a fitted pair $(L_\\gamma,\\mu_\\gamma)$ these coefficients should satisfy $c_{gg}=1/(2(L_\\gamma-\\mu_\\gamma))$, $c_{xg}=-\\mu_\\gamma/(L_\\gamma-\\mu_\\gamma)$, and $c_{xx}=L_\\gamma\\mu_\\gamma/(2(L_\\gamma-\\mu_\\gamma))$, so we read off $\\mu_\\gamma=-c_{xg}/(2c_{gg})$ and $L_\\gamma=\\mu_\\gamma+1/(2c_{gg})$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "gd-curvature",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-07T11:43:13.886676Z",
     "iopub.status.busy": "2026-05-07T11:43:13.886586Z",
     "iopub.status.idle": "2026-05-07T11:43:14.016064Z",
     "shell.execute_reply": "2026-05-07T11:43:14.015608Z"
    }
   },
   "outputs": [],
   "source": [
    "curvature_measurements = []\n",
    "for gamma in closed_form_gamma_values:\n",
    "    weights = gd.closed_form_multipliers(float(gamma), L=L, mu=mu)\n",
    "    result = gd.solve_fitted_curvature_candidate_sdp(\n",
    "        L=L,\n",
    "        mu=mu,\n",
    "        gamma=float(gamma),\n",
    "        multipliers=weights,\n",
    "        target_rate=gd.closed_form_rate(float(gamma), L=L, mu=mu),\n",
    "        solver=solver,\n",
    "    )\n",
    "    closed_L, closed_mu = gd.fitted_curvature_parameters(float(gamma), L=L, mu=mu)\n",
    "    curvature_measurements.append(\n",
    "        {\n",
    "            \"gamma\": float(gamma),\n",
    "            \"measured L_gamma\": result[\"L\"],\n",
    "            \"measured mu_gamma\": result[\"mu\"],\n",
    "            \"closed L_gamma\": closed_L,\n",
    "            \"closed mu_gamma\": closed_mu,\n",
    "            \"matrix fit error\": result[\"matrix_fit_error\"],\n",
    "            \"value residual\": result[\"value_residual_norm\"],\n",
    "            \"validity min eigenvalue\": result[\"validity_min_eigenvalue\"],\n",
    "            \"aggregate rank\": result[\"aggregate_rank\"],\n",
    "        }\n",
    "    )\n",
    "\n",
    "measured_L = np.array([row[\"measured L_gamma\"] for row in curvature_measurements])\n",
    "measured_mu = np.array([row[\"measured mu_gamma\"] for row in curvature_measurements])\n",
    "closed_L = np.array([row[\"closed L_gamma\"] for row in curvature_measurements])\n",
    "closed_mu = np.array([row[\"closed mu_gamma\"] for row in curvature_measurements])\n",
    "max_curvature_error = float(max(np.max(np.abs(measured_L - closed_L)), np.max(np.abs(measured_mu - closed_mu))))\n",
    "max_fit_error = float(max(row[\"matrix fit error\"] for row in curvature_measurements))\n",
    "max_value_residual = float(max(row[\"value residual\"] for row in curvature_measurements))\n",
    "measurement_tolerance = 5e-5\n",
    "assert max_curvature_error <= measurement_tolerance\n",
    "assert max_fit_error <= measurement_tolerance\n",
    "assert max_value_residual <= 1e-8\n",
    "\n",
    "fitted_curvature_samples = [\n",
    "    {\n",
    "        \"gamma\": row[\"gamma\"],\n",
    "        \"measured L_gamma\": row[\"measured L_gamma\"],\n",
    "        \"measured mu_gamma\": row[\"measured mu_gamma\"],\n",
    "        \"matrix fit error\": row[\"matrix fit error\"],\n",
    "    }\n",
    "    for row in (curvature_measurements[0], curvature_measurements[len(curvature_measurements) // 2], curvature_measurements[-1])\n",
    "]\n",
    "fitted_curvature_samples\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "gd-curvature-scan",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-07T11:43:14.017353Z",
     "iopub.status.busy": "2026-05-07T11:43:14.017270Z",
     "iopub.status.idle": "2026-05-07T11:43:18.673809Z",
     "shell.execute_reply": "2026-05-07T11:43:18.673338Z"
    }
   },
   "outputs": [],
   "source": [
    "curvature_scan_pdf = repo_root / \"figures\" / \"gd_fitted_curvature_scan.pdf\"\n",
    "sample_gamma = closed_form_gamma_values\n",
    "sample_L = measured_L\n",
    "sample_mu = measured_mu\n",
    "curve_gamma = np.linspace(float(gamma_values[0]), 2.0 / L, 300)\n",
    "curve_L, curve_mu = np.array([gd.fitted_curvature_parameters(float(gamma), L=L, mu=mu) for gamma in curve_gamma]).T\n",
    "\n",
    "apply_plot_style()\n",
    "fig, axes = plt.subplots(1, 2, figsize=(0.96 * PAPER_TEXT_WIDTH_IN, 2.1), sharex=True)\n",
    "curvature_specs = [\n",
    "    (axes[0], curve_L, sample_L, L, r\"$L_\\gamma$\"),\n",
    "    (axes[1], curve_mu, sample_mu, mu, r\"$\\mu_\\gamma$\"),\n",
    "]\n",
    "\n",
    "for ax, curve_values, point_values, original_value, ylabel in curvature_specs:\n",
    "    ax.plot(curve_gamma, curve_values, color=\"#1F77B4\", linewidth=1.55, label=r\"$(L_\\gamma, \\mu_\\gamma)$\")\n",
    "    ax.scatter(sample_gamma, point_values, color=\"#111827\", s=13, zorder=3, label=\"Candidate SDP\\nmeasurements\")\n",
    "    ax.axhline(original_value, color=\"#9CA3AF\", linestyle=\":\", linewidth=0.85, label=r\"$(L, \\mu)$\")\n",
    "    ax.axvline(2.0 / (L + mu), color=\"#B45309\", linestyle=\"--\", linewidth=0.85)\n",
    "    ax.axvline(2.0 / L, color=\"#6B7280\", linestyle=\"--\", linewidth=0.85)\n",
    "    ax.axvspan(2.0 / L, float(gamma_values[-1]), color=\"#E5E7EB\", alpha=0.35, linewidth=0.0)\n",
    "    ax.set_xlim(float(gamma_values[0]), float(gamma_values[-1]))\n",
    "    ax.set_xlabel(r\"$\\gamma$\", labelpad=2)\n",
    "    ax.set_ylabel(ylabel, labelpad=2)\n",
    "    ax.grid(axis=\"y\", color=\"#D1D5DB\", alpha=0.30, linewidth=0.45)\n",
    "    ax.grid(axis=\"x\", color=\"#E5E7EB\", alpha=0.18, linewidth=0.40)\n",
    "\n",
    "transition_xlim = (1.64, 2.02)\n",
    "transition_ylim = (0.99, 1.13)\n",
    "inset = axes[0].inset_axes([0.17, 0.30, 0.32, 0.36])\n",
    "inset.plot(curve_gamma, curve_L, color=\"#1F77B4\", linewidth=1.15)\n",
    "inset.scatter(sample_gamma, sample_L, color=\"#111827\", s=10, zorder=3)\n",
    "inset.axhline(L, color=\"#9CA3AF\", linestyle=\":\", linewidth=0.75)\n",
    "inset.axvline(2.0 / (L + mu), color=\"#B45309\", linestyle=\"--\", linewidth=0.75)\n",
    "inset.axvline(2.0 / L, color=\"#6B7280\", linestyle=\"--\", linewidth=0.75)\n",
    "inset.set_xlim(*transition_xlim)\n",
    "inset.set_ylim(*transition_ylim)\n",
    "inset.set_xticks([])\n",
    "inset.set_yticks([])\n",
    "inset.tick_params(left=False, bottom=False, labelleft=False, labelbottom=False)\n",
    "inset.grid(False)\n",
    "inset.set_facecolor(\"white\")\n",
    "for spine in inset.spines.values():\n",
    "    spine.set_visible(True)\n",
    "    spine.set_color(\"#9CA3AF\")\n",
    "    spine.set_linewidth(0.65)\n",
    "zoom_indicator = axes[0].indicate_inset_zoom(\n",
    "    inset,\n",
    "    edgecolor=\"#9CA3AF\",\n",
    "    linewidth=0.55,\n",
    "    alpha=0.55,\n",
    ")\n",
    "if hasattr(zoom_indicator, \"rectangle\"):\n",
    "    zoom_rectangle = zoom_indicator.rectangle\n",
    "    zoom_connectors = zoom_indicator.connectors\n",
    "else:\n",
    "    zoom_rectangle, zoom_connectors = zoom_indicator\n",
    "zoom_rectangle.set_alpha(0.42)\n",
    "zoom_rectangle.set_linestyle(\"-\")\n",
    "for connector in zoom_connectors:\n",
    "    connector.set_alpha(0.34)\n",
    "    connector.set_linewidth(0.50)\n",
    "    connector.set_linestyle(\"-\")\n",
    "\n",
    "axes[0].set_title(\"Smoothness parameter\", pad=4)\n",
    "axes[1].set_title(\"Strong convexity parameter\", pad=4)\n",
    "for ax in axes:\n",
    "    for x_value, label, color, x_offset, rotation in (\n",
    "        (2.0 / (L + mu), r\"$\\frac{2}{L+\\mu}$\", \"#92400E\", -9, 90),\n",
    "        (2.0 / L, r\"$\\frac{2}{L}$\", \"#4B5563\", 9, 270),\n",
    "    ):\n",
    "        y_min, y_max = ax.get_ylim()\n",
    "        ax.annotate(\n",
    "            label,\n",
    "            xy=(x_value, 0.5 * (y_min + y_max)),\n",
    "            xytext=(x_offset, 0),\n",
    "            textcoords=\"offset points\",\n",
    "            ha=\"center\",\n",
    "            va=\"center\",\n",
    "            rotation=rotation,\n",
    "            rotation_mode=\"anchor\",\n",
    "            fontsize=12,\n",
    "            color=color,\n",
    "        )\n",
    "handles, labels_for_legend = axes[0].get_legend_handles_labels()\n",
    "legend = axes[1].legend(handles, labels_for_legend, loc=\"lower left\", frameon=True, borderpad=0.25, handlelength=1.3)\n",
    "style_shared_legend(legend)\n",
    "fig.subplots_adjust(left=0.10, right=0.985, bottom=0.20, top=0.84, wspace=0.32)\n",
    "\n",
    "save_pdf_and_png(fig, curvature_scan_pdf)\n",
    "plt.close(fig)\n",
    "assert curvature_scan_pdf.exists()\n",
    "display(Image(filename=str(curvature_scan_pdf.with_suffix(\".png\"))))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "tmp",
   "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.13.1"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
