{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9cbe1d4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import ot\n",
    "from scipy.spatial import cKDTree\n",
    "from sklearn.linear_model import LinearRegression\n",
    "from sklearn.metrics import r2_score\n",
    "import time\n",
    "from concurrent.futures import ProcessPoolExecutor, as_completed"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b8baed4d",
   "metadata": {},
   "outputs": [],
   "source": [
    "dimensions = [3, 5, 10] # dimension\n",
    "N = 2000 # support size of base measures\n",
    "sample_sizes = np.arange(10,101,10)\n",
    "T = 100 # number of iterations per sample size"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4df9a7fd",
   "metadata": {},
   "outputs": [],
   "source": [
    "def k_star(x):\n",
    "  y = x.copy()\n",
    "  if np.random.rand() < 0.5:\n",
    "    y[0] += 1\n",
    "  else:\n",
    "    y[0] -= 1\n",
    "  return y\n",
    "\n",
    "np.random.seed(1234)\n",
    "mu_by_dimension = {}\n",
    "nu_by_dimension = {}\n",
    "for d in dimensions:\n",
    "  mu_by_dimension[d] = np.concatenate((np.zeros((N,1)), np.random.rand(N,d-1)), axis=1)\n",
    "  nu_by_dimension[d] = np.concatenate((np.zeros((N,1)), np.random.rand(N,d-1)), axis=1)\n",
    "  nu_by_dimension[d] = np.apply_along_axis(k_star, 1, nu_by_dimension[d])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7318cad3",
   "metadata": {},
   "outputs": [],
   "source": [
    "NN_E1_errors = {}\n",
    "NN_L1_errors = {}\n",
    "for d in dimensions:\n",
    "  print(f'd:{d}')\n",
    "  mu = mu_by_dimension[d]\n",
    "  nu = nu_by_dimension[d]\n",
    "\n",
    "  C = ot.dist(mu, nu, metric='euclidean')\n",
    "  pi_N = ot.emd(np.ones(N)/N, np.ones(N)/N, C)\n",
    "  sigma_N = pi_N.argmax(axis=1)\n",
    "  opt_destinations = nu[sigma_N]\n",
    "\n",
    "  for n in sample_sizes:\n",
    "    print(f'n:{n}')\n",
    "    NN_E1_errors[(d,n)] = []\n",
    "    NN_L1_errors[(d,n)] = []\n",
    "    for t in range(T):\n",
    "      indices_mu = np.random.choice(N, n, replace=True)\n",
    "      indices_nu = np.random.choice(N, n, replace=True)\n",
    "      mu_n = mu[indices_mu,:]\n",
    "      nu_n = nu[indices_nu,:]\n",
    "\n",
    "      # compute NN estimator\n",
    "      C = ot.dist(mu_n, nu_n, metric='euclidean')\n",
    "      pi_n = ot.emd(np.ones(n)/n, np.ones(n)/n, C)\n",
    "      sigma_n = pi_n.argmax(axis=1)\n",
    "      tree = cKDTree(mu_n)\n",
    "      _, nn_indices = tree.query(mu)\n",
    "      destinations = nu_n[sigma_n[nn_indices]]\n",
    "\n",
    "      NN_L1_errors[(d,n)].append(np.linalg.norm(destinations - opt_destinations, axis=1).mean())\n",
    "\n",
    "      L1_transport_cost = np.linalg.norm(destinations - mu, axis=1).mean()\n",
    "      opt_gap = max(L1_transport_cost - 1,0)\n",
    "      C = ot.dist(destinations, nu, metric='euclidean')\n",
    "      feas_gap = ot.emd2(np.ones(N)/N, np.ones(N)/N, C)\n",
    "      NN_E1_errors[(d,n)].append(opt_gap + feas_gap)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "72da6e7f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "with open('NN_errors_plot1.pkl', 'wb') as f:\n",
    "  pickle.dump((NN_E1_errors,NN_L1_errors), f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ef739f4f",
   "metadata": {},
   "outputs": [],
   "source": [
    "rounding_E1_errors = {}\n",
    "for d in dimensions:\n",
    "  print(f'd:{d}')\n",
    "  mu = np.concatenate((np.zeros((N,1)), np.random.rand(N,d-1)), axis=1)\n",
    "  nu = np.concatenate((np.zeros((N,1)), np.random.rand(N,d-1)), axis=1)\n",
    "  nu = np.apply_along_axis(k_star, 1, nu)\n",
    "\n",
    "  for n in sample_sizes:\n",
    "    print(f'n:{n}')\n",
    "    rounding_E1_errors[(d,n)] = []\n",
    "    for t in range(T):\n",
    "      indices_mu = np.random.choice(N, n, replace=True)\n",
    "      indices_nu = np.random.choice(N, n, replace=True)\n",
    "      mu_n = mu[indices_mu,:]\n",
    "      nu_n = nu[indices_nu,:]\n",
    "\n",
    "      # compute rounding estimator\n",
    "      delta = n**(-1/(d+2))\n",
    "      mu_n_rounded = delta * np.round(mu_n / delta)\n",
    "      C = ot.dist(mu_n_rounded, nu_n, metric='euclidean')\n",
    "      pi_n = ot.emd(np.ones(n)/n, np.ones(n)/n, C)\n",
    "\n",
    "      mu_rounded = delta * np.round(mu / delta)\n",
    "      tree = cKDTree(mu_n_rounded)\n",
    "      _, nn_indices = tree.query(mu_rounded)\n",
    "      kernel_dists_raw = pi_n[nn_indices,:] # N x n\n",
    "      row_sums = kernel_dists_raw.sum(axis=1, keepdims=True)\n",
    "      kernel_dists = kernel_dists_raw / row_sums\n",
    "\n",
    "      L1_transport_cost = 0\n",
    "      for k in range(N):\n",
    "        for l in range(n):\n",
    "          L1_transport_cost += np.linalg.norm(mu[k,:] - nu_n[l,:]) * kernel_dists[k,l]\n",
    "      L1_transport_cost /= N\n",
    "\n",
    "      destination_weights = kernel_dists.mean(axis=0)\n",
    "\n",
    "      opt_gap = max(L1_transport_cost - 1,0)\n",
    "      C = ot.dist(nu_n, nu, metric='euclidean')\n",
    "      feas_gap = ot.emd2(destination_weights, np.ones(N)/N, C)\n",
    "      rounding_E1_errors[(d,n)].append(opt_gap + feas_gap)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "52257b0f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "with open('rounding_errors_plot1_23510_1pm.pkl', 'wb') as f:\n",
    "  pickle.dump(rounding_E1_errors, f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "a48cbef8",
   "metadata": {},
   "outputs": [],
   "source": [
    "def k_star(x):\n",
    "  return x + np.sign(x-np.ones(x.shape)/2)\n",
    "\n",
    "np.random.seed(1234)\n",
    "mu_by_dimension = {}\n",
    "nu_by_dimension = {}\n",
    "for d in dimensions:\n",
    "  mu_by_dimension[d] = np.random.rand(N,d)\n",
    "  nu_by_dimension[d] = np.random.rand(N,d)\n",
    "  nu_by_dimension[d] = np.apply_along_axis(k_star, 1, nu_by_dimension[d])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "03ac3fd0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x311f21940>"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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5nfw1i6n/n2355TKT9QoPSVRTkf7ButpKlR8tP9rfbj/qL6sOKlundprZM8LFBdv09ETNa2qtbn6O6fYJZsish4AiBWtPf5H+vzR3HTDenTS5oPHGp07Ai0rXDUHEjPhxQzRCi8AACB8Qjl/x9Sj0gAAAM0hvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEshvAAAAEuJaHjZsGGDJk2apG7duslms2nFihXNzrN+/XoNHTpUdrtdffr00bJlyyJZIgAAsJiIhpeqqiplZ2dr8eLFQU1fUlKi6667TldddZW2bdumu+66S7feeqvefffdSJYJAAAspF0kF37ttdfq2muvDXr6JUuWKCsrS0888YQk6ZJLLtH777+vJ598UuPHj49UmQAAwEJiqs1LYWGh8vPzG7w3fvx4FRYWNjlPTU2NPB5PgxcAAGi7Yiq8uN1uZWRkNHgvIyNDHo9HJ0+e9DtPQUGBHA6H75WZmRmNUgEAgEliKry0xLx581RZWel7HT582OySAABABEW0zUuonE6nysrKGrxXVlam1NRUtW/f3u88drtddrs9GuUBAIAYEFNXXvLy8rR27doG761evVp5eXkmVQQAAGJNRMPLiRMntG3bNm3btk3S2Ueht23bpkOHDkk6e8tn2rRpvul/9rOf6bPPPtOvfvUr7d69W08//bT+9re/6e67745kmQAAwEIiGl4++eQTDRkyREOGDJEkzZkzR0OGDNH9998vSSotLfUFGUnKysrS22+/rdWrVys7O1tPPPGEnn32WR6TBgAAPjbDMAyziwgnj8cjh8OhyspKpaamml0OAAAIQijn75hq8wIAANAcwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALCUqISXxYsXq2fPnkpKStKIESO0adOmJqddtmyZbDZbg1dSUlI0ygQAABYQ8fDyyiuvaM6cOZo/f762bNmi7OxsjR8/XkeOHGlyntTUVJWWlvpeBw8ejHSZAADAIiIeXn77299q5syZmjFjhgYMGKAlS5YoOTlZzz33XJPz2Gw2OZ1O3ysjIyPSZQIAAIuIaHipra3V5s2blZ+ff26FcXHKz89XYWFhk/OdOHFCPXr0UGZmpiZPnqydO3c2OW1NTY08Hk+DFwAAaLsiGl6OHj2qurq6RldOMjIy5Ha7/c7Tr18/Pffcc3rjjTf00ksvyev1atSoUfr888/9Tl9QUCCHw+F7ZWZmhv1zAACA2BFzTxvl5eVp2rRpysnJ0ZgxY/Taa6+pS5cueuaZZ/xOP2/ePFVWVvpehw8fjnLFAAAgmtpFcuGdO3dWfHy8ysrKGrxfVlYmp9MZ1DISEhI0ZMgQ7du3z+/v7Xa77HZ7q2sFAADWENErL4mJiRo2bJjWrl3re8/r9Wrt2rXKy8sLahl1dXXasWOHXC5XpMoEAAAWEtErL5I0Z84cTZ8+XcOHD1dubq4WLVqkqqoqzZgxQ5I0bdo0XXjhhSooKJAkPfjggxo5cqT69OmjiooKPfbYYzp48KBuvfXWSJcKAAAsIOLh5frrr9dXX32l+++/X263Wzk5OVq1apWvEe+hQ4cUF3fuAtCxY8c0c+ZMud1uderUScOGDdOHH36oAQMGRLpUAABgATbDMAyziwgnj8cjh8OhyspKpaamml0OAAAIQijn75h72ggAACAQwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALAUwgsAALCUqISXxYsXq2fPnkpKStKIESO0adOmgNO/+uqr6t+/v5KSkjRo0CC988470SgTAABYQMTDyyuvvKI5c+Zo/vz52rJli7KzszV+/HgdOXLE7/Qffvihpk6dqltuuUVbt27VlClTNGXKFBUVFUW6VAAAYAE2wzCMSK5gxIgRuuyyy/TUU09JkrxerzIzM/WLX/xCc+fObTT99ddfr6qqKr311lu+90aOHKmcnBwtWbKk2fV5PB45HA5VVlYqNTU1fB8EAABETCjn74heeamtrdXmzZuVn59/boVxccrPz1dhYaHfeQoLCxtML0njx49vcnoAAPDt0i6SCz969Kjq6uqUkZHR4P2MjAzt3r3b7zxut9vv9G632+/0NTU1qqmp8f3s8XhaWTUAAIhlln/aqKCgQA6Hw/fKzMw0uyQAABBBEQ0vnTt3Vnx8vMrKyhq8X1ZWJqfT6Xcep9MZ0vTz5s1TZWWl73X48OHwFA8AAGJSRMNLYmKihg0bprVr1/re83q9Wrt2rfLy8vzOk5eX12B6SVq9enWT09vtdqWmpjZ4AQCAtiuibV4kac6cOZo+fbqGDx+u3NxcLVq0SFVVVZoxY4Ykadq0abrwwgtVUFAgSbrzzjs1ZswYPfHEE7ruuuv08ssv65NPPtHSpUsjXSoAALCAiIeX66+/Xl999ZXuv/9+ud1u5eTkaNWqVb5GuYcOHVJc3LkLQKNGjdLy5ct177336je/+Y369u2rFStWaODAgZEuFQAAWEDE+3mJNvp5AQDAemKmnxcAAIBwI7wAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABLIbwAAABL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ggVGhw4djK1btxpbt241jh8/7pumX79+xmuvveb7+ZFHHjHS0tKMN954w9i+fbsxefJkIysryzh58qQZHyFqJkyYYAwZMsTYuHGj8f777xt9+/Y1pk6d6vv9559/bvTr18/YuHGjYRiGsW/fPuPBBx80PvnkE6OkpMR44403jF69ehmjR4826yNExMsvv2zY7XZj2bJlRnFxsXHbbbcZaWlphtvtNgzDMG666SZj7ty5vuk/+OADo127dsbjjz9u7Nq1y5g/f76RkJBg7Nixw6yPEDWhbqsFCxYY7777rrF//35j8+bNxg033GAkJSUZO3fuNOsjRM3x48d9xyNJxm9/+1tj69atxsGDBw3DMIy5c+caN910k2/6zz77zEhOTjZ++ctfGrt27TIWL15sxMfHG6tWrTLrI0RNqNvqySefNFasWGHs3bvX2LFjh3HnnXcacXFxxpo1a8z6CFEza9Ysw+FwGOvXrzdKS0t9r+rqat80sXDMIrw0Y/r06YakRq9//vOfvmkkGc8//7zvZ6/Xa9x3331GRkaGYbfbjXHjxhl79uyJfvFR9vXXXxtTp041OnToYKSmphozZsxoEPJKSkoabLtDhw4Zo0ePNtLT0w273W706dPH+OUvf2lUVlaa9Aki5w9/+IPRvXt3IzEx0cjNzTU++ugj3+/GjBljTJ8+vcH0f/vb34yLL77YSExMNC699FLj7bffjnLF5gllW911112+aTMyMoyJEycaW7ZsMaHq6Kt/nPebr/rtM336dGPMmDGN5snJyTESExONXr16NThutWWhbquFCxcavXv3NpKSkoz09HRj7Nixxrp168wpPsr8badvnuNi4Zhl+3exAAAAlvCtbPMCAACsi/ACAAAshfACAAAshfACAAAshfACAAAshfACAAAshfACAAAshfACAAAshfACAAAshfACAAAshfACAAAshfACAAAs5f8D2gHZSjc/TIoAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.scatter(nu_by_dimension[2][:,0], nu_by_dimension[2][:,1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "43055177",
   "metadata": {},
   "outputs": [],
   "source": [
    "NN_E1_errors = {}\n",
    "NN_L1_errors = {}\n",
    "\n",
    "for d in dimensions:\n",
    "  mu = mu_by_dimension[d]\n",
    "  nu = nu_by_dimension[d]\n",
    "\n",
    "  C = ot.dist(mu, nu, metric='euclidean')\n",
    "  pi_N = ot.emd(np.ones(N)/N, np.ones(N)/N, C)\n",
    "  sigma_N = pi_N.argmax(axis=1)\n",
    "  opt_destinations = nu[sigma_N]\n",
    "\n",
    "  for n in sample_sizes:\n",
    "    print(f'd: {d}, n:{n}')\n",
    "    NN_E1_errors[(d,n)] = []\n",
    "    NN_L1_errors[(d,n)] = []\n",
    "    for t in range(T):\n",
    "      indices_mu = np.random.choice(N, n, replace=True)\n",
    "      indices_nu = np.random.choice(N, n, replace=True)\n",
    "      mu_n = mu[indices_mu,:]\n",
    "      nu_n = nu[indices_nu,:]\n",
    "\n",
    "      # compute NN estimator\n",
    "      C = ot.dist(mu_n, nu_n, metric='euclidean')\n",
    "      pi_n = ot.emd(np.ones(n)/n, np.ones(n)/n, C)\n",
    "      sigma_n = pi_n.argmax(axis=1)\n",
    "      tree = cKDTree(mu_n)\n",
    "      _, nn_indices = tree.query(mu)\n",
    "      destinations = nu_n[sigma_n[nn_indices]]\n",
    "\n",
    "      NN_L1_errors[(d,n)].append(np.linalg.norm(destinations - opt_destinations, axis=1).mean())\n",
    "\n",
    "      L1_transport_cost = np.linalg.norm(destinations - mu, axis=1).mean()\n",
    "      opt_gap = max(L1_transport_cost - np.sqrt(d),0)\n",
    "      C = ot.dist(destinations, nu, metric='euclidean')\n",
    "      feas_gap = ot.emd2(np.ones(N)/N, np.ones(N)/N, C)\n",
    "      NN_E1_errors[(d,n)].append(opt_gap + feas_gap)\n",
    "    print(np.mean(NN_E1_errors[(d,n)]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "640b5075",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "with open('NN_errors_plot2.pkl', 'wb') as f:\n",
    "  pickle.dump((NN_E1_errors,NN_L1_errors), f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "db008103",
   "metadata": {},
   "outputs": [],
   "source": [
    "rounding_E1_errors = {}\n",
    "for d in dimensions:\n",
    "  print(f'd:{d}')\n",
    "  mu = np.concatenate((np.zeros((N,1)), np.random.rand(N,d-1)), axis=1)\n",
    "  nu = np.concatenate((np.zeros((N,1)), np.random.rand(N,d-1)), axis=1)\n",
    "  nu = np.apply_along_axis(k_star, 1, nu)\n",
    "\n",
    "  for n in sample_sizes:\n",
    "    print(f'n:{n}')\n",
    "    rounding_E1_errors[(d,n)] = []\n",
    "    for t in range(T):\n",
    "      indices_mu = np.random.choice(N, n, replace=True)\n",
    "      indices_nu = np.random.choice(N, n, replace=True)\n",
    "      mu_n = mu[indices_mu,:]\n",
    "      nu_n = nu[indices_nu,:]\n",
    "\n",
    "      # compute rounding estimator\n",
    "      delta = n**(-1/(d+2))\n",
    "      mu_n_rounded = delta * np.round(mu_n / delta)\n",
    "      C = ot.dist(mu_n_rounded, nu_n, metric='euclidean')\n",
    "      pi_n = ot.emd(np.ones(n)/n, np.ones(n)/n, C)\n",
    "\n",
    "      mu_rounded = delta * np.round(mu / delta)\n",
    "      tree = cKDTree(mu_n_rounded)\n",
    "      _, nn_indices = tree.query(mu_rounded)\n",
    "      kernel_dists_raw = pi_n[nn_indices,:] # N x n\n",
    "      row_sums = kernel_dists_raw.sum(axis=1, keepdims=True)\n",
    "      kernel_dists = kernel_dists_raw / row_sums\n",
    "\n",
    "      L1_transport_cost = 0\n",
    "      for k in range(N):\n",
    "        for l in range(n):\n",
    "          L1_transport_cost += np.linalg.norm(mu[k,:] - nu_n[l,:]) * kernel_dists[k,l]\n",
    "      L1_transport_cost /= N\n",
    "\n",
    "      destination_weights = kernel_dists.mean(axis=0)\n",
    "\n",
    "      opt_gap = max(L1_transport_cost - np.sqrt(d),0)\n",
    "      C = ot.dist(nu_n, nu, metric='euclidean')\n",
    "      feas_gap = ot.emd2(destination_weights, np.ones(N)/N, C)\n",
    "      rounding_E1_errors[(d,n)].append(opt_gap + feas_gap)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "79f2cb7b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "with open('rounding_errors_plot2.pkl', 'wb') as f:\n",
    "  pickle.dump(rounding_E1_errors, f)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "non-myopic",
   "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.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
