{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "62098abb",
   "metadata": {},
   "source": [
    "# Sensitivity/Ablation Analysis\n",
    "\n",
    "Create a table/boxplot about approximation gap + computational time ablation on different $n$ and $|\\Lambda|$ for different optiization settings\n",
    "\n",
    "**Row**: $|\\Lambda|$ and $n$\n",
    "\n",
    "**Column**: Optimization settings\n",
    "\n",
    "**Sub Column**: Approximation Gap, Computational Time,\n",
    "\n",
    "----\n",
    "\n",
    "Separate chart to include posthoc and validity"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "4651f1ff",
   "metadata": {},
   "outputs": [],
   "source": [
    "# from utils.setting import config\n",
    "# import numpy as np\n",
    "# import matplotlib.pyplot as plt\n",
    "\n",
    "# lam_list = \n",
    "\n",
    "# # opt: should have a way to simulate data + find B in a sample based manner\n",
    "\n",
    "# def runner(lam_list, nsample, ntrial, opt):\n",
    "#     '''\n",
    "#     - lam_list: list of float\n",
    "#     - nsample: int\n",
    "#     - opt: a optimization Module with some embedded features\n",
    "#     '''\n",
    "#     for _ in range(ntrial):\n",
    "#         data = opt.simulate_data(nsample) # [ nsample, data_dim ]\n",
    "#         for lam in lam_list:\n",
    "#             opt.lam = lam\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "0e4710f6",
   "metadata": {},
   "outputs": [],
   "source": [
    "from model.v1.sampler import Sampler\n",
    "import numpy as np\n",
    "from model.v1.crc import ConformalRegretControl\n",
    "from utils.setting import config\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm import tqdm\n",
    "import time\n",
    "\n",
    "import os\n",
    "from model.v1.predictor import OffsetScalarPredictor\n",
    "from model.v1.crc import ConformalRegretControl\n",
    "from model.v1.newsvendor import Newsvendor\n",
    "from model.v1.lp import LinearProgramming, circle_as_polytope\n",
    "from model.v1.shortest_path import ShortestPath, build_incidence_many_chains\n",
    "from model.v1.portfolio import PortfolioOptimization\n",
    "\n",
    "def runner(crc, mean, sigma,\n",
    "           nsample_list=[10, 100, 100],\n",
    "           nlam_max = 30,\n",
    "           ntrial=10,\n",
    "           B=None, width=None, use_gaussian=False, *args, **kwds):\n",
    "    lam_list = np.linspace(0.01, 1, nlam_max)\n",
    "    \n",
    "    # ground true\n",
    "    x = Sampler().uniform(mean, sigma, 100) if not use_gaussian else Sampler().gaussian(mean, width, sigma, 100) # [ nsample, data_dim ]\n",
    "    y = Sampler().uniform(mean, sigma, 100) if not use_gaussian else Sampler().gaussian(mean, width, sigma, 100) # [ nsample, data_dim ]\n",
    "    true_list = []\n",
    "    for lam in lam_list:\n",
    "        alpha_R, alpha_I, _, _ = crc.estimate(x, y, lam=lam, B=B, also_mc=True)   # two scalars\n",
    "        true_list.append(np.array([alpha_R, alpha_I]))\n",
    "    true_arr =  np.array(true_list) # [ nlam_max, 2 ]\n",
    "\n",
    "    # estimate\n",
    "    result_list = []\n",
    "    pbar = tqdm(total=ntrial * len(nsample_list) * nlam_max)\n",
    "    for _ in range(ntrial):\n",
    "        for nsample in nsample_list:\n",
    "            \n",
    "            # generate calibration data\n",
    "            x = Sampler().uniform(mean, sigma, nsample) if not use_gaussian else Sampler().gaussian(mean, width, sigma, nsample) # [ nsample, data_dim ]\n",
    "            y = Sampler().uniform(mean, sigma, nsample) if not use_gaussian else Sampler().gaussian(mean, width, sigma, nsample) # [ nsample, data_dim ]\n",
    "            \n",
    "            # collect estimation\n",
    "            for lam in lam_list:\n",
    "                \n",
    "                start = time.perf_counter()\n",
    "                alpha_R, alpha_I, _, _ = crc.estimate(x, y, lam=lam, B=B, also_mc=True)   # two scalars\n",
    "                elapsed = time.perf_counter() - start  # scalar\n",
    "\n",
    "                result_list.append(np.array([alpha_R, alpha_I, elapsed]))\n",
    "                pbar.update(1)\n",
    "    pbar.close()\n",
    "    arr = np.array(result_list) # [ ntrial * len(nsample_list) * nlam_max, 3 ]\n",
    "    arr = arr.reshape(ntrial, len(nsample_list), nlam_max, -1) # [ ntrial, len(nsample_list), nlam_max, 3 ]\n",
    "\n",
    "    arr = np.concatenate([\n",
    "        arr, # [ ntrial, len(nsample_list), nlam_max, 3 ]\n",
    "        true_arr[None, None, :, :] * np.ones_like(arr[..., :2]) # [ ntrial, len(nsample_list), nlam_max, 2 ]\n",
    "    ], axis=-1)\n",
    "\n",
    "    return arr # [ ntrial, len(nsample_list), nlam_max, 3 + 2 ] est_R, est_I, time, true_R, true_I"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "71e45584",
   "metadata": {},
   "outputs": [],
   "source": [
    "nsample_list = [10, 20, 30]\n",
    "nlam_max     = 30\n",
    "ntrial       = 10"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "91aef55c",
   "metadata": {},
   "source": [
    "# Linear Programming"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "b32e3061",
   "metadata": {},
   "outputs": [],
   "source": [
    "mean    = [-1.1, -1.]\n",
    "sigma   = [1., 1.]\n",
    "\n",
    "model_kwds = {'offset': mean}\n",
    "model = OffsetScalarPredictor(**model_kwds)\n",
    "A, b = circle_as_polytope(R=1.0, m=32)\n",
    "A_pos = np.array([\n",
    "    [-1., 0.],\n",
    "    [0., -1.]\n",
    "])\n",
    "b_pos = np.array([0., 0.])\n",
    "A, b = np.concatenate([A, A_pos]), np.concatenate([b, b_pos])\n",
    "\n",
    "opt_kwds = {\n",
    "    'model':    model,\n",
    "    'feasible_region':  None,    # [ nres**2, 2 ]\n",
    "    'outcome_space':    None,    # [ nres**2, 2 ]\n",
    "    'A':    A,\n",
    "    'b':    b,\n",
    "}\n",
    "optimization = LinearProgramming(**opt_kwds)\n",
    "\n",
    "crc_kwds = {'optimization': optimization}\n",
    "crc = ConformalRegretControl(**crc_kwds)\n",
    "\n",
    "path = 'cache/1_28_exp4/result_list_lp.npy'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "cbaf3b19",
   "metadata": {},
   "outputs": [],
   "source": [
    "# config()\n",
    "\n",
    "# run_kwds = {\n",
    "#     'crc':      crc,\n",
    "#     'mean':     mean,\n",
    "#     'sigma':    sigma,\n",
    "#     'ntrial':   ntrial,\n",
    "#     'nsample_list':  nsample_list,\n",
    "#     'nlam_max':     nlam_max,\n",
    "#     'B':        0.8,\n",
    "#     'width':        None,\n",
    "#     'use_gaussian': False\n",
    "# }\n",
    "\n",
    "# result_list = runner(**run_kwds)\n",
    "# np.save(path, result_list)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a1e841bf",
   "metadata": {},
   "source": [
    "# Newsvendor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "83a16cd9",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Newsvendor model\n",
    "mean    = 2\n",
    "sigma   = 1 \n",
    "\n",
    "model_kwds = {'offset': [mean]} # unbiased if set to \"mean\"\n",
    "model = OffsetScalarPredictor(**model_kwds)\n",
    "\n",
    "opt_kwds = {\n",
    "    'model':    model,\n",
    "    'p':        3,\n",
    "    'v':        0,\n",
    "    'c':        2\n",
    "}\n",
    "optimization = Newsvendor(**opt_kwds)\n",
    "crc_kwds = {'optimization': optimization}\n",
    "crc = ConformalRegretControl(**crc_kwds)\n",
    "\n",
    "path = 'cache/1_28_exp4/result_list_nv.npy'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "39df8a3b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# config()\n",
    "\n",
    "# run_kwds = {\n",
    "#     'crc':      crc,\n",
    "#     'mean':     mean,\n",
    "#     'sigma':    sigma,\n",
    "#     'ntrial':   ntrial,\n",
    "#     'nsample_list':  nsample_list,\n",
    "#     'nlam_max':     nlam_max,\n",
    "#     'B':        1.5,\n",
    "#     'width':        None,\n",
    "#     'use_gaussian': False\n",
    "# }\n",
    "\n",
    "# result_list = runner(**run_kwds)\n",
    "# np.save(path, result_list)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "aa644f43",
   "metadata": {},
   "source": [
    "## Portfolio"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "fa202f59",
   "metadata": {},
   "outputs": [],
   "source": [
    "mean    = [2., 2.]\n",
    "sigma   = [1., 1.]\n",
    "nsample = 10\n",
    "\n",
    "model_kwds = {'offset': mean}   # unbiased if set to \"mean\"\n",
    "model = OffsetScalarPredictor(**model_kwds)\n",
    "\n",
    "opt_kwds = {\n",
    "    'model':    model,\n",
    "    'feasible_region':  None,    # [ nres**2, 2 ]\n",
    "    'outcome_space':    None,    # [ nres**2, 2 ]\n",
    "    'gamma':    1.,\n",
    "    'alpha':    0.95,\n",
    "    'nsample':  nsample,\n",
    "    'ndim':     2\n",
    "}\n",
    "opt = PortfolioOptimization(**opt_kwds)\n",
    "crc_kwds = {'optimization': opt}\n",
    "crc = ConformalRegretControl(**crc_kwds)\n",
    "\n",
    "path = 'cache/1_28_exp4/result_list_pf.npy'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "1414fc33",
   "metadata": {},
   "outputs": [],
   "source": [
    "# config()\n",
    "\n",
    "# run_kwds = {\n",
    "#     'crc':      crc,\n",
    "#     'mean':     mean,\n",
    "#     'sigma':    sigma,\n",
    "#     'ntrial':   ntrial,\n",
    "#     'nsample_list':  nsample_list,\n",
    "#     'nlam_max':     nlam_max,\n",
    "#     'B':        3,\n",
    "#     'width':        None,\n",
    "#     'use_gaussian': False\n",
    "# }\n",
    "\n",
    "# result_list = runner(**run_kwds)\n",
    "# np.save(path, result_list)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "98c2e16a",
   "metadata": {},
   "source": [
    "# Shortest Path"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "9332d0d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "\n",
    "nchains = 5\n",
    "arr     = np.array([ (11 - (2.**(-np.arange(j + 1))).sum()) / (j + 1) for j in range(nchains)])\n",
    "mean    = np.repeat(arr, np.arange(len(arr)) + 1)       # [ nedge ] np\n",
    "width   = np.ones_like(mean) \n",
    "sigma   = np.ones_like(mean) * 0.2                            # [ nedge ] np\n",
    "\n",
    "model_kwds = {\n",
    "    'offset': np.array(mean)  # unbiased if set to \"mean\"\n",
    "}\n",
    "model = OffsetScalarPredictor(**model_kwds)\n",
    "\n",
    "A, b = build_incidence_many_chains(nchains)  # [ nnode, nedge ] and [ nnode ], both np\n",
    "opt_kwds = {\n",
    "    'model':    model,\n",
    "    'feasible_region':  None,    # [ nres**2, 2 ] or None\n",
    "    'outcome_space':    None,    # [ nres**2, 2 ] or None\n",
    "    'A':    A,\n",
    "    'b':    b,\n",
    "    'width':    width\n",
    "}\n",
    "opt = ShortestPath(**opt_kwds)\n",
    "crc_kwds = {'optimization': opt}\n",
    "crc = ConformalRegretControl(**crc_kwds)\n",
    "\n",
    "path = 'cache/1_28_exp4/result_list_sp.npy'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "bd90f492",
   "metadata": {},
   "outputs": [],
   "source": [
    "# config()\n",
    "\n",
    "# run_kwds = {\n",
    "#     'crc':      crc,\n",
    "#     'mean':     mean,\n",
    "#     'sigma':    sigma,\n",
    "#     'ntrial':   ntrial,\n",
    "#     'nsample_list':  nsample_list,\n",
    "#     'nlam_max':     nlam_max,\n",
    "#     'B':        1.2,\n",
    "#     'width':        None,\n",
    "#     'use_gaussian': False\n",
    "# }\n",
    "\n",
    "# result_list = runner(**run_kwds)\n",
    "# np.save(path, result_list)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "716a2a28",
   "metadata": {},
   "source": [
    "# Visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "f930699b",
   "metadata": {},
   "outputs": [],
   "source": [
    "lam_list = np.arange(0, nlam_max, 10)[1:].tolist() + [nlam_max]\n",
    "\n",
    "# load data\n",
    "path_list = [\n",
    "    'cache/1_28_exp4/result_list_lp.npy',\n",
    "    'cache/1_28_exp4/result_list_nv.npy',\n",
    "    'cache/1_28_exp4/result_list_pf.npy',\n",
    "    'cache/1_28_exp4/result_list_sp.npy',\n",
    "]\n",
    "result_list = []\n",
    "for path in path_list:\n",
    "    arr = np.load(path)             # [ ntrial, len(nsample_list), nlam_max, 5 ] est_R, est_I, time, true_R, true_I\n",
    "    result_list.append(arr)\n",
    "result_list = np.array(result_list) # [ nopt, ntrial, len(nsample_list), nlam_max, 5 ] est_R, est_I, time, true_R, true_I\n",
    "\n",
    "\n",
    "result_list = result_list[None, ...] * np.ones((len(lam_list), *result_list.shape)) # [ len(nlam_list), nopt, ntrial, len(nsample_list), nlam_max, 5 ]\n",
    "for i in range(len(lam_list)):\n",
    "    index = np.random.choice(np.arange(lam_list[-1]), lam_list[-1]-lam_list[i], replace=False)\n",
    "\n",
    "    for j in range(nlam_max):\n",
    "        if j % (lam_list[-1]/lam_list[i]) != 0: # keep this point\n",
    "            # no time used        \n",
    "            result_list[i, ..., j, 2] = 0.\n",
    "            # use value from previous kept point (conservative)\n",
    "            j_ = int(j - j % (lam_list[-1]/lam_list[i]))\n",
    "            result_list[i, ..., j, :2] = result_list[i, ..., j_, :2]\n",
    "\n",
    "# compute time\n",
    "time_arr = result_list[..., 2] # [  len(lam_list), nopt, ntrial, len(nsample_list), nlam_max ]\n",
    "time_arr = time_arr.sum(-1)    # [  len(lam_list), nopt, ntrial, len(nsample_list) ] sum over lam for total\n",
    "\n",
    "# compute area\n",
    "from scipy.spatial.distance import cdist\n",
    "area_arr = 1 / nlam_max * np.sum(\n",
    "    np.sqrt(np.sum((result_list[..., [0, 1]] - result_list[..., [3, 4]])**2, axis=-1)),\n",
    "axis=-1) # [  len(lam_list), nopt, ntrial, len(nsample_list) ]\n",
    "\n",
    "arr = np.concatenate([\n",
    "    area_arr[..., None],   # [  len(lam_list), nopt, ntrial, len(nsample_list), 1 ] estimation error\n",
    "    time_arr[..., None]    # [  len(lam_list), nopt, ntrial, len(nsample_list), 1 ] computation time\n",
    "], axis=-1) # [  len(lam_list), nopt, ntrial, len(nsample_list), 2 ] estimation error, computation time"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "810b6c49",
   "metadata": {},
   "source": [
    "## Box Plot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "58791a7e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "arr_ = arr[:, 0, :, :, 0]\n",
    "arr_ = arr_.mean(1) # [ len(lam_list), len(nsample_list) ]\n",
    "plt.boxplot(arr_ / arr_.max(), boxprops=dict(color='red'))\n",
    "\n",
    "arr_ = arr[:, 0, :, :, 1]\n",
    "arr_ = arr_.mean(1) # [ len(lam_list), len(nsample_list) ]\n",
    "plt.boxplot(arr_ / arr_.max(), boxprops=dict(color='blue'))\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "id": "13700ea4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.16533781718385715\n",
      "4.494994820049032\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiMAAAGdCAYAAADAAnMpAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8fJSN1AAAACXBIWXMAAA9hAAAPYQGoP6dpAAAmzUlEQVR4nO3df1Bc133+8WeFxLLIgCJRL5KFgNhIbEQVmSWxgVEUxwkqTuUo/WFqB8l2QRWlkQcT2yOqUWIpnpK4NsFJDBG1FKLKTZhWiZtOsZXNtJXkUI/HKzRpI4jkWnQJWsygKoAMBgf2+wcV324AmbtmOezyfs3c2dmjc+75rOeMeTj3stcWCAQCAgAAMGSJ6QIAAMDiRhgBAABGEUYAAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYNRS0wXMxvj4uC5fvqyEhATZbDbT5QAAgFkIBAIaHBzUmjVrtGTJzPsfERFGLl++rNTUVNNlAACAEHR1dWnt2rUz/ntEhJGEhARJEx8mMTHRcDUAAGA2BgYGlJqaOvlzfCYREUauX5pJTEwkjAAAEGHe7xYLbmAFAABGEUYAAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYBRhBAAAGEUYAQAARlkOI6dPn9b27du1Zs0a2Ww2vfTSS+875tSpU3K73YqLi9OHP/xhfec73wmlVgAAEIUsh5F33nlHH/3oR/Xtb397Vv0vXbqke+65R1u2bFFbW5v+8i//Uo888ohOnDhhuVgAABB9LD+bpqioSEVFRbPu/53vfEfr1q1TXV2dJMnlcumNN97QM888oz/8wz+0Oj0AAIgyYX9Q3r//+7+rsLAwqG3btm06cuSI3nvvPS1btmzKmJGREY2MjEy+HxgYCHeZAIAo1dfXp5MnT866/+DgoH7+859r06ZN7/u02d+2bds2JScnWy1x0Qt7GOnp6ZHT6Qxqczqd+s1vfqO+vj6tXr16ypiamhodPHgw3KUBABaBkydPqqSkZF7mOn78uL7whS/My1zRJOxhRJr66OBAIDBt+3XV1dWqqqqafD8wMKDU1NTwFQgAiFrbtm3T8ePHZ93/Zz/7mRoaGvTnf/7nKigosDwXrAt7GElJSVFPT09QW29vr5YuXapVq1ZNO8Zut8tut4e7NADAIpCcnGx5t6KhoUEFBQXscsyTsH/PSF5enjweT1DbT37yE+Xm5k57vwgAAFhcLIeRa9eu6dy5czp37pykiT/dPXfunHw+n6SJSyy7du2a7F9eXq7//u//VlVVldrb23X06FEdOXJEjz322Nx8AgAAENEsX6Z54403dNddd02+v35vx4MPPqimpib5/f7JYCJJGRkZamlp0aOPPqrnn39ea9as0Te/+U3+rBcAAEgKIYx88pOfnLwBdTpNTU1T2rZu3aqzZ89anQoAACwCPJsGAAAYRRgBAABGEUYAAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYBRhBAAAGEUYAQAARhFGAACAUYQRAABglOUH5WHhGBoaUkdHh6Uxw8PD6uzsVHp6uhwOx6zHZWVlKT4+3mqJWGRYkwBCQRiJYB0dHXK73fMyl9frVU5OzrzMhcjFmgQQCsJIBMvKypLX67U0pr29XSUlJTp+/LhcLpeluYD3w5oEEArCSASLj48P+TdDl8vFb5WYc6xJAKHgBlYAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYBRhBAAAGEUYAQAARhFGAACAUYQRAABgFGEEAAAYRRgBAABGEUYAAIBRhBEAAGAUYQQAABhFGAEAAEaFFEbq6+uVkZGhuLg4ud1unTlz5ob9n3/+eblcLjkcDm3YsEHHjh0LqVgAABB9llod0NzcrMrKStXX16ugoECHDx9WUVGRzp8/r3Xr1k3p39DQoOrqav3N3/yNPvaxj+n111/X7t279aEPfUjbt2+fkw8BAFhELl6UBgfDd/5Ll/7/69mz4ZsnIUHKzAzf+SOI5TBSW1ur0tJSlZWVSZLq6up08uRJNTQ0qKamZkr/v/3bv9WePXtUXFwsSfrwhz+s1157TV//+tcJIwAAay5elNavn5+5DhyYOMLpwgUCiSyGkdHRUXm9Xu3bty+ovbCwUK2trdOOGRkZUVxcXFCbw+HQ66+/rvfee0/Lli2bdszIyMjk+4GBAStlAgCi1fUdkePHJZcrLFOknzsnlZYq/cgRafPmsMyh9nappCS8OzwRxFIY6evr09jYmJxOZ1C70+lUT0/PtGO2bdumF154QTt27FBOTo68Xq+OHj2q9957T319fVq9evWUMTU1NTp48KCV0gAAi4nLJeXkhOXUjuuvmzeHbQ4EC+kGVpvNFvQ+EAhMabvuwIEDKioq0p133qlly5bpc5/7nB566CFJUkxMzLRjqqur1d/fP3l0dXWFUiYAAIgAlsJIcnKyYmJipuyC9Pb2Ttktuc7hcOjo0aMaGhpSZ2enfD6f0tPTlZCQoOTk5GnH2O12JSYmBh0AACA6WQojsbGxcrvd8ng8Qe0ej0f5+fk3HLts2TKtXbtWMTEx+sEPfqDf//3f15IlfM0JAACLneW/pqmqqtLOnTuVm5urvLw8NTY2yufzqby8XNLEJZbu7u7J7xK5cOGCXn/9dd1xxx26evWqamtr9Z//+Z/63ve+N7efBAAARCTLYaS4uFhXrlzRoUOH5Pf7lZ2drZaWFqWlpUmS/H6/fD7fZP+xsTE9++yz+uUvf6lly5bprrvuUmtrq9LT0+fsQwAAgMhlOYxIUkVFhSoqKqb9t6ampqD3LpdLbW1toUwDAMAUF3WbBtsd798xRO3twa/hmcShBN0mvmFkQkhhBAAAEy767Fqvi1JJ+OcqCescLkkXdcH3C2Xy18OEEQBA5Bh8Z+IPH45/9ZJc92SEZY7h4Sx1dnqVnp4lR5g2YNpbLqnkQMbk51nsCCMAgIjjyng3jN9HFq+CgjBvV7S/G97zRxgiGQAAMIqdkYUm3E+jnI87s3gSZXSJhjUpsS6BBYwwspDM59Mow3tnFk+ijBbRtCYl1iWwQBFGFpJ5eBpl1vCwvJ2dykpPV1juzOJJlNElGtakxLoEFjjCyEIUxqdRxkvKKSgIy7kRxViTAMKIG1gBAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYBRhBAAAGEUYAQAARhFGAACAUYQRAABgFGEEAAAYRRgBAABGEUYAAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYBRhBAAAGEUYAQAARhFGAACAUUtNFwAAgBUpN/XI8W6HdHnEdCkhc7x7SSk3fch0GQtGSGGkvr5ef/3Xfy2/36+NGzeqrq5OW7ZsmbH/iy++qKeffloXL15UUlKSfu/3fk/PPPOMVq1aFXLhAIDFaY/7u3J1fU1qNF1J6FyS9rj3SdphuJKFwXIYaW5uVmVlperr61VQUKDDhw+rqKhI58+f17p166b0f/XVV7Vr1y594xvf0Pbt29Xd3a3y8nKVlZXpRz/60Zx8CADA4nHY+7CKd2bJVZRhupSQtb98SYe9ebp391XTpSwIlsNIbW2tSktLVVZWJkmqq6vTyZMn1dDQoJqamin9X3vtNaWnp+uRRx6RJGVkZGjPnj16+umnP2DpAIDFqOdaiobjsqQ1LtOlhGw4zq6eaymSCCOSxRtYR0dH5fV6VVhYGNReWFio1tbWacfk5+frV7/6lVpaWhQIBPT222/rH/7hH/TZz352xnlGRkY0MDAQdAAAgOhkKYz09fVpbGxMTqczqN3pdKqnp2faMfn5+XrxxRdVXFys2NhYpaSkaMWKFfrWt7414zw1NTVKSkqaPFJTU62UCQAAIkhIf9prs9mC3gcCgSlt150/f16PPPKIvvzlL8vr9eqVV17RpUuXVF5ePuP5q6ur1d/fP3l0dXWFUiYAAIgAlu4ZSU5OVkxMzJRdkN7e3im7JdfV1NSooKBAjz/+uCRp06ZNWr58ubZs2aKnnnpKq1evnjLGbrfLbrdbKQ0AAEQoSzsjsbGxcrvd8ng8Qe0ej0f5+fnTjhkaGtKSJcHTxMTESJrYUQEAAIub5b+mqaqq0s6dO5Wbm6u8vDw1NjbK5/NNXnaprq5Wd3e3jh07Jknavn27du/erYaGBm3btk1+v1+VlZX6+Mc/rjVr1sztp4kGN9mkX1+QLkfol+P++sLEZ0D0iPQ1KbEugQXOchgpLi7WlStXdOjQIfn9fmVnZ6ulpUVpaWmSJL/fL5/PN9n/oYce0uDgoL797W/rS1/6klasWKFPfepT+vrXvz53nyKauGOl03uk06YL+QDcsaYrwFyKhjUpsS6BBSykb2CtqKhQRUXFtP/W1NQ0pW3v3r3au3dvKFMtPt5R6ctNUlaW6UpC09EhPfuA6SowlyJ9TUqsS2CB49k0C821gLRivbRms+lKQtMzPvEZED0ifU1KrEtggYvgi8AAACAaEEYAAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFH8aS8AIKKk3NQjx7sd0uUR06WEzPHuJaXc9CHTZSwYhBEAQETZ4/6uXF1fkxpNVxI6l6Q97n2SdhiuZGEgjAAAIsph78Mq3pklV1GG6VJC1v7yJR325une3VdNl7IgEEYAABGl51qKhuOypDUu06WEbDjOrp5rKZIIIxI3sAIAAMMIIwAAwCjCCAAAMIowAgAAjCKMAAAAowgjAADAKMIIAAAwijACAACMIowAAACjCCMAAMAowggAADCKMAIAAIwijAAAAKMIIwAAwCjCCAAAMIowAgAAjCKMAAAAowgjAADAKMIIAAAwKqQwUl9fr4yMDMXFxcntduvMmTMz9n3ooYdks9mmHBs3bgy5aAAAED0sh5Hm5mZVVlZq//79amtr05YtW1RUVCSfzzdt/+eee05+v3/y6Orq0sqVK/XHf/zHH7h4AAAQ+SyHkdraWpWWlqqsrEwul0t1dXVKTU1VQ0PDtP2TkpKUkpIyebzxxhu6evWqHn744Q9cPAAAiHyWwsjo6Ki8Xq8KCwuD2gsLC9Xa2jqrcxw5ckSf/vSnlZaWNmOfkZERDQwMBB0AACA6WQojfX19Ghsbk9PpDGp3Op3q6el53/F+v18vv/yyysrKbtivpqZGSUlJk0dqaqqVMgEAQAQJ6QZWm80W9D4QCExpm05TU5NWrFihHTt23LBfdXW1+vv7J4+urq5QygQAABFgqZXOycnJiomJmbIL0tvbO2W35LcFAgEdPXpUO3fuVGxs7A372u122e12K6UBAIAIZWlnJDY2Vm63Wx6PJ6jd4/EoPz//hmNPnTqlN998U6WlpdarBAAAUcvSzogkVVVVaefOncrNzVVeXp4aGxvl8/lUXl4uaeISS3d3t44dOxY07siRI7rjjjuUnZ09N5UDAICoYDmMFBcX68qVKzp06JD8fr+ys7PV0tIy+dcxfr9/yneO9Pf368SJE3ruuefmpmoAABA1LIcRSaqoqFBFRcW0/9bU1DSlLSkpSUNDQ6FMBQAAohzPpgEAAEYRRgAAgFGEEQAAYBRhBAAAGEUYAQAARhFGAACAUYQRAABgFGEEAAAYRRgBAABGhfQNrAAAmDD07sTv0Gc74qWz4ZljeHhInZ0dSk/PksMRH5Y52i/FheW8kYowAgCIGB2dEz/Edz+VJj0VtlkkuSV5JeWEaY4MSVLC8vEwnT+yEEYWkuvP7zkbprgvaWh4WB2dncpKT1e8wzH3E7S3z/05YU40rEmJdRlFdnzy19JTX1XWC48r/vYNYZmjvV0qKZGOH5dcrrBMIbW3K6HkXmWuaw7TBJGFMLKQdHRMvO7eHb4pFP68L0lKSAjn2TFfomlNSqzLKJD8oTGV6Yh0e0XYF4zLJeWEbY5hSW+G6+QRhzCykOzYMfGalSXFh+c65bxE/oQEKTMzPOfG/IqWNSmxLoEFjDCykCQnS2Vl8zNXeCM/ogVrEgvNPFw6HD53Lug1LLh0GIQwAgCIHPNw6bDz+mtpqQrCNsv/4tKhJMIIACCSzMelw5YW6cAB6atfle65JzxzSFw6/D8IIwCAyDEflw6vX0LJyODS4TzhG1gBAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYBRhBAAAGEUYAQAARhFGAACAUYQRAABgFGEEAAAYRRgBAABGEUYAAIBRIYWR+vp6ZWRkKC4uTm63W2fOnLlh/5GREe3fv19paWmy2+269dZbdfTo0ZAKBgAA0WWp1QHNzc2qrKxUfX29CgoKdPjwYRUVFen8+fNat27dtGPuu+8+vf322zpy5Ihuu+029fb26je/+c0HLh4AAEQ+y2GktrZWpaWlKisrkyTV1dXp5MmTamhoUE1NzZT+r7zyik6dOqW33npLK1eulCSlp6d/sKoBAEDUsHSZZnR0VF6vV4WFhUHthYWFam1tnXbMj3/8Y+Xm5urpp5/WLbfcovXr1+uxxx7T8PDwjPOMjIxoYGAg6AAAANHJ0s5IX1+fxsbG5HQ6g9qdTqd6enqmHfPWW2/p1VdfVVxcnH70ox+pr69PFRUV+p//+Z8Z7xupqanRwYMHrZQGAAAiVEg3sNpstqD3gUBgStt14+PjstlsevHFF/Xxj39c99xzj2pra9XU1DTj7kh1dbX6+/snj66urlDKBAAAEcDSzkhycrJiYmKm7IL09vZO2S25bvXq1brllluUlJQ02eZyuRQIBPSrX/1KmZmZU8bY7XbZ7XYrpQEAgAhlaWckNjZWbrdbHo8nqN3j8Sg/P3/aMQUFBbp8+bKuXbs22XbhwgUtWbJEa9euDaFkAAAQTSxfpqmqqtILL7ygo0ePqr29XY8++qh8Pp/Ky8slTVxi2bVr12T/Bx54QKtWrdLDDz+s8+fP6/Tp03r88cf1p3/6p3I4HHP3SQAAQESy/Ke9xcXFunLlig4dOiS/36/s7Gy1tLQoLS1NkuT3++Xz+Sb733TTTfJ4PNq7d69yc3O1atUq3XfffXrqqafm7lMAAICIZTmMSFJFRYUqKiqm/bempqYpbVlZWVMu7QAAAEg8mwYAABhGGAEAAEYRRgAAgFGEEQAAYBRhBAAAGEUYAQAARhFGAACAUYQRAABgFGEEAAAYRRgBAABGhfR18AAARIqhoSF1dHTMuv+lS5cmX8+ePWtprqysLMXHx1saA8IIACDKdXR0yO12Wx534MABHThwwNIYr9ernJwcy3MtdoQRAEBUy8rKktfrnXX/c+fOqbS0VEeOHNHmzZstzwXrCCMAgKgWHx8f0m7F5s2b2eWYJ9zACgAAjCKMAAAAowgjAADAKMIIAAAwijACAACMIowAAACjCCMAAMAowggAADCKMAIAAIwijAAAAKP4OvgIZvVJlJLU3t4e9DpbPIkSs8GaBBAKwkgEC/VJlJJUUlJiqT9PosRssCYBhIIwEsGsPolSkoaHh9XZ2an09HQ5HA5LcwHvhzUJIBSEkQgW6pMoCwoKwlANwJoEEBpuYAUAAEYRRgAAgFEhhZH6+nplZGQoLi5ObrdbZ86cmbHvv/3bv8lms005rN5xDwAAopPlMNLc3KzKykrt379fbW1t2rJli4qKiuTz+W447pe//KX8fv/kkZmZGXLRAAAgelgOI7W1tSotLVVZWZlcLpfq6uqUmpqqhoaGG467+eablZKSMnnExMSEXDQAAIgelsLI6OiovF6vCgsLg9oLCwvV2tp6w7G33367Vq9erbvvvlv/+q//esO+IyMjGhgYCDoAAEB0shRG+vr6NDY2JqfTGdTudDrV09Mz7ZjVq1ersbFRJ06c0A9/+ENt2LBBd999t06fPj3jPDU1NUpKSpo8UlNTrZQJAAAiSEjfM2Kz2YLeBwKBKW3XbdiwQRs2bJh8n5eXp66uLj3zzDP6xCc+Me2Y6upqVVVVTb4fGBggkAAAEKUs7YwkJycrJiZmyi5Ib2/vlN2SG7nzzjt18eLFGf/dbrcrMTEx6AAAANHJUhiJjY2V2+2Wx+MJavd4PMrPz5/1edra2rR69WorUwMAgChl+TJNVVWVdu7cqdzcXOXl5amxsVE+n0/l5eWSJi6xdHd369ixY5Kkuro6paena+PGjRodHdXx48d14sQJnThxYm4/CQAA07D6NOlQnyQt8TTpUFkOI8XFxbpy5YoOHTokv9+v7OxstbS0KC0tTZLk9/uDvnNkdHRUjz32mLq7u+VwOLRx40b98z//s+655565+xQAAMwg1KdJW32StMTTpENlCwQCAdNFvJ+BgQElJSWpv7+f+0cAAJZY3RkJ9UnSEjsjv222P78JIwAAICxm+/ObB+UBAACjCCMAAMAowggAADCKMAIAAIwijAAAAKMIIwAAwCjCCAAAMIowAgAAjCKMAAAAowgjAADAKMIIAAAwijACAACMIowAAACjCCMAAMAowggAADCKMAIAAIwijAAAAKMIIwAAwCjCCAAAMIowAgAAjCKMAAAAowgjAADAKMIIAAAwijACAACMIowAAACjCCMAAMAowggAADCKMAIAAIwijAAAAKNCCiP19fXKyMhQXFyc3G63zpw5M6txP/vZz7R06VJt3rw5lGkBAEAUshxGmpubVVlZqf3796utrU1btmxRUVGRfD7fDcf19/dr165duvvuu0MuFgAARB9bIBAIWBlwxx13KCcnRw0NDZNtLpdLO3bsUE1NzYzj/uRP/kSZmZmKiYnRSy+9pHPnzs16zoGBASUlJam/v1+JiYlWygUAAIbM9ue3pZ2R0dFReb1eFRYWBrUXFhaqtbV1xnHf/e539V//9V/6yle+Mqt5RkZGNDAwEHQAAIDoZCmM9PX1aWxsTE6nM6jd6XSqp6dn2jEXL17Uvn379OKLL2rp0qWzmqempkZJSUmTR2pqqpUyAQBABAnpBlabzRb0PhAITGmTpLGxMT3wwAM6ePCg1q9fP+vzV1dXq7+/f/Lo6uoKpUwAABABZrdV8b+Sk5MVExMzZRekt7d3ym6JJA0ODuqNN95QW1ubvvjFL0qSxsfHFQgEtHTpUv3kJz/Rpz71qSnj7Ha77Ha7ldIAAECEsrQzEhsbK7fbLY/HE9Tu8XiUn58/pX9iYqL+4z/+Q+fOnZs8ysvLtWHDBp07d0533HHHB6seAABEPEs7I5JUVVWlnTt3Kjc3V3l5eWpsbJTP51N5ebmkiUss3d3dOnbsmJYsWaLs7Oyg8TfffLPi4uKmtAMAgMXJchgpLi7WlStXdOjQIfn9fmVnZ6ulpUVpaWmSJL/f/77fOQIAAHCd5e8ZMYHvGQEAIPKE5XtGAAAA5hphBAAAGEUYAQAARhFGAACAUYQRAABgFGEEAAAYRRgBAABGEUYAAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYBRhBAAAGEUYAQAARhFGAACAUYQRAABgFGEEAAAYRRgBAABGEUYAAIBRS00XgNANDQ2po6PD0pjh4WF1dnYqPT1dDodj1uOysrIUHx9vtUQsMqxJAKEgjESwjo4Oud3ueZnL6/UqJydnXuZC5GJNAggFYSSCZWVlyev1WhrT3t6ukpISHT9+XC6Xy9JcwPthTQIIBWEkgsXHx4f8m6HL5eK3Ssw51iSAUHADKwAAMIowAgAAjCKMAAAAowgjAADAKMIIAAAwijACAACMIowAAACjQgoj9fX1ysjIUFxcnNxut86cOTNj31dffVUFBQVatWqVHA6HsrKy9I1vfCPkggEAQHSx/KVnzc3NqqysVH19vQoKCnT48GEVFRXp/PnzWrdu3ZT+y5cv1xe/+EVt2rRJy5cv16uvvqo9e/Zo+fLl+rM/+7M5+RAAACByWd4Zqa2tVWlpqcrKyuRyuVRXV6fU1FQ1NDRM2//222/X/fffr40bNyo9PV0lJSXatm3bDXdTAADA4mEpjIyOjsrr9aqwsDCovbCwUK2trbM6R1tbm1pbW7V169YZ+4yMjGhgYCDoAAAA0clSGOnr69PY2JicTmdQu9PpVE9Pzw3Hrl27Vna7Xbm5ufqLv/gLlZWVzdi3pqZGSUlJk0dqaqqVMgEAQAQJ6UF5Npst6H0gEJjS9tvOnDmja9eu6bXXXtO+fft022236f7775+2b3V1taqqqibfDwwMLJpAcvGiNDgYvvO3twe/hkNCgpSZGb7zY35Fw5qUWJfAQmYpjCQnJysmJmbKLkhvb++U3ZLflpGRIUn63d/9Xb399tt68sknZwwjdrtddrvdSmlR4eJFaf36+ZmrpCS8579wgf/xR4NoWpMS6xJYqCyFkdjYWLndbnk8Hn3+85+fbPd4PPrc5z436/MEAgGNjIxYmXpRuP7b5/HjkssVnjmGh7PU2elVenqWHI65P397+8QPlXD+Jo35Ew1rUmJdAgud5cs0VVVV2rlzp3Jzc5WXl6fGxkb5fD6Vl5dLmrjE0t3drWPHjkmSnn/+ea1bt05ZWVmSJr535JlnntHevXvn8GNEF5dLyskJ19njVVAQtpMjSrEmAYST5TBSXFysK1eu6NChQ/L7/crOzlZLS4vS0tIkSX6/Xz6fb7L/+Pi4qqurdenSJS1dulS33nqrvva1r2nPnj1z9ykAAEDECukG1oqKClVUVEz7b01NTUHv9+7dyy4IAACYEc+mAQAARhFGAACAUYQRAABgFGEEAAAYRRgBAABGEUYAAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYFRIz6ZB+KTc1CPHr3uky6YrCY3j11LKTSmSUkyXgjkS6WtSYl0CCx1hZIHZ4/6uXKe/Jp02XUloXJL2uPdJqjZdCuZIpK9JiXUJLHSEkQXmsPdhFX+5SK4s05WEpr1DOvxsiu41XQjmTKSvSYl1CSx0hJEFpudaioZXpEhrTFcSmuEeqeea6SowlyJ9TUqsS2Ch4wZWAABgFGEEAAAYRRgBAABGEUYAAIBRhBEAAGAUYQQAABhFGAEAAEYRRgAAgFGEEQAAYBTfwLqADA1NvJ49G745hoeH1NnZofT0LDkc8XN+/vb2OT8lDIqGNSmxLoGFjjCygHR0TLzu3h3WWSS5JXkl5YRtloSEsJ0a8yia1qTEugQWKsLIArJjx8RrVpYUH55fENXeLpWUSMePSy5XeOZISJAyM8NzbsyvaFmTEusSWMgIIwtIcrJUVjY/c7lcUk54fwlFFGBNApgP3MAKAACMIowAAACjQgoj9fX1ysjIUFxcnNxut86cOTNj3x/+8If6zGc+o9/5nd9RYmKi8vLydPLkyZALBgAA0cVyGGlublZlZaX279+vtrY2bdmyRUVFRfL5fNP2P336tD7zmc+opaVFXq9Xd911l7Zv3662trYPXDwAAIh8lsNIbW2tSktLVVZWJpfLpbq6OqWmpqqhoWHa/nV1dXriiSf0sY99TJmZmfqrv/orZWZm6p/+6Z8+cPEAACDyWQojo6Oj8nq9KiwsDGovLCxUa2vrrM4xPj6uwcFBrVy5csY+IyMjGhgYCDoAAEB0shRG+vr6NDY2JqfTGdTudDrV09Mzq3M8++yzeuedd3TffffN2KempkZJSUmTR2pqqpUyAQBABAnpBlabzRb0PhAITGmbzve//309+eSTam5u1s033zxjv+rqavX3908eXV1doZQJAAAigKUvPUtOTlZMTMyUXZDe3t4puyW/rbm5WaWlpfr7v/97ffrTn75hX7vdLrvdbqU0AAAQoSztjMTGxsrtdsvj8QS1ezwe5efnzzju+9//vh566CH93d/9nT772c+GVikAAIhKlr8OvqqqSjt37lRubq7y8vLU2Ngon8+n8vJySROXWLq7u3Xs2DFJE0Fk165deu6553TnnXdO7qo4HA4lJSXN4UdZfIaGhtRx/Ulms9T+v48vbbf4GNOsrCzFh+vhJIgarEkAIQmE4Pnnnw+kpaUFYmNjAzk5OYFTp05N/tuDDz4Y2Lp16+T7rVu3BiRNOR588MFZz9ff3x+QFOjv7w+l3Kjl9Xqn/W8bjsPr9Zr+uIgArEkA/9dsf37bAoFAYI5yTdgMDAwoKSlJ/f39SkxMNF3OghHKb6HDw8Pq7OxUenq6HA7HrMfxWyhmgzUJ4P+a7c9vwggAAAiL2f785kF5AADAKMIIAAAwijACAACMIowAAACjCCMAAMAowggAADCKMAIAAIwijAAAAKMIIwAAwCjCCAAAMIowAgAAjCKMAAAAowgjAADAqKWmC5iN6w8WHhgYMFwJAACYres/t6//HJ9JRISRwcFBSVJqaqrhSgAAgFWDg4NKSkqa8d9tgfeLKwvA+Pi4Ll++rISEBNlsNtPlRLTu7m595CMf0fnz53XLLbeYLgdgTWLBYU3OnUAgoMHBQa1Zs0ZLlsx8Z0hE7IwsWbJEa9euNV1GVLi+ZZaQkKDExETD1QCsSSw8rMm5daMdkeu4gRUAABhFGAEAAEYRRhaZxMREbd26la1HLBisSSw0rMn5FxE3sAIAgOjFzggAADCKMAIAAIwijAAAAKMIIwAAwCjCyCLxrW99S06nUzExMbLZbKqurjZdEhaxbdu2afny5bLZbFqyZIlWr16tl19+2XRZWOTuv/9+ORwO2Ww22Ww2JSQk6NChQ6bLWhQII4vE1atXlZmZqS996UumSwHk9Xp1//3366WXXtIPfvADjY+Pa/v27ert7TVdGhax2267Tfv27dMrr7yiV155RZs2bdJXvvIV/eM//qPp0qIef9q7CNlsNu3bt081NTWmSwEkSe3t7frIRz6ib37zm9q7d6/pcoBJS5Ys0a5du9TU1GS6lKjGzggA4y5fvixJPIMKC8bo6KgeeeQRBQIB/cEf/IHpcqJeRDwoD0D0Gh8f1xe+8AUlJibq85//vOlysMidOHFCf/RHfzT5/sknn9S9995rsKLFgZ0RAEZ99KMfVV9fn37605+aLgXQtm3b9NOf/lTf+973dOedd+rgwYP68Y9/bLqsqMc9I4sQ94xgodi0aZPOnz+vf/mXf9EnPvEJ0+UAU6xcuVJOp1Pt7e2mS4lq7IwAmHfj4+PatGmTfvGLX+jll18miGBBe++990yXEPW4Z2SR6Onp0alTpybft7e3q7m5WevWrVNeXp7ByrAYXQ8idXV1cjqd+vnPfy5p4gbWlStXGq4Oi1V+fr4eeOABbd68WW+//bZqa2t19epVPfHEE6ZLi3pcplkk6urq9Oijj05pv/XWW/Xmm28aqAiLmc1mm7a9tLRUL7zwwjxXA0xYv3693nrrLY2Njclms2nFihV64okntG/fPtOlRT3CCAAAMIp7RgAAgFGEEQAAYBRhBAAAGEUYAQAARhFGAACAUYQRAABgFGEEAAAYRRgBAABGEUYAAIBRhBEAAGAUYQQAABhFGAEAAEb9P8t19R+cnUcyAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "arr_ = arr[:, 2, :, :, 0]\n",
    "arr_ = arr_.mean(1) # [ len(lam_list), len(nsample_list) ]\n",
    "arr_ = arr_.T\n",
    "plt.boxplot(arr_ / arr_.max(), boxprops=dict(color='red'))\n",
    "print(arr_.max())\n",
    "\n",
    "arr_ = arr[:, 2, :, :, 1]\n",
    "arr_ = arr_.mean(1) # [ len(lam_list), len(nsample_list) ]\n",
    "arr_ = arr_.T\n",
    "plt.boxplot(arr_ / arr_.max(), boxprops=dict(color='blue'))\n",
    "print(arr_.max())\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a7075ed3",
   "metadata": {},
   "source": [
    "## Table"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "id": "11c2e163",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead tr th {\n",
       "        text-align: left;\n",
       "    }\n",
       "\n",
       "    .dataframe thead tr:last-of-type th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th colspan=\"2\" halign=\"left\">Linear Programming</th>\n",
       "      <th colspan=\"2\" halign=\"left\">Newsvendor</th>\n",
       "      <th colspan=\"2\" halign=\"left\">Portfolio Optimization</th>\n",
       "      <th colspan=\"2\" halign=\"left\">Shortest Path</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th>Gap</th>\n",
       "      <th>Time (s)</th>\n",
       "      <th>Gap</th>\n",
       "      <th>Time (s)</th>\n",
       "      <th>Gap</th>\n",
       "      <th>Time (s)</th>\n",
       "      <th>Gap</th>\n",
       "      <th>Time (s)</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th rowspan=\"3\" valign=\"top\">$|\\Lambda|=10$</th>\n",
       "      <th>$n=10$</th>\n",
       "      <td>$ 0.10 \\pm  0.03$</td>\n",
       "      <td>$ 0.58 \\pm  0.03$</td>\n",
       "      <td>$ 0.19 \\pm  0.08$</td>\n",
       "      <td>$ 0.00 \\pm  0.00$</td>\n",
       "      <td>$ 0.16 \\pm  0.07$</td>\n",
       "      <td>$ 0.84 \\pm  0.10$</td>\n",
       "      <td>$ 0.11 \\pm  0.03$</td>\n",
       "      <td>$ 0.72 \\pm  0.04$</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>$n=20$</th>\n",
       "      <td>$ 0.09 \\pm  0.03$</td>\n",
       "      <td>$ 0.81 \\pm  0.03$</td>\n",
       "      <td>$ 0.12 \\pm  0.04$</td>\n",
       "      <td>$ 0.00 \\pm  0.00$</td>\n",
       "      <td>$ 0.13 \\pm  0.07$</td>\n",
       "      <td>$ 1.14 \\pm  0.12$</td>\n",
       "      <td>$ 0.06 \\pm  0.01$</td>\n",
       "      <td>$ 0.99 \\pm  0.06$</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>$n=30$</th>\n",
       "      <td>$ 0.08 \\pm  0.01$</td>\n",
       "      <td>$ 1.00 \\pm  0.04$</td>\n",
       "      <td>$ 0.13 \\pm  0.04$</td>\n",
       "      <td>$ 0.00 \\pm  0.00$</td>\n",
       "      <td>$ 0.09 \\pm  0.02$</td>\n",
       "      <td>$ 1.50 \\pm  0.13$</td>\n",
       "      <td>$ 0.06 \\pm  0.01$</td>\n",
       "      <td>$ 1.20 \\pm  0.07$</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"3\" valign=\"top\">$|\\Lambda|=20$</th>\n",
       "      <th>$n=10$</th>\n",
       "      <td>$ 0.10 \\pm  0.03$</td>\n",
       "      <td>$ 0.58 \\pm  0.03$</td>\n",
       "      <td>$ 0.19 \\pm  0.08$</td>\n",
       "      <td>$ 0.00 \\pm  0.00$</td>\n",
       "      <td>$ 0.16 \\pm  0.07$</td>\n",
       "      <td>$ 0.84 \\pm  0.10$</td>\n",
       "      <td>$ 0.11 \\pm  0.03$</td>\n",
       "      <td>$ 0.72 \\pm  0.04$</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>$n=20$</th>\n",
       "      <td>$ 0.09 \\pm  0.03$</td>\n",
       "      <td>$ 0.81 \\pm  0.03$</td>\n",
       "      <td>$ 0.12 \\pm  0.04$</td>\n",
       "      <td>$ 0.00 \\pm  0.00$</td>\n",
       "      <td>$ 0.13 \\pm  0.07$</td>\n",
       "      <td>$ 1.14 \\pm  0.12$</td>\n",
       "      <td>$ 0.06 \\pm  0.01$</td>\n",
       "      <td>$ 0.99 \\pm  0.06$</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>$n=30$</th>\n",
       "      <td>$ 0.08 \\pm  0.01$</td>\n",
       "      <td>$ 1.00 \\pm  0.04$</td>\n",
       "      <td>$ 0.13 \\pm  0.04$</td>\n",
       "      <td>$ 0.00 \\pm  0.00$</td>\n",
       "      <td>$ 0.09 \\pm  0.02$</td>\n",
       "      <td>$ 1.50 \\pm  0.13$</td>\n",
       "      <td>$ 0.06 \\pm  0.01$</td>\n",
       "      <td>$ 1.20 \\pm  0.07$</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"3\" valign=\"top\">$|\\Lambda|=30$</th>\n",
       "      <th>$n=10$</th>\n",
       "      <td>$ 0.09 \\pm  0.03$</td>\n",
       "      <td>$ 1.73 \\pm  0.06$</td>\n",
       "      <td>$ 0.16 \\pm  0.07$</td>\n",
       "      <td>$ 0.00 \\pm  0.00$</td>\n",
       "      <td>$ 0.17 \\pm  0.08$</td>\n",
       "      <td>$ 2.50 \\pm  0.28$</td>\n",
       "      <td>$ 0.11 \\pm  0.04$</td>\n",
       "      <td>$ 2.13 \\pm  0.08$</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>$n=20$</th>\n",
       "      <td>$ 0.08 \\pm  0.02$</td>\n",
       "      <td>$ 2.42 \\pm  0.09$</td>\n",
       "      <td>$ 0.11 \\pm  0.04$</td>\n",
       "      <td>$ 0.00 \\pm  0.00$</td>\n",
       "      <td>$ 0.13 \\pm  0.07$</td>\n",
       "      <td>$ 3.43 \\pm  0.34$</td>\n",
       "      <td>$ 0.05 \\pm  0.01$</td>\n",
       "      <td>$ 2.93 \\pm  0.15$</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>$n=30$</th>\n",
       "      <td>$ 0.06 \\pm  0.02$</td>\n",
       "      <td>$ 2.98 \\pm  0.13$</td>\n",
       "      <td>$ 0.11 \\pm  0.04$</td>\n",
       "      <td>$ 0.00 \\pm  0.00$</td>\n",
       "      <td>$ 0.08 \\pm  0.03$</td>\n",
       "      <td>$ 4.49 \\pm  0.44$</td>\n",
       "      <td>$ 0.04 \\pm  0.01$</td>\n",
       "      <td>$ 3.63 \\pm  0.25$</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                      Linear Programming                     \\\n",
       "                                     Gap           Time (s)   \n",
       "                                                              \n",
       "$|\\Lambda|=10$ $n=10$  $ 0.10 \\pm  0.03$  $ 0.58 \\pm  0.03$   \n",
       "               $n=20$  $ 0.09 \\pm  0.03$  $ 0.81 \\pm  0.03$   \n",
       "               $n=30$  $ 0.08 \\pm  0.01$  $ 1.00 \\pm  0.04$   \n",
       "$|\\Lambda|=20$ $n=10$  $ 0.10 \\pm  0.03$  $ 0.58 \\pm  0.03$   \n",
       "               $n=20$  $ 0.09 \\pm  0.03$  $ 0.81 \\pm  0.03$   \n",
       "               $n=30$  $ 0.08 \\pm  0.01$  $ 1.00 \\pm  0.04$   \n",
       "$|\\Lambda|=30$ $n=10$  $ 0.09 \\pm  0.03$  $ 1.73 \\pm  0.06$   \n",
       "               $n=20$  $ 0.08 \\pm  0.02$  $ 2.42 \\pm  0.09$   \n",
       "               $n=30$  $ 0.06 \\pm  0.02$  $ 2.98 \\pm  0.13$   \n",
       "\n",
       "                              Newsvendor                     \\\n",
       "                                     Gap           Time (s)   \n",
       "                                                              \n",
       "$|\\Lambda|=10$ $n=10$  $ 0.19 \\pm  0.08$  $ 0.00 \\pm  0.00$   \n",
       "               $n=20$  $ 0.12 \\pm  0.04$  $ 0.00 \\pm  0.00$   \n",
       "               $n=30$  $ 0.13 \\pm  0.04$  $ 0.00 \\pm  0.00$   \n",
       "$|\\Lambda|=20$ $n=10$  $ 0.19 \\pm  0.08$  $ 0.00 \\pm  0.00$   \n",
       "               $n=20$  $ 0.12 \\pm  0.04$  $ 0.00 \\pm  0.00$   \n",
       "               $n=30$  $ 0.13 \\pm  0.04$  $ 0.00 \\pm  0.00$   \n",
       "$|\\Lambda|=30$ $n=10$  $ 0.16 \\pm  0.07$  $ 0.00 \\pm  0.00$   \n",
       "               $n=20$  $ 0.11 \\pm  0.04$  $ 0.00 \\pm  0.00$   \n",
       "               $n=30$  $ 0.11 \\pm  0.04$  $ 0.00 \\pm  0.00$   \n",
       "\n",
       "                      Portfolio Optimization                     \\\n",
       "                                         Gap           Time (s)   \n",
       "                                                                  \n",
       "$|\\Lambda|=10$ $n=10$      $ 0.16 \\pm  0.07$  $ 0.84 \\pm  0.10$   \n",
       "               $n=20$      $ 0.13 \\pm  0.07$  $ 1.14 \\pm  0.12$   \n",
       "               $n=30$      $ 0.09 \\pm  0.02$  $ 1.50 \\pm  0.13$   \n",
       "$|\\Lambda|=20$ $n=10$      $ 0.16 \\pm  0.07$  $ 0.84 \\pm  0.10$   \n",
       "               $n=20$      $ 0.13 \\pm  0.07$  $ 1.14 \\pm  0.12$   \n",
       "               $n=30$      $ 0.09 \\pm  0.02$  $ 1.50 \\pm  0.13$   \n",
       "$|\\Lambda|=30$ $n=10$      $ 0.17 \\pm  0.08$  $ 2.50 \\pm  0.28$   \n",
       "               $n=20$      $ 0.13 \\pm  0.07$  $ 3.43 \\pm  0.34$   \n",
       "               $n=30$      $ 0.08 \\pm  0.03$  $ 4.49 \\pm  0.44$   \n",
       "\n",
       "                           Shortest Path                     \n",
       "                                     Gap           Time (s)  \n",
       "                                                             \n",
       "$|\\Lambda|=10$ $n=10$  $ 0.11 \\pm  0.03$  $ 0.72 \\pm  0.04$  \n",
       "               $n=20$  $ 0.06 \\pm  0.01$  $ 0.99 \\pm  0.06$  \n",
       "               $n=30$  $ 0.06 \\pm  0.01$  $ 1.20 \\pm  0.07$  \n",
       "$|\\Lambda|=20$ $n=10$  $ 0.11 \\pm  0.03$  $ 0.72 \\pm  0.04$  \n",
       "               $n=20$  $ 0.06 \\pm  0.01$  $ 0.99 \\pm  0.06$  \n",
       "               $n=30$  $ 0.06 \\pm  0.01$  $ 1.20 \\pm  0.07$  \n",
       "$|\\Lambda|=30$ $n=10$  $ 0.11 \\pm  0.04$  $ 2.13 \\pm  0.08$  \n",
       "               $n=20$  $ 0.05 \\pm  0.01$  $ 2.93 \\pm  0.15$  \n",
       "               $n=30$  $ 0.04 \\pm  0.01$  $ 3.63 \\pm  0.25$  "
      ]
     },
     "execution_count": 119,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# arr [ len(lam_list), nopt, ntrial, len(nsample_list), 2 ] estimation error, computation time\n",
    "\n",
    "arr_ = arr.mean(2)                  # [ len(lam_list), nopt, len(nsample_list), 2 ] mean over trials\n",
    "arr_ = arr_.transpose(0, 2, 1, 3)   # [ len(lam_list), len(nsample_list), nopt, 2 ]\n",
    "arr_ = arr_.round(2)\n",
    "\n",
    "std = arr.std(2)                  # [ len(lam_list), nopt, len(nsample_list), 2 ] std over trials\n",
    "std = std.transpose(0, 2, 1, 3)   # [ len(lam_list), len(nsample_list), nopt, 2 ]\n",
    "std = std.round(2)\n",
    "\n",
    "opt_names = [\n",
    "    'Linear Programming',\n",
    "    'Newsvendor',\n",
    "    'Portfolio Optimization',\n",
    "    'Shortest Path'\n",
    "]\n",
    "nsample_names = [ r'$n={}$'.format(n) for n in nsample_list ]\n",
    "lam_names = [ r'$|\\Lambda|={}$'.format(lam) for lam in lam_list ] \n",
    "\n",
    "\n",
    "import pandas as pd\n",
    "# # Create row MultiIndex: lam_list, nopt, ntrial\n",
    "# row_index = pd.MultiIndex.from_product([\n",
    "#     lam_names,\n",
    "#     opt_names,\n",
    "# ], names=['Lambda', 'Optimization'])\n",
    "\n",
    "# # Create column MultiIndex: nsample_list, metrics\n",
    "# col_index = pd.MultiIndex.from_product([\n",
    "#     nsample_names,\n",
    "#     ['Approximation Gap', 'Computation Time']\n",
    "# ], names=['Sample_Size', 'Metric'])\n",
    "\n",
    "# Create row MultiIndex: lam_list, nopt, ntrial\n",
    "row_index = pd.MultiIndex.from_product([\n",
    "    lam_names,\n",
    "    nsample_names,\n",
    "], names=['', ''])\n",
    "\n",
    "# Create column MultiIndex: nsample_list, metrics\n",
    "col_index = pd.MultiIndex.from_product([\n",
    "    opt_names,\n",
    "    ['Gap', 'Time (s)']\n",
    "# ], names=['Sample Size', 'Metric'])\n",
    "], names=['', ''])\n",
    "\n",
    "# data = arr_.reshape(len(lam_list)*len(nsample_names), len(opt_names)*2)\n",
    "\n",
    "arr_str = arr_.reshape(len(lam_list)*len(nsample_names), len(opt_names)*2)\n",
    "std_str = std.reshape(len(lam_list)*len(nsample_names), len(opt_names)*2)\n",
    "\n",
    "# Combine with proper string concatenation\n",
    "data = arr_str\n",
    "# data = np.array([[f\"{arr_str[i,j] : .2f} ± {std_str[i,j] : .2f}\" \n",
    "#                   for j in range(arr_str.shape[1])] \n",
    "#                  for i in range(arr_str.shape[0])])\n",
    "data = np.array([[f\"${arr_str[i,j] : .2f} \\\\pm {std_str[i,j] : .2f}$\" \n",
    "                  for j in range(arr_str.shape[1])] \n",
    "                 for i in range(arr_str.shape[0])])\n",
    "\n",
    "\n",
    "# Reshape array and create DataFrame\n",
    "df = pd.DataFrame(\n",
    "    data =  data,\n",
    "    index = row_index,\n",
    "    columns=col_index\n",
    ")\n",
    "\n",
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 120,
   "id": "52a9c702",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\\begin{table*}[t]\n",
      "\\centering\n",
      "\\caption{Sensitivity analysis of \\texttt{CREME} against number of samples $n$ and the robustness level index set size $|\\Lambda|$.}\n",
      "\\label{tab:ablation}\n",
      "\\begin{adjustbox}{width=\\textwidth}\n",
      "\\begin{tabular}{cccccccccc}\n",
      "\\toprule \\midrule\n",
      " &  & \\multicolumn{2}{c}{Linear Programming} & \\multicolumn{2}{c}{Newsvendor} & \\multicolumn{2}{c}{Portfolio Optimization} & \\multicolumn{2}{c}{Shortest Path} \\\\\n",
      " \\cmidrule(lr){3-4}\\cmidrule(lr){5-6}\\cmidrule(lr){7-8}\\cmidrule(lr){9-10}\n",
      " &  & Gap & Time (s) & Gap & Time (s) & Gap & Time (s) & Gap & Time (s) \\\\\n",
      "\\midrule\n",
      "\\multirow[c]{3}{*}{$|\\Lambda|=10$} & $n=10$ & $ 0.10 \\pm  0.03$ & $ 0.58 \\pm  0.03$ & $ 0.19 \\pm  0.08$ & $ 0.00 \\pm  0.00$ & $ 0.16 \\pm  0.07$ & $ 0.84 \\pm  0.10$ & $ 0.11 \\pm  0.03$ & $ 0.72 \\pm  0.04$ \\\\\n",
      " & $n=20$ & $ 0.09 \\pm  0.03$ & $ 0.81 \\pm  0.03$ & $ 0.12 \\pm  0.04$ & $ 0.00 \\pm  0.00$ & $ 0.13 \\pm  0.07$ & $ 1.14 \\pm  0.12$ & $ 0.06 \\pm  0.01$ & $ 0.99 \\pm  0.06$ \\\\\n",
      " & $n=30$ & $ 0.08 \\pm  0.01$ & $ 1.00 \\pm  0.04$ & $ 0.13 \\pm  0.04$ & $ 0.00 \\pm  0.00$ & $ 0.09 \\pm  0.02$ & $ 1.50 \\pm  0.13$ & $ 0.06 \\pm  0.01$ & $ 1.20 \\pm  0.07$ \\\\\n",
      "\\midrule\n",
      "\\multirow[c]{3}{*}{$|\\Lambda|=20$} & $n=10$ & $ 0.10 \\pm  0.03$ & $ 0.58 \\pm  0.03$ & $ 0.19 \\pm  0.08$ & $ 0.00 \\pm  0.00$ & $ 0.16 \\pm  0.07$ & $ 0.84 \\pm  0.10$ & $ 0.11 \\pm  0.03$ & $ 0.72 \\pm  0.04$ \\\\\n",
      " & $n=20$ & $ 0.09 \\pm  0.03$ & $ 0.81 \\pm  0.03$ & $ 0.12 \\pm  0.04$ & $ 0.00 \\pm  0.00$ & $ 0.13 \\pm  0.07$ & $ 1.14 \\pm  0.12$ & $ 0.06 \\pm  0.01$ & $ 0.99 \\pm  0.06$ \\\\\n",
      " & $n=30$ & $ 0.08 \\pm  0.01$ & $ 1.00 \\pm  0.04$ & $ 0.13 \\pm  0.04$ & $ 0.00 \\pm  0.00$ & $ 0.09 \\pm  0.02$ & $ 1.50 \\pm  0.13$ & $ 0.06 \\pm  0.01$ & $ 1.20 \\pm  0.07$ \\\\\n",
      "\\midrule\n",
      "\\multirow[c]{3}{*}{$|\\Lambda|=30$} & $n=10$ & $ 0.09 \\pm  0.03$ & $ 1.73 \\pm  0.06$ & $ 0.16 \\pm  0.07$ & $ 0.00 \\pm  0.00$ & $ 0.17 \\pm  0.08$ & $ 2.50 \\pm  0.28$ & $ 0.11 \\pm  0.04$ & $ 2.13 \\pm  0.08$ \\\\\n",
      " & $n=20$ & $ 0.08 \\pm  0.02$ & $ 2.42 \\pm  0.09$ & $ 0.11 \\pm  0.04$ & $ 0.00 \\pm  0.00$ & $ 0.13 \\pm  0.07$ & $ 3.43 \\pm  0.34$ & $ 0.05 \\pm  0.01$ & $ 2.93 \\pm  0.15$ \\\\\n",
      " & $n=30$ & $ 0.06 \\pm  0.02$ & $ 2.98 \\pm  0.13$ & $ 0.11 \\pm  0.04$ & $ 0.00 \\pm  0.00$ & $ 0.08 \\pm  0.03$ & $ 4.49 \\pm  0.44$ & $ 0.04 \\pm  0.01$ & $ 3.63 \\pm  0.25$ \\\\\n",
      "\\midrule\n",
      "\\bottomrule\n",
      "\\end{tabular}\n",
      "\\end{adjustbox}\n",
      "\\end{table*}\n"
     ]
    }
   ],
   "source": [
    "caption = 'Sensitivity analysis of \\\\texttt{CREME} against number of samples $n$ and the robustness level index set size $|\\\\Lambda|$.'\n",
    "\n",
    "code = df.to_latex()\n",
    "code = '\\n\\\\begin{adjustbox}{width=\\\\textwidth}\\n' + code + '\\\\end{adjustbox}\\n'\n",
    "code = '\\\\begin{table*}[t]\\n\\\\centering\\n\\\\caption{' + caption + '}\\n\\\\label{tab:ablation}' + code + '\\\\end{table*}'\n",
    "code = code.replace('llllllllll', 'cccccccccc')\n",
    "code = code.replace('\\\\multicolumn{2}{r}', '\\\\multicolumn{2}{c}')\n",
    "code = code.replace('\\\\multirow[t]', '\\\\multirow[c]')\n",
    "code = code.replace(' &  &  &  &  &  &  &  &  &  \\\\\\\\\\n', '')\n",
    "code = code.replace('\\\\cline{1-10}', '\\\\midrule')\n",
    "code = code.replace('\\\\toprule', '\\\\toprule \\\\midrule')\n",
    "\n",
    "cmidrule_list = ['\\\\cmidrule(lr){3-4}', '\\\\cmidrule(lr){5-6}', '\\\\cmidrule(lr){7-8}', '\\\\cmidrule(lr){9-10}']\n",
    "code = code.replace('& \\\\multicolumn{2}{c}{Shortest Path} \\\\\\\\', '& \\\\multicolumn{2}{c}{Shortest Path} \\\\\\\\\\n ' + ''.join(cmidrule_list))\n",
    "\n",
    "print(code)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
