{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:21.866972Z",
     "start_time": "2021-09-27T17:47:19.704011Z"
    }
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import matplotlib\n",
    "from sklearn.metrics import zero_one_loss\n",
    "\n",
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:21.879576Z",
     "start_time": "2021-09-27T17:47:21.869036Z"
    }
   },
   "outputs": [],
   "source": [
    "from utils.data_gen import compute_bayes_risk_binary, compute_bayes_risk_binary_label_shift,\\\n",
    "    generate_2d_example\n",
    "\n",
    "from utils.concentrations import hoeffding_ci_lower_limit,\\\n",
    "    hoeffding_ci_upper_limit\n",
    "from utils.concentrations import pm_bernstein_ci_upper_limit,\\\n",
    "    pm_bernstein_ci_lower_limit\n",
    "\n",
    "from utils.concentrations import pm_bernstein_lower_limit, pm_bernstein_upper_limit\n",
    "from utils.concentrations import pm_hoeffding_upper_limit, pm_hoeffding_lower_limit\n",
    "\n",
    "from utils.data_gen import LDA_predictor\n",
    "\n",
    "from utils.concentrations import betting_ci_lower_limit, betting_ci_upper_limit\n",
    "\n",
    "from utils.tests import Drop_tester,misclas_losses"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:21.887562Z",
     "start_time": "2021-09-27T17:47:21.882754Z"
    }
   },
   "outputs": [],
   "source": [
    "sns.set(\n",
    "    style=\"whitegrid\",\n",
    "    font_scale=1.4,\n",
    "    rc={\n",
    "        \"lines.linewidth\": 2,\n",
    "        #             \"axes.facecolor\": \".9\",\n",
    "        'figure.figsize': (12, 6)\n",
    "    })\n",
    "sns.set_palette('Set2')\n",
    "matplotlib.rcParams['text.usetex'] = True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:21.895174Z",
     "start_time": "2021-09-27T17:47:21.890125Z"
    }
   },
   "outputs": [],
   "source": [
    "legend_dict = ['$\\{0\\}$', '$\\{1\\}$']\n",
    "clr1 = np.array([193, 142, 206]) / 256\n",
    "clr2 = np.array([125, 225, 125]) / 256\n",
    "colors = [clr1, clr2]\n",
    "\n",
    "color1 = np.array([244, 236, 118]) / 256\n",
    "color2 = np.array([110, 200, 235]) / 256\n",
    "color3 = np.array([233, 113, 183]) / 256"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:21.974279Z",
     "start_time": "2021-09-27T17:47:21.897267Z"
    }
   },
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Null vs alternative"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:26.660933Z",
     "start_time": "2021-09-27T17:47:26.619834Z"
    }
   },
   "outputs": [],
   "source": [
    "mu_1 = np.array([1,0])\n",
    "mu_0 = np.array([-1,0])\n",
    "target_prob_class_1 = np.linspace(0.1,0.9,100)\n",
    "\n",
    "eps_tol = 0.05"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:26.827465Z",
     "start_time": "2021-09-27T17:47:26.789281Z"
    }
   },
   "outputs": [],
   "source": [
    "risk_source = compute_bayes_risk_binary_label_shift(0.25, 0.25, mu_1,\n",
    "                                                    mu_0)[0][0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:26.991254Z",
     "start_time": "2021-09-27T17:47:26.949618Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.1270168302054692"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "risk_source"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:27.134027Z",
     "start_time": "2021-09-27T17:47:27.099790Z"
    }
   },
   "outputs": [],
   "source": [
    "target_risks = list()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:27.345856Z",
     "start_time": "2021-09-27T17:47:27.274343Z"
    }
   },
   "outputs": [],
   "source": [
    "for cur_target_prob in target_prob_class_1:\n",
    "    target_risks += [\n",
    "        compute_bayes_risk_binary_label_shift(0.25, cur_target_prob, mu_1,\n",
    "                                              mu_0)[0][0]\n",
    "    ]\n",
    "target_risks = np.array(target_risks)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:27.522165Z",
     "start_time": "2021-09-27T17:47:27.488018Z"
    }
   },
   "outputs": [],
   "source": [
    "nulls = (target_risks-risk_source<=eps_tol)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:28.306136Z",
     "start_time": "2021-09-27T17:47:28.266452Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.43939393939393945"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "target_prob_class_1[np.argmin(nulls == True)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T17:47:31.031227Z",
     "start_time": "2021-09-27T17:47:28.449868Z"
    }
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.scatter(target_prob_class_1[nulls],\n",
    "            target_risks[nulls],\n",
    "            label='Target risk, nulls')\n",
    "plt.scatter(target_prob_class_1[~nulls],\n",
    "            target_risks[~nulls],\n",
    "            label='Target risk, alternatives')\n",
    "plt.axhline(y=risk_source, linestyle='dashed', c='black', label='Source risk')\n",
    "plt.axvline(x=target_prob_class_1[np.argmin(nulls == True)], c='red')\n",
    "plt.legend(loc=2, markerscale=1.5, prop={'size': 20})\n",
    "plt.ylabel('Misclassification risk', fontsize=25)\n",
    "plt.xlabel('Class 1 probability', fontsize=25)\n",
    "# plt.show()\n",
    "plt.savefig('img/nulls_vs_alternatives.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Estimate number of samples to reject"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 75,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T18:03:23.168875Z",
     "start_time": "2021-09-27T18:03:23.134129Z"
    }
   },
   "outputs": [],
   "source": [
    "num_of_repeats = 250\n",
    "num_of_batches_to_sample = 40\n",
    "\n",
    "size_of_batch = 50\n",
    "size_source_sample = 1000\n",
    "\n",
    "max_num_of_target_samples = num_of_batches_to_sample * size_of_batch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 76,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T18:03:23.932484Z",
     "start_time": "2021-09-27T18:03:23.897367Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.05"
      ]
     },
     "execution_count": 76,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "eps_tol"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T18:03:24.148579Z",
     "start_time": "2021-09-27T18:03:24.114331Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "2000"
      ]
     },
     "execution_count": 77,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "max_num_of_target_samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 78,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T18:03:24.308093Z",
     "start_time": "2021-09-27T18:03:24.274669Z"
    }
   },
   "outputs": [],
   "source": [
    "target_cand_probs = np.linspace(0.1,0.9,20)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T18:03:24.782316Z",
     "start_time": "2021-09-27T18:03:24.748437Z"
    }
   },
   "outputs": [],
   "source": [
    "mu_1 = np.array([1, 0])\n",
    "mu_0 = np.array([-1, 0])\n",
    "prob_class_1 = 0.25\n",
    "prob_class_0 = 1 - prob_class_1\n",
    "cov = np.eye(2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T18:03:25.399108Z",
     "start_time": "2021-09-27T18:03:25.365011Z"
    }
   },
   "outputs": [],
   "source": [
    "clf = LDA_predictor()\n",
    "\n",
    "clf.mean_class_0 = mu_0\n",
    "clf.mean_class_1 = mu_1\n",
    "clf.class_0_prior = prob_class_0\n",
    "clf.class_1_prior = prob_class_1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T18:03:25.798917Z",
     "start_time": "2021-09-27T18:03:25.764093Z"
    }
   },
   "outputs": [],
   "source": [
    "tester_hoeffding = Drop_tester()\n",
    "tester_bernstein = Drop_tester()\n",
    "tester_betting_bernstein = Drop_tester()\n",
    "tester_betting = Drop_tester()\n",
    "\n",
    "tester_hoeffding.eps_tol = eps_tol\n",
    "tester_hoeffding.target_conc_type = 'pm_hoeffding'\n",
    "tester_hoeffding.source_conc_type = 'hoeffding'\n",
    "\n",
    "tester_bernstein.eps_tol = eps_tol\n",
    "tester_bernstein.source_conc_type = 'pm_bernstein'\n",
    "tester_bernstein.target_conc_type = 'pm_bernstein'\n",
    "\n",
    "tester_betting.eps_tol = eps_tol\n",
    "tester_betting.source_conc_type = 'betting'\n",
    "tester_betting.target_conc_type = 'betting'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2021-09-27T18:03:26.213648Z",
     "start_time": "2021-09-27T18:03:26.179862Z"
    }
   },
   "outputs": [],
   "source": [
    "bern_num_rejects = list()\n",
    "hoef_num_rejects = list()\n",
    "bet_num_rejects = list()\n",
    "\n",
    "bern_num_samples_to_reject = list()\n",
    "hoef_num_samples_to_reject = list()\n",
    "bet_num_samples_to_reject = list()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "ExecuteTime": {
     "start_time": "2021-09-27T18:03:26.701Z"
    }
   },
   "outputs": [],
   "source": [
    "for cur_ind, cur_target_prob in enumerate(target_cand_probs):\n",
    "\n",
    "    bern_num_samples_to_reject += [[]]\n",
    "    hoef_num_samples_to_reject += [[]]\n",
    "    bet_num_samples_to_reject += [[]]\n",
    "\n",
    "    bern_num_rejects += [0]\n",
    "    hoef_num_rejects += [0]\n",
    "    bet_num_rejects += [0]\n",
    "\n",
    "    for cur_sim in range(num_of_repeats):\n",
    "\n",
    "        X_val_source, y_val_source = generate_2d_example(\n",
    "            0.25, mu_0, mu_1, size_source_sample)\n",
    "\n",
    "        y_pred_val = clf.predict(X_val_source)\n",
    "        ind_loss_source = misclas_losses(y_pred_val, y_val_source)\n",
    "\n",
    "        tester_hoeffding.estimate_risk_source(ind_loss_source)\n",
    "        tester_bernstein.estimate_risk_source(ind_loss_source)\n",
    "        tester_betting.estimate_risk_source(ind_loss_source)\n",
    "\n",
    "        X_new_target, y_new_target = generate_2d_example(\n",
    "            cur_target_prob, mu_0, mu_1,\n",
    "            size_of_batch * num_of_batches_to_sample)\n",
    "\n",
    "        y_pred_target = clf.predict(X_new_target)\n",
    "        ind_loss_target = misclas_losses(y_pred_target, y_new_target)\n",
    "\n",
    "        for cur_batch in range(num_of_batches_to_sample):\n",
    "            cur_losses = ind_loss_target[0:(cur_batch + 1) * size_of_batch]\n",
    "            tester_hoeffding.estimate_risk_target(cur_losses)\n",
    "            \n",
    "            if tester_hoeffding.test_for_drop():\n",
    "                hoef_num_rejects[cur_ind] += 1\n",
    "                hoef_num_samples_to_reject[cur_ind] += [tester_hoeffding.target_num_of_samples_used]\n",
    "                break\n",
    "             \n",
    "        for cur_batch in range(num_of_batches_to_sample):\n",
    "            cur_losses = ind_loss_target[0:(cur_batch + 1) * size_of_batch]\n",
    "            tester_bernstein.estimate_risk_target(cur_losses)\n",
    "            \n",
    "            if tester_bernstein.test_for_drop():\n",
    "                bern_num_rejects[cur_ind] += 1\n",
    "                bern_num_samples_to_reject[cur_ind] += [tester_bernstein.target_num_of_samples_used]\n",
    "                break\n",
    "                \n",
    "        for cur_batch in range(num_of_batches_to_sample):\n",
    "            cur_losses = ind_loss_target[0:(cur_batch + 1) * size_of_batch]\n",
    "            tester_betting.estimate_risk_target(cur_losses)\n",
    "            \n",
    "            if tester_betting.test_for_drop():\n",
    "                bet_num_rejects[cur_ind] += 1\n",
    "                bet_num_samples_to_reject[cur_ind] += [tester_betting.target_num_of_samples_used]\n",
    "                break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "ExecuteTime": {
     "start_time": "2021-09-27T18:03:27.319Z"
    }
   },
   "outputs": [],
   "source": [
    "hoef_fraction_rejected = [num_rej / num_of_repeats for num_rej in hoef_num_rejects]\n",
    "bern_fraction_rejected = [num_rej / num_of_repeats for num_rej in bern_num_rejects]\n",
    "bet_fraction_rejected = [num_rej / num_of_repeats for num_rej in bet_num_rejects]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "ExecuteTime": {
     "start_time": "2021-09-27T18:03:27.588Z"
    }
   },
   "outputs": [],
   "source": [
    "plt.plot(target_cand_probs, hoef_fraction_rejected, marker='*', label = 'Hoeffding\\'s CB',color='indianred')\n",
    "plt.plot(target_cand_probs, bern_fraction_rejected,  marker='*',label = 'PM-EB CB',color='steelblue')\n",
    "plt.plot(target_cand_probs, bet_fraction_rejected,  marker='*',label = 'Betting CB',color='mediumaquamarine')\n",
    "plt.axvline(x=target_prob_class_1[np.argmin(nulls == True)], c='goldenrod', linestyle='dashed')\n",
    "legend=plt.legend(loc=2, markerscale=1.5, prop={'size': 20})\n",
    "legend.get_title().set_fontsize('22')\n",
    "plt.ylabel('Proportion of rejections', fontsize=23)\n",
    "plt.xlabel('Marginal probability of class 1', fontsize=23)\n",
    "plt.savefig('img/prop_rej_sim.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## number of samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "ExecuteTime": {
     "start_time": "2021-09-27T18:31:33.597Z"
    }
   },
   "outputs": [],
   "source": [
    "hoef_lengths = np.array([len(cur_prob) for cur_prob in hoef_num_samples_to_reject])\n",
    "bern_lengths = np.array([len(cur_prob) for cur_prob in bern_num_samples_to_reject])\n",
    "bet_lengths = np.array([len(cur_prob) for cur_prob in bet_num_samples_to_reject])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "ExecuteTime": {
     "start_time": "2021-09-27T18:31:33.812Z"
    }
   },
   "outputs": [],
   "source": [
    "min_num_rej = 5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "ExecuteTime": {
     "start_time": "2021-09-27T18:31:34.197Z"
    }
   },
   "outputs": [],
   "source": [
    "hoef_len_ind = (np.array(hoef_lengths) >= min_num_rej).astype('bool')\n",
    "bern_len_ind = (np.array(bern_lengths) >= min_num_rej).astype('bool')\n",
    "bet_len_ind = (np.array(bet_lengths) >= min_num_rej).astype('bool')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "ExecuteTime": {
     "start_time": "2021-09-27T18:31:36.195Z"
    }
   },
   "outputs": [],
   "source": [
    "hoef_avg = np.array([\n",
    "    np.mean(cur_prob)\n",
    "    for cur_ind, cur_prob in enumerate(hoef_num_samples_to_reject)\n",
    "    if hoef_lengths[cur_ind] >= min_num_rej\n",
    "])\n",
    "\n",
    "bern_avg = np.array([\n",
    "    np.mean(cur_prob)\n",
    "    for cur_ind, cur_prob in enumerate(bern_num_samples_to_reject)\n",
    "    if bern_lengths[cur_ind] >= min_num_rej\n",
    "])\n",
    "\n",
    "bet_avg = np.array([\n",
    "    np.mean(cur_prob)\n",
    "    for cur_ind, cur_prob in enumerate(bet_num_samples_to_reject)\n",
    "    if bet_lengths[cur_ind] >= min_num_rej\n",
    "])\n",
    "\n",
    "hoef_stds = np.array([\n",
    "    np.std(cur_prob)\n",
    "    for cur_ind, cur_prob in enumerate(hoef_num_samples_to_reject)\n",
    "    if hoef_lengths[cur_ind] >= min_num_rej\n",
    "])\n",
    "\n",
    "bern_stds = np.array([\n",
    "    np.std(cur_prob)\n",
    "    for cur_ind, cur_prob in enumerate(bern_num_samples_to_reject)\n",
    "    if bern_lengths[cur_ind] >= min_num_rej\n",
    "])\n",
    "\n",
    "bet_stds = np.array([\n",
    "    np.std(cur_prob)\n",
    "    for cur_ind, cur_prob in enumerate(bet_num_samples_to_reject)\n",
    "    if bet_lengths[cur_ind] >= min_num_rej\n",
    "])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "ExecuteTime": {
     "start_time": "2021-09-27T18:31:37.167Z"
    }
   },
   "outputs": [],
   "source": [
    "plt.plot(target_cand_probs[hoef_len_ind],\n",
    "         hoef_avg,\n",
    "         marker='*',\n",
    "         label='Hoeffding\\'s CB',color='indianred')\n",
    "\n",
    "plt.fill_between(\n",
    "    target_cand_probs[hoef_len_ind],\n",
    "    y1=hoef_avg - 2 * hoef_stds / np.sqrt(hoef_lengths[hoef_len_ind]),\n",
    "    y2=hoef_avg + 2 * hoef_stds / np.sqrt(hoef_lengths[hoef_len_ind]),\n",
    "    alpha=0.5,color='indianred')\n",
    "\n",
    "plt.plot(target_cand_probs[bern_len_ind],\n",
    "         bern_avg,\n",
    "         marker='*',\n",
    "         label='PM-EB CB',color='steelblue')\n",
    "\n",
    "plt.fill_between(\n",
    "    target_cand_probs[bern_len_ind],\n",
    "    y1=bern_avg - 2 * bern_stds / np.sqrt(bern_lengths[bern_len_ind]),\n",
    "    y2=bern_avg + 2 * bern_stds / np.sqrt(bern_lengths[bern_len_ind]),\n",
    "    alpha=0.5,color='steelblue')\n",
    "\n",
    "plt.plot(target_cand_probs[bet_len_ind],\n",
    "         bet_avg,\n",
    "         marker='*',\n",
    "         label='Betting CB',color='mediumaquamarine')\n",
    "\n",
    "plt.fill_between(\n",
    "    target_cand_probs[bet_len_ind],\n",
    "    y1=bet_avg - 2 * bet_stds / np.sqrt(bet_lengths[bet_len_ind]),\n",
    "    y2=bet_avg + 2 * bet_stds / np.sqrt(bet_lengths[bet_len_ind]),\n",
    "    alpha=0.5,color='mediumaquamarine')\n",
    "\n",
    "plt.legend(loc=2, markerscale=1.5, prop={'size': 20})\n",
    "plt.ylabel('Sample size to reject the null', fontsize=23)\n",
    "plt.xlabel('Marginal probability of class 1', fontsize=23)\n",
    "plt.axvline(x=target_prob_class_1[np.argmin(nulls == True)], c='goldenrod', linestyle='dashed')\n",
    "plt.savefig('img/number_of_samples.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.7.7"
  },
  "toc": {
   "base_numbering": 1,
   "nav_menu": {},
   "number_sections": true,
   "sideBar": true,
   "skip_h1_title": false,
   "title_cell": "Table of Contents",
   "title_sidebar": "Contents",
   "toc_cell": false,
   "toc_position": {},
   "toc_section_display": true,
   "toc_window_display": false
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
