{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b1e32d09-d17b-475c-a35d-45374a6af14d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Full Conformal Bayes with AOI from Fong & Holmes (2021) paper on simulation data\n",
    "\n",
    "\n",
    "import time\n",
    "from dataclasses import dataclass\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from numpy.random import default_rng\n",
    "import random\n",
    "\n",
    "\n",
    "\n",
    "@dataclass\n",
    "class GenParams:\n",
    "    a0: float; a1: float; a2: float\n",
    "    b0: float; b1: float; b2: float\n",
    "    c0: float; c1: float; c2: float\n",
    "    sigmas: np.ndarray  \n",
    "    weights: np.ndarray \n",
    "\n",
    "def draw_generator_params(seed: int = 20250814) -> GenParams:\n",
    "    rng = default_rng(seed)\n",
    "    a0 = rng.uniform(-4.0, -2.0); a1 = rng.uniform(0.8, 1.5); a2 = rng.uniform(0.5, 1.0)\n",
    "    b0 = rng.uniform(-0.5, 0.5);  b1 = rng.uniform(0.5, 1.2); b2 = rng.uniform(0.4, 1.0)\n",
    "    c0 = rng.uniform( 2.0, 4.0);  c1 = rng.uniform(-1.0,-0.3); c2 = rng.uniform(0.8, 1.5)\n",
    "    sigmas = np.array([0.5, 0.5, 0.5], float)\n",
    "    weights = np.array([1/3, 1/3, 1/3], float)\n",
    "    return GenParams(a0,a1,a2,b0,b1,b2,c0,c1,c2,sigmas,weights)\n",
    "\n",
    "def sample_from_generator(n: int, P: GenParams, seed: int) -> pd.DataFrame:\n",
    "    rng = default_rng(seed)\n",
    "    X1 = rng.uniform(-1, 2, size=n)\n",
    "    X2 = rng.normal(1.0, 0.5, size=n)\n",
    "    comp = rng.choice(3, size=n, p=P.weights)\n",
    "    mu1 = P.a0 + P.a1*X1 + P.a2*(X2**2)\n",
    "    mu2 = P.b0 + P.b1*(X1**2) + P.b2*(X1*X2)\n",
    "    mu3 = P.c0 + P.c1*(X1**2) + P.c2*np.cos(2*X2)\n",
    "    MU  = np.stack([mu1, mu2, mu3], axis=1)\n",
    "    Y = rng.normal(loc=MU[np.arange(n), comp], scale=P.sigmas[comp])\n",
    "    return pd.DataFrame({\"X1\": X1, \"X2\": X2, \"Y\": Y})\n",
    "\n",
    "def phi1(x1, x2):\n",
    "    return np.array([1.0, x1, x2**2], float)\n",
    "\n",
    "def phi2(x1, x2):\n",
    "    return np.array([1.0, x1**2, x1*x2], float)\n",
    "\n",
    "def phi3(x1, x2):\n",
    "    return np.array([1.0, x1**2, np.cos(2*x2)], float)\n",
    "\n",
    "def design_phi1(X1, X2):\n",
    "    return np.column_stack([np.ones_like(X1), X1, X2**2])\n",
    "\n",
    "def design_phi2(X1, X2):\n",
    "    return np.column_stack([np.ones_like(X1), X1**2, X1*X2])\n",
    "\n",
    "def design_phi3(X1, X2):\n",
    "    return np.column_stack([np.ones_like(X1), X1**2, np.cos(2*X2)])\n",
    "\n",
    "def log_norm_pdf(y, mean, sd):\n",
    "    z = (y - mean) / sd\n",
    "    return -0.5*np.log(2*np.pi) - np.log(sd) - 0.5*z*z\n",
    "\n",
    "\n",
    "\n",
    "def true_prior_means_vars():\n",
    "    \n",
    "    m1 = np.array([-3.0, 1.15, 0.75], float)\n",
    "    v1 = np.array([(2.0**2)/12, (0.7**2)/12, (0.5**2)/12], float)\n",
    "    # check if it works ok\n",
    "    m2 = np.array([ 0.0, 0.85, 0.70], float)\n",
    "    v2 = np.array([(1.0**2)/12, (0.7**2)/12, (0.6**2)/12], float)\n",
    "   \n",
    "    m3 = np.array([ 3.0, -0.65, 1.15], float)\n",
    "    v3 = np.array([(2.0**2)/12, (0.7**2)/12, (0.7**2)/12], float)\n",
    "    return (m1,v1), (m2,v2), (m3,v3)\n",
    "\n",
    "\n",
    "\n",
    "def gibbs_M2_single(Phi, y, m0, v0, sigma=0.5, rng=None, n_burn=300, n_samp=200, thin=1):\n",
    "    \n",
    "    rng = default_rng() if rng is None else rng\n",
    "    n, p = Phi.shape\n",
    "    V0_inv = np.diag(1.0 / v0)\n",
    "    s2 = sigma**2\n",
    "    XtX = Phi.T @ Phi\n",
    "    Xty = Phi.T @ y\n",
    "    samples = []\n",
    "    for it in range(n_burn + n_samp*thin):\n",
    "        V_inv = V0_inv + XtX / s2\n",
    "        V = np.linalg.inv(V_inv)\n",
    "        m = V @ (V0_inv @ m0 + Xty / s2)\n",
    "        beta = rng.multivariate_normal(m, V)\n",
    "        if it >= n_burn and (it - n_burn) % thin == 0:\n",
    "            samples.append(beta.copy())\n",
    "    return np.array(samples)  # (T,p)\n",
    "\n",
    "def gibbs_MK_mixture(Phi_list, y, K, m0_list, v0_list, alpha_dir, sigma_list,\n",
    "                     rng=None, n_burn=500, n_samp=200, thin=1):\n",
    "\n",
    "    rng = default_rng() if rng is None else rng\n",
    "    n = len(y)\n",
    "    z = rng.integers(0, K, size=n)  # init allocations\n",
    "    p_list = [Phi.shape[1] for Phi in Phi_list]\n",
    "    V0_inv_list = [np.diag(1.0 / v0) for v0 in v0_list]\n",
    "    s2 = [sig**2 for sig in sigma_list]\n",
    "\n",
    "    beta_list = [rng.multivariate_normal(m0_list[k], np.diag(v0_list[k])) for k in range(K)]\n",
    "    counts = np.bincount(z, minlength=K)\n",
    "    pi = rng.dirichlet(alpha_dir + counts)\n",
    "\n",
    "    samples = []\n",
    "    for it in range(n_burn + n_samp*thin):\n",
    "        # β_k | z, y\n",
    "        for k in range(K):\n",
    "            idx = np.where(z == k)[0]\n",
    "            if len(idx) == 0:\n",
    "                V = np.linalg.inv(V0_inv_list[k])\n",
    "                m = V @ (V0_inv_list[k] @ m0_list[k])\n",
    "                beta_list[k] = rng.multivariate_normal(m, V)\n",
    "            else:\n",
    "                Phi_k = Phi_list[k][idx]; y_k = y[idx]\n",
    "                XtX = Phi_k.T @ Phi_k\n",
    "                Xty = Phi_k.T @ y_k\n",
    "                V_inv = V0_inv_list[k] + XtX / s2[k]\n",
    "                V = np.linalg.inv(V_inv)\n",
    "                m = V @ (V0_inv_list[k] @ m0_list[k] + Xty / s2[k])\n",
    "                beta_list[k] = rng.multivariate_normal(m, V)\n",
    "\n",
    "        \n",
    "        counts = np.bincount(z, minlength=K)\n",
    "        pi = rng.dirichlet(alpha_dir + counts)\n",
    "\n",
    "        \n",
    "        for i in range(n):\n",
    "            logps = np.empty(K)\n",
    "            for k in range(K):\n",
    "                mu = Phi_list[k][i] @ beta_list[k]\n",
    "                logps[k] = np.log(pi[k] + 1e-300) + log_norm_pdf(y[i], mu, np.sqrt(s2[k]))\n",
    "            mlog = np.max(logps); probs = np.exp(logps - mlog); probs /= probs.sum()\n",
    "            z[i] = rng.choice(K, p=probs)\n",
    "\n",
    "        if it >= n_burn and (it - n_burn) % thin == 0:\n",
    "            samples.append((np.array(beta_list, dtype=float), pi.copy()))\n",
    "    return samples  \n",
    "\n",
    "# ----------------------------posterior predictive density as conformity score same as their paper------------------------------\n",
    "\n",
    "def batch_log_pred_dataset_M2(Y, X1, X2, betas, sigma=0.5):\n",
    "    T = len(betas); n = len(Y)\n",
    "    X = np.vstack([np.ones(n), X1, X2**2]).T  \n",
    "    out = np.empty((T, n), float)\n",
    "    for t in range(T):\n",
    "        mu = X @ betas[t]\n",
    "        out[t] = log_norm_pdf(Y, mu, sigma)\n",
    "    return out  \n",
    "def batch_log_pred_dataset_MK(Y, X1, X2, theta_samples, active_comps, sigma_list):\n",
    "    T = len(theta_samples); n = len(Y)\n",
    "    out = np.empty((T, n), float)\n",
    "    X1_mat = np.vstack([np.ones(n), X1, X2**2]).T\n",
    "    X2_mat = np.vstack([np.ones(n), X1**2, X1*X2]).T\n",
    "    X3_mat = np.vstack([np.ones(n), X1**2, np.cos(2*X2)]).T\n",
    "    for t in range(T):\n",
    "        betas_t, pi_t = theta_samples[t]\n",
    "        logs = []\n",
    "        for j,k in enumerate(active_comps):\n",
    "            if k == 0: mu = X1_mat @ betas_t[j]\n",
    "            elif k == 1: mu = X2_mat @ betas_t[j]\n",
    "            else:        mu = X3_mat @ betas_t[j]\n",
    "            logs.append(np.log(pi_t[j] + 1e-300) + log_norm_pdf(Y, mu, sigma_list[j]))\n",
    "        logs = np.stack(logs, axis=0)     \n",
    "        m = np.max(logs, axis=0)\n",
    "        out[t] = m + np.log(np.sum(np.exp(logs - m), axis=0))\n",
    "    return out  \n",
    "\n",
    "def log_f_star_grid_M2(y_grid, x1, x2, betas, sigma=0.5):\n",
    "    T = len(betas); Ny = len(y_grid)\n",
    "    out = np.empty((T, Ny), float)\n",
    "    phi = phi1(x1, x2)\n",
    "    for t in range(T):\n",
    "        mu = phi @ betas[t]\n",
    "        out[t] = log_norm_pdf(y_grid, mu, sigma)\n",
    "    return out\n",
    "\n",
    "def log_f_star_grid_MK(y_grid, x1, x2, theta_samples, active_comps, sigma_list):\n",
    "    T = len(theta_samples); Ny = len(y_grid)\n",
    "    out = np.empty((T, Ny), float)\n",
    "    for t in range(T):\n",
    "        betas_t, pi_t = theta_samples[t]\n",
    "        comps = []\n",
    "        for j,k in enumerate(active_comps):\n",
    "            if k == 0: mu = phi1(x1,x2) @ betas_t[j]\n",
    "            elif k == 1: mu = phi2(x1,x2) @ betas_t[j]\n",
    "            else:        mu = phi3(x1,x2) @ betas_t[j]\n",
    "            comps.append(np.log(pi_t[j] + 1e-300) + log_norm_pdf(y_grid, mu, sigma_list[j]))\n",
    "        comps = np.stack(comps, axis=0)  \n",
    "        m = np.max(comps, axis=0)\n",
    "        out[t] = m + np.log(np.sum(np.exp(comps - m), axis=0))\n",
    "    return out\n",
    "\n",
    "# AOI FROM THE PAPER\n",
    "\n",
    "def randomized_full_conformal_mask(logf_dataset_Tn, logf_star_TNy, alpha=0.2, rng=None):\n",
    "    \n",
    "    rng = default_rng() if rng is None else rng\n",
    "    T, n = logf_dataset_Tn.shape\n",
    "    Ny = logf_star_TNy.shape[1]\n",
    "    mask = np.zeros(Ny, dtype=bool)\n",
    "\n",
    "    for j in range(Ny):\n",
    "        a = logf_star_TNy[:, j]                  \n",
    "        mA = np.max(a); logZ = np.log(np.sum(np.exp(a - mA))) + mA\n",
    "        log_w = a - logZ                        \n",
    "\n",
    "        \n",
    "        M = log_w[:, None] + logf_dataset_Tn     \n",
    "        mcol = np.max(M, axis=0)\n",
    "        log_s_i = mcol + np.log(np.sum(np.exp(M - mcol), axis=0))   \n",
    "\n",
    "        \n",
    "        log_s_star = mA + np.log(np.sum(np.exp((a - logZ) + (a - mA))))  \n",
    "\n",
    "        lt = np.sum(log_s_i < log_s_star)\n",
    "        eq = np.sum(log_s_i == log_s_star)\n",
    "        u  = rng.random()\n",
    "        pval = (1.0 + lt + u*eq) / (n + 1.0)     \n",
    "        mask[j] = (pval > alpha)\n",
    "    return mask\n",
    "\n",
    "def pvalue_at_y_true(logf_dataset_Tn, logf_star_T, alpha=0.2, rng=None):\n",
    "    \n",
    "    rng = default_rng() if rng is None else rng\n",
    "    T, n = logf_dataset_Tn.shape\n",
    "    a = logf_star_T\n",
    "    mA = np.max(a); logZ = np.log(np.sum(np.exp(a - mA))) + mA\n",
    "    log_w = a - logZ\n",
    "\n",
    "    M = log_w[:, None] + logf_dataset_Tn\n",
    "    mcol = np.max(M, axis=0)\n",
    "    log_s_i = mcol + np.log(np.sum(np.exp(M - mcol), axis=0))\n",
    "    log_s_star = mA + np.log(np.sum(np.exp((a - logZ) + (a - mA))))\n",
    "    lt = np.sum(log_s_i < log_s_star)\n",
    "    eq = np.sum(log_s_i == log_s_star)\n",
    "    u  = rng.random()\n",
    "    return (1.0 + lt + u*eq) / (n + 1.0)\n",
    "\n",
    "\n",
    "def run_experiment2_full_conformal(\n",
    "    n_list=(100,300,600,1000),\n",
    "    E=3, alpha=0.2, grid_size=100, m_test=100,\n",
    "    T_samples=200, burn_M2=300, burn_MK=500, seed0=random.randint(1,1000000)\n",
    "):\n",
    "    rng = default_rng(seed0)\n",
    "    # ?\n",
    "    P = draw_generator_params(seed0)\n",
    "\n",
    "    \n",
    "    pilot = sample_from_generator(3000, P, seed0+7)\n",
    "    y_grid = np.linspace(pilot[\"Y\"].min()-1.0, pilot[\"Y\"].max()+1.0, grid_size)\n",
    "    dy = y_grid[1] - y_grid[0]\n",
    "\n",
    "    \n",
    "    (m1,v1), (m2,v2), (m3,v3) = true_prior_means_vars()\n",
    "    alpha3 = np.array([100.0,100.0,100.0])  \n",
    "    alpha2 = np.array([100.0,100.0])\n",
    "    sigma_true = 0.5\n",
    "    sigma3 = [sigma_true, sigma_true, sigma_true]\n",
    "    sigma2 = [sigma_true, sigma_true]\n",
    "\n",
    "    results = []\n",
    "    t0 = time.time()\n",
    "\n",
    "    for n in n_list:\n",
    "        t_n = time.time()\n",
    "        cov_M1, len_M1 = [], []\n",
    "        cov_M2, len_M2 = [], []\n",
    "        cov_M3, len_M3 = [], []\n",
    "\n",
    "        for rep in range(E):\n",
    "            seed_base = (n*97 + rep*313 + seed0) % (2**32 - 1)\n",
    "            \n",
    "            D = sample_from_generator(n, P, seed_base)\n",
    "            X1, X2, Y = D[\"X1\"].to_numpy(), D[\"X2\"].to_numpy(), D[\"Y\"].to_numpy()\n",
    "\n",
    "            \n",
    "            Phi1 = design_phi1(X1, X2)\n",
    "            Phi2 = design_phi2(X1, X2)\n",
    "            Phi3 = design_phi3(X1, X2)\n",
    "\n",
    "            \n",
    "            samp_M1 = gibbs_MK_mixture(\n",
    "                [Phi1, Phi2, Phi3], Y, K=3,\n",
    "                m0_list=[m1, m2, m3], v0_list=[v1, v2, v3],\n",
    "                alpha_dir=alpha3, sigma_list=sigma3,\n",
    "                rng=default_rng(seed_base+10),\n",
    "                n_burn=burn_MK, n_samp=T_samples, thin=1\n",
    "            )\n",
    "            \n",
    "            betas_M2 = gibbs_M2_single(\n",
    "                Phi1, Y, m0=m1, v0=v1, sigma=sigma_true,\n",
    "                rng=default_rng(seed_base+20),\n",
    "                n_burn=burn_M2, n_samp=T_samples, thin=1\n",
    "            )\n",
    "            \n",
    "            samp_M3 = gibbs_MK_mixture(\n",
    "                [Phi1, Phi2], Y, K=2,\n",
    "                m0_list=[m1, m2], v0_list=[v1, v2],\n",
    "                alpha_dir=alpha2, sigma_list=sigma2,\n",
    "                rng=default_rng(seed_base+30),\n",
    "                n_burn=burn_MK, n_samp=T_samples, thin=1\n",
    "            )\n",
    "\n",
    "            \n",
    "            logf_M1 = batch_log_pred_dataset_MK(Y, X1, X2, samp_M1, active_comps=[0,1,2], sigma_list=sigma3)\n",
    "            logf_M2 = batch_log_pred_dataset_M2(Y, X1, X2, betas_M2, sigma=sigma_true)\n",
    "            logf_M3 = batch_log_pred_dataset_MK(Y, X1, X2, samp_M3, active_comps=[0,1],   sigma_list=sigma2)\n",
    "\n",
    "            \n",
    "            Dtest = sample_from_generator(m_test, P, seed_base+3)\n",
    "            X1t, X2t, Yt = Dtest[\"X1\"].to_numpy(), Dtest[\"X2\"].to_numpy(), Dtest[\"Y\"].to_numpy()\n",
    "\n",
    "            \n",
    "            for j in range(m_test):\n",
    "                x1s, x2s, y_true = X1t[j], X2t[j], Yt[j]\n",
    "\n",
    "                \n",
    "                lstar_grid_M1 = log_f_star_grid_MK(y_grid, x1s, x2s, samp_M1, [0,1,2], sigma3)\n",
    "                mask_M1 = randomized_full_conformal_mask(logf_M1, lstar_grid_M1, alpha=alpha, rng=default_rng(seed_base+41+j))\n",
    "                len_M1.append(mask_M1.sum() * dy)\n",
    "                \n",
    "                lstar_true_M1 = log_f_star_grid_MK(np.array([y_true]), x1s, x2s, samp_M1, [0,1,2], sigma3)[:,0]\n",
    "                pval_M1 = pvalue_at_y_true(logf_M1, lstar_true_M1, alpha=alpha, rng=default_rng(seed_base+42+j))\n",
    "                cov_M1.append(pval_M1 > alpha)\n",
    "\n",
    "                \n",
    "                lstar_grid_M2 = log_f_star_grid_M2(y_grid, x1s, x2s, betas_M2, sigma=sigma_true)\n",
    "                mask_M2 = randomized_full_conformal_mask(logf_M2, lstar_grid_M2, alpha=alpha, rng=default_rng(seed_base+43+j))\n",
    "                len_M2.append(mask_M2.sum() * dy)\n",
    "                lstar_true_M2 = log_f_star_grid_M2(np.array([y_true]), x1s, x2s, betas_M2, sigma=sigma_true)[:,0]\n",
    "                pval_M2 = pvalue_at_y_true(logf_M2, lstar_true_M2, alpha=alpha, rng=default_rng(seed_base+44+j))\n",
    "                cov_M2.append(pval_M2 > alpha)\n",
    "\n",
    "                \n",
    "                lstar_grid_M3 = log_f_star_grid_MK(y_grid, x1s, x2s, samp_M3, [0,1], sigma2)\n",
    "                mask_M3 = randomized_full_conformal_mask(logf_M3, lstar_grid_M3, alpha=alpha, rng=default_rng(seed_base+45+j))\n",
    "                len_M3.append(mask_M3.sum() * dy)\n",
    "                lstar_true_M3 = log_f_star_grid_MK(np.array([y_true]), x1s, x2s, samp_M3, [0,1], sigma2)[:,0]\n",
    "                pval_M3 = pvalue_at_y_true(logf_M3, lstar_true_M3, alpha=alpha, rng=default_rng(seed_base+46+j))\n",
    "                cov_M3.append(pval_M3 > alpha)\n",
    "\n",
    "        \n",
    "        def summarize(covs, lens):\n",
    "            cov_arr = np.array(covs, float); len_arr = np.array(lens, float)\n",
    "            reps = len(cov_arr)  \n",
    "            return (cov_arr.mean(),\n",
    "                    cov_arr.std(ddof=1)/np.sqrt(reps),\n",
    "                    len_arr.mean(),\n",
    "                    len_arr.std(ddof=1)/np.sqrt(reps))\n",
    "        c1,se1,l1,seL1 = summarize(cov_M1, len_M1)\n",
    "        c2,se2,l2,seL2 = summarize(cov_M2, len_M2)\n",
    "        c3,se3,l3,seL3 = summarize(cov_M3, len_M3)\n",
    "\n",
    "        elapsed_n = time.time() - t_n\n",
    "        results.append({\"model\":\"M1_trueK3\",\"n\":n,\"coverage_mean\":c1,\"coverage_se\":se1,\"length_mean\":l1,\"length_se\":seL1,\"time_n\":round(elapsed_n,2)})\n",
    "        results.append({\"model\":\"M2_K1\",\"n\":n,\"coverage_mean\":c2,\"coverage_se\":se2,\"length_mean\":l2,\"length_se\":seL2,\"time_n\":round(elapsed_n,2)})\n",
    "        results.append({\"model\":\"M3_K2\",\"n\":n,\"coverage_mean\":c3,\"coverage_se\":se3,\"length_mean\":l3,\"length_se\":seL3,\"time_n\":round(elapsed_n,2)})\n",
    "\n",
    "    total_elapsed = time.time() - t0\n",
    "    return pd.DataFrame(results), y_grid, total_elapsed\n",
    "\n",
    "# plot\n",
    "\n",
    "if __name__ == \"__main__\":\n",
    "    res, y_grid, total_time = run_experiment2_full_conformal(\n",
    "        n_list=(100, 300,600,1000),\n",
    "        E=10, alpha=0.2, grid_size=500, m_test=50,\n",
    "        T_samples=1000, burn_M2=300, burn_MK=500, seed0=random.randint(1,1000000)\n",
    "    )\n",
    "\n",
    " \n",
    "    plt.figure()\n",
    "    for model in [\"M1_trueK3\",\"M2_K1\",\"M3_K2\"]:\n",
    "        dfm = res[res[\"model\"]==model].sort_values(\"n\")\n",
    "        plt.errorbar(dfm[\"n\"], dfm[\"coverage_mean\"], yerr=dfm[\"coverage_se\"],\n",
    "                     marker=\"o\", capsize=3, label=model)\n",
    "    plt.axhline(0.8, linestyle=\"--\")\n",
    "    plt.title(\"CB on Simulation Data Average Coverage vs n\")\n",
    "    plt.xlabel(\"n\"); plt.ylabel(\"Coverage Rate\"); plt.legend(); plt.show()\n",
    "\n",
    "    plt.figure()\n",
    "    for model in [\"M1_trueK3\",\"M2_K1\",\"M3_K2\"]:\n",
    "        dfm = res[res[\"model\"]==model].sort_values(\"n\")\n",
    "        plt.errorbar(dfm[\"n\"], dfm[\"length_mean\"], yerr=dfm[\"length_se\"],\n",
    "                     marker=\"o\", capsize=3, label=model)\n",
    "    plt.title(\"CB on Simulation Data Average Set Length vs n\")\n",
    "    plt.xlabel(\"n\"); plt.ylabel(\"Average Set Length\"); plt.legend(); plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "061b2c6e-3083-442f-83c6-9bfc8dfc56e4",
   "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.12.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
