{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "from joblib import Parallel, delayed\n",
    "import matplotlib\n",
    "from utils.testing import SeqIndTester"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sns.set(style=\"whitegrid\",\n",
    "        font_scale=1.4,\n",
    "        rc={\n",
    "            \"lines.linewidth\": 2,\n",
    "#             \"axes.facecolor\": \".9\",\n",
    "            'figure.figsize': (8, 6)\n",
    "        })\n",
    "sns.set_palette('Set2')\n",
    "matplotlib.rcParams['text.usetex'] = True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "First sims"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from keras.datasets import mnist"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "(orig_train_X, orig_train_y), (orig_test_X, orig_test_y) = mnist.load_data()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "[1,2,3].pop()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "orig_train_X=orig_train_X.astype(\"float32\") / 255"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.imshow(orig_train_X[27].reshape(28,28), cmap='gray')\n",
    "plt.axis('off')\n",
    "plt.savefig('sims_res/img/three_ex_3', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.imshow(orig_train_X[0].reshape(28,28), cmap='gray')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "orig_train_y[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "y_sort_indeces = np.argsort(orig_train_y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "y_sort_indeces[999]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "orig_train_y[y_sort_indeces]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "first_ind_0 = 0\n",
    "first_ind_1 = np.argmax(orig_train_y[y_sort_indeces]==1)\n",
    "first_ind_2 = np.argmax(orig_train_y[y_sort_indeces]==2)\n",
    "first_ind_3 = np.argmax(orig_train_y[y_sort_indeces]==3)\n",
    "first_ind_4 = np.argmax(orig_train_y[y_sort_indeces]==4)\n",
    "first_ind_5 = np.argmax(orig_train_y[y_sort_indeces]==5)\n",
    "first_ind_6 = np.argmax(orig_train_y[y_sort_indeces]==6)\n",
    "first_ind_7 = np.argmax(orig_train_y[y_sort_indeces]==7)\n",
    "first_ind_8 = np.argmax(orig_train_y[y_sort_indeces]==8)\n",
    "first_ind_9 = np.argmax(orig_train_y[y_sort_indeces]==9)\n",
    "\n",
    "\n",
    "# last_ind_0 = first_ind_1-1\n",
    "# last_ind_1 = first_ind_2-1\n",
    "# last_ind_2 = first_ind_3-1\n",
    "# last_ind_3 = first_ind_4-1\n",
    "# last_ind_4 = first_ind_5-1\n",
    "# last_ind_5 = first_ind_6-1\n",
    "# last_ind_6 = first_ind_7-1\n",
    "# last_ind_7 = first_ind_8-1\n",
    "# last_ind_8 = first_ind_9-1\n",
    "# last_ind_9 = len(y_sort_indeces)-1\n",
    "\n",
    "\n",
    "list_indeces = []\n",
    "list_indeces += [y_sort_indeces[first_ind_0:first_ind_1]]\n",
    "list_indeces += [y_sort_indeces[first_ind_1:first_ind_2]]\n",
    "list_indeces += [y_sort_indeces[first_ind_2:first_ind_3]]\n",
    "list_indeces += [y_sort_indeces[first_ind_3:first_ind_4]]\n",
    "list_indeces += [y_sort_indeces[first_ind_4:first_ind_5]]\n",
    "list_indeces += [y_sort_indeces[first_ind_5:first_ind_6]]\n",
    "list_indeces += [y_sort_indeces[first_ind_6:first_ind_7]]\n",
    "list_indeces += [y_sort_indeces[first_ind_7:first_ind_8]]\n",
    "list_indeces += [y_sort_indeces[first_ind_8:first_ind_9]]\n",
    "list_indeces += [y_sort_indeces[first_ind_9:]]\n",
    "\n",
    "# len_0 = list_indeces[0].shape[0]\n",
    "# len_1 = list_indeces[1].shape[0]\n",
    "# len_2 = list_indeces[2].shape[0]\n",
    "# len_3 = list_indeces[3].shape[0]\n",
    "# len_4 = list_indeces[4].shape[0]\n",
    "# len_5 = list_indeces[5].shape[0]\n",
    "# len_6 = list_indeces[6].shape[0]\n",
    "# len_7 = list_indeces[7].shape[0]\n",
    "# len_8 = list_indeces[8].shape[0]\n",
    "# len_9 = list_indeces[9].shape[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "list_indeces[0].shape[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def sample_random_dep_set_mnist(n):\n",
    "    indices = np.random.choice(np.arange(10, dtype='int'), replace=True, size= n)\n",
    "    list_of_pairs = [np.random.choice(list_indeces[cur_ind], replace= True, size=2).tolist() for cur_ind in indices]\n",
    "    return list_of_pairs\n",
    "\n",
    "def sample_train_data(n):\n",
    "    list_of_dep_pairs = sample_random_dep_set_mnist(n)\n",
    "    x_ind, y_ind = list(zip(*list_of_dep_pairs))\n",
    "    x_seq = orig_train_X[np.array(x_ind)].reshape(n,-1)\n",
    "    y_seq = orig_train_X[np.array(y_ind)].reshape(n,-1)\n",
    "    return x_seq, y_seq\n",
    "\n",
    "def sample_indep_images(n):\n",
    "    x_ind = np.random.choice(np.arange(orig_train_X.shape[0], dtype='int'), replace=True, size= n)\n",
    "    y_ind = np.random.choice(np.arange(orig_train_X.shape[0], dtype='int'), replace=True, size= n)\n",
    "    x_seq = orig_train_X[x_ind].reshape(n,-1)\n",
    "    y_seq = orig_train_X[y_ind].reshape(n,-1)\n",
    "    return x_seq, y_seq"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "orig_train_X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "a=sample_random_dep_set_mnist(4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "a"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "b=list(zip(*a))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sample_size=400\n",
    "x_tr, y_tr = sample_train_data(sample_size)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.imshow(x_tr[2].reshape(28,28), cmap='gray')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.imshow(y_tr[2].reshape(28,28), cmap='gray')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "fig, (ax1, ax2) = plt.subplots(1,2)\n",
    "\n",
    "cur_ind=2\n",
    "# cur_ind=50\n",
    "ax1.imshow(x_tr[cur_ind].reshape(28,28), cmap='gray')\n",
    "ax1.axis('off')\n",
    "ax2.imshow(y_tr[cur_ind].reshape(28,28), cmap='gray')\n",
    "ax2.axis('off')\n",
    "# fig.tight_layout()\n",
    "# plt.colorbar()\n",
    "# print(cand_labels[a[cur_ind]])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "fig, (ax1, ax2) = plt.subplots(1,2)\n",
    "\n",
    "cur_ind=6\n",
    "# cur_ind=50\n",
    "ax1.imshow(x_tr[cur_ind].reshape(28,28), cmap='gray')\n",
    "ax1.axis('off')\n",
    "ax2.imshow(y_tr[cur_ind].reshape(28,28), cmap='gray')\n",
    "ax2.axis('off')\n",
    "fig.tight_layout()\n",
    "# plt.savefig('img/mnist_joint.pdf',bbox_inches='tight')\n",
    "# plt.colorbar()\n",
    "# print(cand_labels[a[cur_ind]])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "fig, (ax1, ax2) = plt.subplots(1,2)\n",
    "\n",
    "cur_ind=6\n",
    "# cur_ind=50\n",
    "ax1.imshow(x_tr[cur_ind].reshape(28,28), cmap='gray')\n",
    "ax1.axis('off')\n",
    "ax2.imshow(y_tr[cur_ind+1].reshape(28,28), cmap='gray')\n",
    "ax2.axis('off')\n",
    "fig.tight_layout()\n",
    "# plt.savefig('img/mnist_product.pdf',bbox_inches='tight')\n",
    "# plt.colorbar()\n",
    "# print(cand_labels[a[cur_ind]])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "global_sims = 200\n",
    "rej_hist_dep = list()\n",
    "rej_hist_indep = list()\n",
    "sample_size = 1000\n",
    "n_pairs = sample_size//2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "for cur_sim in range(global_sims):\n",
    "    rej_hist_dep += [[]]\n",
    "    # sample data\n",
    "    x_tr, y_tr = sample_train_data(sample_size)\n",
    "    # use some data to estimate hyperparam\n",
    "    tester = SeqIndTester()\n",
    "    tester.payoff_style = 'classification'\n",
    "    tester.pred_model = 'MLP'\n",
    "    tester.bet_scheme = 'aGRAPA'\n",
    "    tester.truncation_level = 0.9\n",
    "    for cur_pair in range(n_pairs):\n",
    "        tester.process_pair(x_tr[2 * cur_pair:2 * (cur_pair+1)], y_tr[2 *cur_pair: 2 * (cur_pair+1)])\n",
    "        if tester.null_rejected:\n",
    "            # account for the first pair\n",
    "            rej_hist_dep[cur_sim] += [True]\n",
    "            rej_hist_dep[cur_sim] += [True for _ in range(n_pairs-1-cur_pair)]\n",
    "            break\n",
    "            #add break and fill the rest with true\n",
    "        else:\n",
    "            rej_hist_dep[cur_sim] += [False]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "for cur_sim in range(global_sims):\n",
    "    rej_hist_indep += [[]]\n",
    "    # sample data\n",
    "    x_tr, y_tr = sample_indep_images(sample_size)\n",
    "    # use some data to estimate hyperparam\n",
    "    tester = SeqIndTester()\n",
    "    tester.payoff_style = 'classification'\n",
    "    tester.pred_model = 'MLP'\n",
    "    tester.bet_scheme = 'aGRAPA'\n",
    "    tester.truncation_level = 0.9\n",
    "    for cur_pair in range(n_pairs):\n",
    "        tester.process_pair(x_tr[2 * cur_pair:2 * (cur_pair+1)], y_tr[2 *cur_pair: 2 * (cur_pair+1)])\n",
    "        if tester.null_rejected:\n",
    "            # account for the first pair\n",
    "            rej_hist_indep[cur_sim] += [True]\n",
    "            rej_hist_indep[cur_sim] += [True for _ in range(n_pairs-1-cur_pair)]\n",
    "            break\n",
    "            #add break and fill the rest with true\n",
    "        else:\n",
    "            rej_hist_indep[cur_sim] += [False]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from utils.testing import compute_hyperparam"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "est_size = 20"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "rej_hist_dep_hsic = list()\n",
    "rej_hist_indep_hsic = list()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "for cur_sim in range(global_sims):\n",
    "    rej_hist_dep_hsic += [[]]\n",
    "    # randomly permute data\n",
    "    # use some data to estimate hyperparam\n",
    "    x_tr, y_tr = sample_train_data(sample_size)\n",
    "    x_est, y_est = sample_train_data(est_size)\n",
    "    tester = SeqIndTester()\n",
    "    tester.payoff_style = 'HSIC'\n",
    "    tester.kernel_param_x = compute_hyperparam(x_est)\n",
    "    tester.kernel_param_y = compute_hyperparam(y_est)\n",
    "    tester.bet_scheme = 'aGRAPA'\n",
    "    tester.truncation_level = 0.9\n",
    "    for cur_pair in range(1, n_pairs):\n",
    "        tester.process_pair(x_tr[2 * cur_pair:2 * (cur_pair+1)], y_tr[2 *\n",
    "                            cur_pair: 2 * (cur_pair+1)], x_tr[:2 * cur_pair], y_tr[:2 * cur_pair])\n",
    "        if tester.null_rejected:\n",
    "            # account for the first pair\n",
    "            rej_hist_dep_hsic[cur_sim] += [True]\n",
    "            rej_hist_dep_hsic[cur_sim] += [\n",
    "                True for _ in range(n_pairs-1-cur_pair)]\n",
    "            break\n",
    "            # add break and fill the rest with true\n",
    "        else:\n",
    "            rej_hist_dep_hsic[cur_sim] += [False]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "for cur_sim in range(global_sims):\n",
    "    rej_hist_indep_hsic += [[]]\n",
    "    # randomly permute data\n",
    "    # use some data to estimate hyperparam\n",
    "    x_tr, y_tr = sample_indep_images(sample_size)\n",
    "    x_est, y_est = sample_indep_images(est_size)\n",
    "    tester = SeqIndTester()\n",
    "    tester.payoff_style = 'HSIC'\n",
    "    tester.kernel_param_x = compute_hyperparam(x_est)\n",
    "    tester.kernel_param_y = compute_hyperparam(y_est)\n",
    "    tester.bet_scheme = 'aGRAPA'\n",
    "    tester.truncation_level = 0.9\n",
    "    for cur_pair in range(1, n_pairs):\n",
    "        tester.process_pair(x_tr[2 * cur_pair:2 * (cur_pair+1)], y_tr[2 *\n",
    "                            cur_pair: 2 * (cur_pair+1)], x_tr[:2 * cur_pair], y_tr[:2 * cur_pair])\n",
    "        if tester.null_rejected:\n",
    "            # account for the first pair\n",
    "            rej_hist_indep_hsic[cur_sim] += [True]\n",
    "            rej_hist_indep_hsic[cur_sim] += [\n",
    "                True for _ in range(n_pairs-1-cur_pair)]\n",
    "            break\n",
    "            # add break and fill the rest with true\n",
    "        else:\n",
    "            rej_hist_indep_hsic[cur_sim] += [False]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "rej_hist_dep_mat = np.stack(rej_hist_dep)\n",
    "rej_hist_indep_mat = np.stack(rej_hist_indep)\n",
    "rej_hist_dep_mat_hsic = np.stack(rej_hist_dep_hsic)\n",
    "rej_hist_indep_mat_hsic = np.stack(rej_hist_indep_hsic)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open('mnist_mlp_dep.npy', 'wb') as f:\n",
    "    np.save(f, rej_hist_dep_mat)\n",
    "with open('mnist_mlp_indep.npy', 'wb') as f:\n",
    "    np.save(f, rej_hist_indep_mat)\n",
    "with open('mnist_hsic_dep.npy', 'wb') as f:\n",
    "    np.save(f, rej_hist_dep_mat_hsic)\n",
    "with open('mnist_hsic_indep.npy', 'wb') as f:\n",
    "    np.save(f, rej_hist_indep_mat_hsic)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open('attic_res/mnist_mlp_dep.npy', 'rb') as f:\n",
    "    rej_hist_dep_mat=np.load(f)\n",
    "with open('attic_res/mnist_mlp_indep.npy', 'rb') as f:\n",
    "    rej_hist_indep_mat = np.load(f)\n",
    "with open('attic_res/mnist_hsic_dep.npy', 'rb') as f:\n",
    "    rej_hist_dep_mat_hsic = np.load(f)\n",
    "with open('attic_res/mnist_hsic_indep.npy', 'rb') as f:\n",
    "    rej_hist_indep_mat_hsic=np.load(f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.text as mtext\n",
    "\n",
    "\n",
    "class LegendTitle(object):\n",
    "    def __init__(self, text_props=None):\n",
    "        self.text_props = text_props or {}\n",
    "        super(LegendTitle, self).__init__()\n",
    "\n",
    "    def legend_artist(self, legend, orig_handle, fontsize, handlebox):\n",
    "        x0, y0 = handlebox.xdescent, handlebox.ydescent\n",
    "        title = mtext.Text(x0, y0, orig_handle + ':', usetex=True, **self.text_props)\n",
    "        handlebox.add_artist(title)\n",
    "        return title"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "rej_hist_dep_mat.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "[line1] = plt.plot(1+np.arange(n_pairs), rej_hist_dep_mat.mean(axis=0), linewidth=2,\n",
    "         color='cornflowerblue')\n",
    "[line2] = plt.plot(1+np.arange(n_pairs), rej_hist_indep_mat.mean(axis=0), linewidth=2, linestyle='dashed',\n",
    "         color='cornflowerblue')\n",
    "[line3] = plt.plot(1+np.arange(n_pairs-1), rej_hist_dep_mat_hsic.mean(axis=0), linewidth=2,\n",
    "         color='lightcoral')\n",
    "[line4] = plt.plot(1+np.arange(n_pairs-1), rej_hist_indep_mat_hsic.mean(axis=0), linewidth=2, linestyle='dashed',\n",
    "         color='lightcoral')\n",
    "\n",
    "# plt.errorbar(1+np.arange(n_pairs), rej_hist_dep_mat.mean(axis=0), yerr=2*rej_hist_dep_mat.std(axis=0)/np.sqrt(global_sims), linewidth=2, elinewidth=1.5, capsize=2,\n",
    "#              color='cornflowerblue')\n",
    "\n",
    "plt.plot(1+np.arange(n_pairs), rej_hist_dep_mat.mean(axis=0)-2*rej_hist_dep_mat.std(axis=0)/np.sqrt(global_sims), linewidth=1.5,  linestyle='dotted', alpha=0.75,\n",
    "         color='cornflowerblue')\n",
    "plt.plot(1+np.arange(n_pairs), rej_hist_dep_mat.mean(axis=0)+2*rej_hist_dep_mat.std(axis=0)/np.sqrt(global_sims), linewidth=1.5,  linestyle='dotted',alpha=0.75,\n",
    "         color='cornflowerblue')\n",
    "\n",
    "plt.plot(1+np.arange(n_pairs-1), rej_hist_dep_mat_hsic.mean(axis=0)-2*rej_hist_dep_mat_hsic.std(axis=0)/np.sqrt(global_sims), linewidth=1.5,  linestyle='dotted', alpha=0.75,\n",
    "         color='lightcoral')\n",
    "plt.plot(1+np.arange(n_pairs-1), rej_hist_dep_mat_hsic.mean(axis=0)+2*rej_hist_dep_mat_hsic.std(axis=0)/np.sqrt(global_sims), linewidth=1.5,  linestyle='dotted',alpha=0.75,\n",
    "         color='lightcoral')\n",
    "\n",
    "plt.xlabel('Number of Pairs', fontsize=18)\n",
    "plt.ylabel('Rejection Rate', fontsize=18)\n",
    "plt.axhline(y=0.05, label=r'$\\alpha=0.05$', linestyle='dotted', c='gray')\n",
    "# plt.legend(loc=2, fontsize=18)\n",
    "\n",
    "plt.legend([r'Seq-C-IT (MLP)', line1,line2, r'SKIT', line3, line4], ['', r'$H_1$'+' is true',r'$H_0$'+' is true', '', r'$H_1$'+' is true',r'$H_0$'+' is true'],\n",
    "           handler_map={str: LegendTitle({'fontsize': 18})}, handlelength=3.2, bbox_to_anchor=(0.95, 0.9))\n",
    "plt.savefig('sims_res/img/mnist_power.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.10.6 64-bit",
   "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.10.6"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "aee8b7b246df8f9039afb4144a1f6fd8d2ca17a180786b69acc140d282b71a49"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
