{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "source": [
    "# External imports \n",
    "import torch\n",
    "from torch.utils.data import DataLoader\n",
    "import random\n",
    "import numpy as np\n",
    "from tqdm import trange\n",
    "import matplotlib.pyplot as plt\n",
    "from IPython.display import display, clear_output\n",
    "\n",
    "# Internal imports\n",
    "import sys; sys.path.insert(0, '..')\n",
    "from src import *"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "source": [
    "N_DIMS = 1\n",
    "NUM_SAMPLES = 50000\n",
    "BS = 500\n",
    "NUM_EPOCHS = 800\n",
    "SEED = 10\n",
    "LR = 1e-2\n",
    "DROPOUT = 0.20\n",
    "DEVICE = 'cuda:1' if torch.cuda.is_available() else 'cpu'\n",
    "\n",
    "\n",
    "# Break by changing num datapoints, scales, means, or to 2D"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "source": [
    "# Setting seed for reproducibility\n",
    "random.seed(SEED)\n",
    "torch.manual_seed(SEED)\n",
    "np.random.seed(SEED)"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "source": [
    "# Define model\n",
    "model = RatioCritic1D(dim_input=N_DIMS, dim_output=3, dropout=DROPOUT)\n",
    "# model.apply(weights_init)\n",
    "\n",
    "# Define optimizer\n",
    "optim = torch.optim.Adam(model.parameters(), lr=LR)\n",
    "\n",
    "\n",
    "\n",
    "# Define distributions\n",
    "p, q, m = get_dists_1d(mu1=-5., mu2=5., mu3=0, scale_p=1., scale_q=1., scale_m=1.0)\n",
    "\n",
    "# -5, 5, m_var=3.0\n",
    "# -10, 10, m_var=3.0"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "source": [
    "# Define dataset & dataloader\n",
    "train_ds = DistDataset(p, q, m, num_samples=NUM_SAMPLES)\n",
    "test_ds = DistDataset(p, q, m, num_samples=NUM_SAMPLES) # Test dataset is only of size batch "
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "Sampling p\n",
      "Sampling q\n",
      "Cauchy(loc: 0.0, scale: 1.0)\n",
      "Sampling m\n",
      "torch.Size([50000])\n",
      "torch.Size([50000])\n",
      "torch.Size([50000])\n",
      "Sampling p\n",
      "Sampling q\n",
      "Cauchy(loc: 0.0, scale: 1.0)\n",
      "Sampling m\n",
      "torch.Size([50000])\n",
      "torch.Size([50000])\n",
      "torch.Size([50000])\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "source": [
    "# Define dataloader\n",
    "train_dl = DataLoader(train_ds, batch_size=BS, shuffle=True)\n",
    "test_dl = DataLoader(test_ds, batch_size=BS, shuffle=True)"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "source": [
    "# Set up viz\n",
    "fig, [ax1,ax2,ax3] = plt.subplots(1, 3,figsize=(15,4))\n",
    "\n",
    "line, = ax1.plot([0,1],[0,1])\n",
    "x, y = np.random.random((2, 500))\n",
    "scat1 = ax2.scatter(x,y,label='True p/q',alpha=0.9,s=10.,c='b')\n",
    "scat2 = ax2.scatter(x,y,label='CoB p/q',alpha=0.9,s=10.,c='r')\n",
    "test_line, = ax3.plot([0,1],[0,1])\n",
    "\n",
    "ax1.set_xlabel(\"Iteration\")\n",
    "ax1.set_ylabel(\"Train Loss\")\n",
    "ax1.set_xlim([0,NUM_EPOCHS*NUM_SAMPLES//BS])\n",
    "ax1.set_ylim([0,10])\n",
    "\n",
    "ax2.set_xlabel(\"Samples\")\n",
    "ax2.set_ylabel(\"Log Ratio\")\n",
    "ax2.legend(loc='best')\n",
    "ax2.set_xlim([-6,10])\n",
    "ax2.set_ylim([-1500,5000])\n",
    "\n",
    "ax3.set_xlabel(\"Iteration\")\n",
    "ax3.set_ylabel(\"Test Loss\")\n",
    "ax3.set_xlim([0,NUM_EPOCHS*NUM_SAMPLES//BS])\n",
    "ax3.set_ylim([0,10])\n",
    "\n",
    "plt.tight_layout()\n",
    "\n",
    "loss_store = []\n",
    "test_loss_store = []"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1080x288 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "source": [
    "## CONFIRM q_list_test in validation/visualization in code\n",
    "\n",
    "model.train()\n",
    "\n",
    "if torch.cuda.is_available():\n",
    "    model = model.to(DEVICE)\n",
    "    \n",
    "i = 0\n",
    "# loss_crit = torch.nn.CrossEntropyLoss()\n",
    "loss_crit = torch.nn.functional.cross_entropy\n",
    "\n",
    "\n",
    "for epoch in trange(NUM_EPOCHS):\n",
    "    for p_batch, q_batch, m_batch in iter(train_dl):\n",
    "        model.train()\n",
    "        i += 1\n",
    "        \n",
    "        model.zero_grad()\n",
    "        \n",
    "        # CUDA\n",
    "        if torch.cuda.is_available():\n",
    "            p_batch, q_batch, m_batch = p_batch.unsqueeze(1).to(DEVICE), q_batch.unsqueeze(1).to(DEVICE), m_batch.unsqueeze(1).to(DEVICE)\n",
    "            \n",
    "        logP = model(p_batch)\n",
    "        logP = logP[:,0] - logP[:,1]\n",
    "        logQ = model(q_batch)\n",
    "        logQ = logQ[:,0] - logQ[:,1]\n",
    "        logM = model(m_batch)\n",
    "        \n",
    "        p_label = torch.empty(p_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(1)\n",
    "        q_label = torch.empty(q_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(0)\n",
    "        m_label = torch.empty(m_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(2)\n",
    "    \n",
    "        loss = torch.nn.functional.binary_cross_entropy_with_logits(logP, p_label.float()) + torch.nn.functional.binary_cross_entropy_with_logits(logQ, q_label.float())\n",
    "#         loss.backward()\n",
    "#         optim.step()\n",
    "        loss_store.append(loss.item())\n",
    "\n",
    "        # Validation/Test\n",
    "        if i % 50 == 0:\n",
    "            model.eval()\n",
    "            \n",
    "            with torch.no_grad():\n",
    "                for p_batch, q_batch, m_batch in iter(test_dl):\n",
    "                    log_ratio_p_q, _, true_kl_p_q = get_gt_ratio_kl(p, q, m_batch, calc_true_kl=True)\n",
    "                    _, kl_from_p_q = get_gt_ratio_kl(p, q, p_batch)\n",
    "\n",
    "                    if torch.cuda.is_available():\n",
    "                        p_batch, q_batch, m_batch = p_batch.unsqueeze(1).to(DEVICE), q_batch.unsqueeze(1).to(DEVICE), m_batch.unsqueeze(1).to(DEVICE)\n",
    "                    \n",
    "                    logP = model(p_batch)\n",
    "                    logP = logP[:,0] - logP[:,1]\n",
    "                    logQ = model(q_batch)\n",
    "                    logQ = logQ[:,0] - logQ[:,1]\n",
    "                    logM = model(m_batch)\n",
    "\n",
    "                    log_ratio_p_q_from_cob = logP\n",
    "                    kl_from_cob = torch.mean(log_ratio_p_q_from_cob)\n",
    "                    \n",
    "                    log_ratio_p_q_from_cob = logM[:, 0] - logM[:, 1]\n",
    "\n",
    "                    p_label = torch.empty(p_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(1)\n",
    "                    q_label = torch.empty(q_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(0)\n",
    "                    m_label = torch.empty(m_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(2)\n",
    "                    \n",
    "                    test_loss = torch.nn.functional.binary_cross_entropy_with_logits(logP, p_label.float()) + torch.nn.functional.binary_cross_entropy_with_logits(logQ, q_label.float())\n",
    "\n",
    "                    # Visualize\n",
    "                    \n",
    "                    line.set_data(range(len(loss_store)), loss_store)\n",
    "                    ax1.set_xlim( 0, len(loss_store) )\n",
    "                    \n",
    "                    scat1.set_offsets(np.vstack([m_batch.cpu().squeeze(), log_ratio_p_q.cpu().detach()]).T)\n",
    "                    scat2.set_offsets(np.vstack([m_batch.cpu().squeeze(), log_ratio_p_q_from_cob.cpu().detach()]).T)\n",
    "\n",
    "                    ax2.set_xlim( -50., 25. )\n",
    "                    ax2.set_ylim( -1000, 1000)\n",
    "            \n",
    "                    test_loss_store.append(test_loss.item())\n",
    "                    test_line.set_data(range(len(test_loss_store)), test_loss_store)\n",
    "                    ax3.set_xlim( 0, len(test_loss_store) )\n",
    "                    print('iteration: ',i)\n",
    "                    print('KLD: ', true_kl_p_q)\n",
    "                    print('CoB: ', kl_from_cob)\n",
    "                    \n",
    "                    clear_output(wait=True)\n",
    "                    display(fig)\n",
    "                    break\n",
    "\n",
    "            model.train()"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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klQX991337DP+d73vlf3OCRJ7cOBOCVJkiRJUksyaSFJkiRJklqSSQtJkiRJktSSTFpIkiRJkqSWZNJCkiSpQzz00EMceeSR7Ljjjuyxxx4cfPDB/P73vx/0+Pvuu49x48YxZcoUdt99d970pjdx5513Dvu+X/7ylznnnHM2JHRJUocyaSFJHSAido6IhTXLExHx0Yg4OSIeqCk/uOac6RGxKCLujIgDqoxf0obLTA4//HC6urq4++67WbBgATNnzuThhx9e63k77rgjCxcu5KabbuLoo4/m1FNPHfa958+fT3d39/qGLknqYCYtJKkDZOadmTklM6cAewBPAReUu0/r25eZlwBExC7AkcCuwIHA/4uIURWELqlOfvnLX7Lxxhtz/PHHryrbfffdeetb30pm8qlPfYrXvOY17LbbbsybN2/AazzxxBNsvfXWzyvv7e1ln3324R3veAc777wzxx9/PCtXrlx1zrPPPsuECRO499572Xvvvdltt9343Oc+x2abbdaYL1aS1DZGVx2AJKnp9gPuzsw/RMRgxxwG/DAzlwH3RsQiYE/gf5oUoySAnh7o7YWuLtjAlgq33nore+yxx4D7zj///FWtKf70pz/xhje8gX322QeAu+++mylTpvDkk0/y1FNPce211w54jeuuu47bb7+dl73sZRx44IGcf/75vPvd7+aKK65gv/32A+DEE0/kQx/6EO9///uZNWvWBn09kqTOYEsLSeo8RwI/qNk+ISJujog5EdH3L9Ttgftrjllclq0hIo6LiBsi4oYlS5Y0LmKpE/X0wFFHwaxZxWtPT8NuddVVVzFt2jRGjRrFi1/8Yv7mb/6G66+/HljdPeTuu+/m9NNP57jjjhvwGnvuuScvf/nLGTVqFNOmTeOqq64C4LLLLuOggw4C4Oqrr2batGkAvO9972vY1yNJah8mLSSpg0TEJsChwI/KojOAHYEpwIPAN4ZzvcycnZlTM3PqhAkT6hmqpN5eWLYMxo8vXnt7N+hyu+66KwsWLNigaxx66KFceeWVA+7r33Krb/u6665jzz33HPQ4SZLWxqSFJHWWg4AbM/NhgMx8ODNXZOZK4D8puoAAPAC8tOa8iWWZpGbp6oIxY2Dp0uK1q2uDLrfvvvuybNkyZs+evars5ptv5te//jVvfetbmTdvHitWrGDJkiVceeWVayQa+lx1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",
      "text/plain": [
       "<Figure size 1080x288 with 3 Axes>"
      ]
     },
     "metadata": {}
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "100%|██████████| 800/800 [20:02<00:00,  1.50s/it]\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "source": [
    "                    print('iteration: ',i)\n",
    "                    print('KLD: ', true_kl_p_q)\n",
    "                    print('CoB: ', kl_from_cob)"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "iteration:  80000\n",
      "KLD:  tensor(50.)\n",
      "CoB:  tensor(-53.7400, device='cuda:1')\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "source": [
    "# Set up viz\n",
    "fig, ax2 = plt.subplots(1, 1,figsize=(6,4))\n",
    "\n",
    "x, y = np.random.random((2, 500))\n",
    "scat1 = ax2.scatter(x,y,label='True Log p/q, KL = '+str(np.around(true_kl_p_q.item(),2)),alpha=0.9,s=10.,c='b')\n",
    "scat2 = ax2.scatter(x,y,label='Single Ratio Log p/q, KL = '+str(np.around(kl_from_cob.item(),2)),alpha=0.9,s=10.,c='r')\n",
    "\n",
    "scat1.set_offsets(np.vstack([m_batch.cpu().squeeze(), log_ratio_p_q.cpu().detach()]).T)\n",
    "scat2.set_offsets(np.vstack([m_batch.cpu().squeeze(), log_ratio_p_q_from_cob.cpu().detach()]).T)                  \n",
    "\n",
    "ax2.set_xlabel(\"Samples\")\n",
    "ax2.set_ylabel(\"Log Ratio\")\n",
    "ax2.legend(loc='best')\n",
    "ax2.set_xlim([-25,25])\n",
    "ax2.set_ylim([-1000,1000])\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig('../plots/bcfre_mu5.png')"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAagAAAEYCAYAAAAJeGK1AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8vihELAAAACXBIWXMAAAsTAAALEwEAmpwYAAAxQ0lEQVR4nO3deXxU5fn//9dF2GRzpYoEtwIuIAYJCIo2igRcwOLHBUVFraVY7YKftl/BDawW27r012q1SEFbreKGqBUNLqj4ESHBGFFKRcUCRURQgaBhu35/nDPDkEzCBGYyJ8n7+XjMY+bc55yZe4Yw19zLuW5zd0RERKKmSbYrICIikowClIiIRJIClIiIRJIClIiIRJIClIiIRJIClIiIRFJWA5SZTTGzz81sYULZPmY2y8w+DO/3DsvNzP5oZkvMrMzMjk04Z2R4/IdmNjIb70VERNIr2y2oB4DBlcquBV529y7Ay+E2wGlAl/A2CrgXgoAG3AQcB/QBbooFNRERqb+yGqDc/XVgbaXis4AHw8cPAt9PKP+bB+YCe5lZB2AQMMvd17r7l8AsqgY9ERGpZ5pmuwJJ7O/uK8PHnwH7h487AssSjlsellVXXoWZjSJofdG6deteRxxxRBqrLSIiu6KkpOQLd29fuTyKASrO3d3M0paLyd0nAZMA8vPzvbi4OF1PLSIiu8jMPk1Wnu0xqGRWhV13hPefh+UrgE4Jx+WGZdWVi4hIPRbFAPUMEJuJNxKYkVB+STibry/wddgV+CJQaGZ7h5MjCsMyERGpx7LaxWdmjwAFwH5mtpxgNt5twGNm9gPgU+C88PDngdOBJcBG4DIAd19rZr8G5ofH3ezulSdeiIhIPWONdbkNjUFJJmzevJnly5fz7bffZrsqIpHTsmVLcnNzadas2Q7lZlbi7vmVj4/0JAmR+mb58uW0bduWQw45BDPLdnVEIsPdWbNmDcuXL+fQQw9N6ZwojkGJ1Fvffvst++67r4KTSCVmxr777lur3gUFKJE0U3ASSa62/zcUoEREJJIUoEQakDVr1pCXl0deXh4HHHAAHTt2jG9v2rQpLa9RUFBAXU4wmjt3Lj/84Q936dxLL72UJ554AoC1a9fSs2dPpk6dytKlS+nevXs6q8khhxzC0UcfTV5eHvn528f7165dy8CBA+nSpQsDBw7kyy+/THr+gw8+SJcuXejSpQsPPvhg0mMaGwUokQZk3333pbS0lNLSUkaPHs2YMWPi282bN2fLli3ZrmKtzZw5k8GDdy+95tdff82gQYMYNWoUl112WZpqVtWrr75KaWnpDgH8tttuY8CAAXz44YcMGDCA2267rcp5a9euZcKECbz99tvMmzePCRMmVBvIGhMFKJEG7tJLL2X06NEcd9xx/OpXv2L8+PHcfvvt8f3du3dn6dKlADz00EP06dOHvLw8fvSjH7F169aUXmPt2rV8//vfp0ePHvTt25eysjIAVq9ezcCBA+nWrRtXXHEFBx98MF988UWV89u0acOYMWPo1q0bAwYMYPXq1fF9L7/8MqeeeirffPMNw4cP58gjj2TYsGEcd9xxKbXkNmzYwGmnncaFF17IlVdemdL7SacZM2YwcmSQe2DkyJE8/fTTVY558cUXGThwIPvssw977703AwcO5IUXXqjjmkaPApRIlhUVwbhxwX2mLF++nP/7v//jzjvvrPaYRYsWMW3aNN58801KS0vJycnh4YcfTun5b7rpJnr27ElZWRm/+c1vuOSSSwCYMGECp5xyCu+//z7nnHMO//nPf5KeX15eTn5+Pu+//z7f+973mDBhAgBffPEFzZo1Y8899+Tee++lVatWLFq0iAkTJlBSUpJS3a655hr69+/PmDFjUjo+5uGHH453jybezjnnnKTHmxmFhYX06tWLSZMmxctXrVpFhw4dADjggANYtWpVlXNXrFhBp07bM7bl5uayYoUytuk6KJEsKiqCiy6CigqYPBkeeggKC9P/Oueeey45OTk1HvPyyy9TUlJC7969Afjmm2/4zne+k9Lzz5kzhyeffBKAU045hTVr1rBu3TrmzJnD9OnTARg8eDB77518qbYmTZpw/vnnA3DRRRdx9tlnA1BUVERh+IG8/vrr/PSnPwWgR48e9OjRI6W6nXLKKcyYMYNf/OIXKb8fgBEjRjBixIiUj58zZw4dO3bk888/Z+DAgRxxxBGcdNJJOxxjZprlWQtqQYlk0ezZQXBq3Tq4nz07M6/TunXr+OOmTZuybdu2+HbsuhR3Z+TIkfExq8WLFzN+/PjMVGgnYl/i6Rh/Gj58OKNHj+b0009n/fr1KZ9X2xZUx47BKj/f+c53GDZsGPPmzQNg//33Z+XKYAWhlStXJg2SHTt2ZNmy7asGLV++PP58jZkClEgWFRRAixZQXh7cFxRk/jUPOeQQFixYAMCCBQv45JNPABgwYABPPPEEn38eLCCwdu1aPv006SoIVZx44onx7sDZs2ez33770a5dO0444QQee+wxIGgNVTfwv23btvhsu3/84x/0798fd6esrIy8vDwATjrpJP7xj38AsHDhwvg4F8All1wSDwjJjBkzhgEDBnD22WenPJtxxIgR8WCdeIvVM1F5eXk8+JWXl1NUVBSfJTh06ND4rLwHH3yQs846q8r5gwYNin8+X375JUVFRQwaNCilejZkClAiWVRYGHTrXXVV5rr3Kvuf//kf1q5dS7du3bj77rvp2rUrAEcddRS33HILhYWF9OjRg4EDB8Z/+Vd2xhlnkJubS25uLueeey7jx4+npKSEHj16cO2118a/kG+66ab4l/Xjjz/OAQccQNu2bas8X+vWrZk3bx7du3fnlVde4cYbb6SkpISePXvGW1NXXnklGzZs4Mgjj+TGG2+kV69e8fPLyso48MADa3zfv/3tb8nNzeXiiy9m27ZtLF68OP4ecnNzefzxx3fp84RgnKl///4cc8wx9OnThzPOOCPe8rv22muZNWsWXbp04aWXXuLaa68FoLi4mCuuuAKAffbZhxtuuIHevXvTu3dvbrzxRvbZZ59drk9DoWSxImm0aNEijjzyyGxXIzIqKirIycmhadOmvPXWW1x55ZWUlpZWOa5NmzZs2LBhh7JbbrmFzp07M3z48KTPXVBQwO23307Xrl35wQ9+sFsBRupOsv8jShYrInXuP//5D+eddx7btm2jefPm3H///Smfe/3116d0XLt27RScGigFKBHJmC5duvDOO+/s9LjKradUzM7UjBKJDI1BiYhIJClAiYhIJClAiYhIJClAiYhIJClAiTQwt956K926daNHjx7k5eXx9ttvA3DFFVfwwQcf7NJz7sryFDk5OeTl5dG9e3eGDBnCV199VePxpaWlPP/88/HtZ555Jmnm7+o88MADXH311bWq4+7q1asXFRUVtT5v9uzZnHnmmfHt66+/nsGDB1NRUZHx5UyqWxbkhhtuiP/NFBYW8t///rfKua+++uoOWTVatmxZJfntT3/6U9q0aZOWukYyQJnZ4WZWmnBbZ2Y/N7PxZrYiofz0hHPGmtkSM1tsZroEWxqlt956i+eee44FCxZQVlbGSy+9FE9COnnyZI466qg6q8see+xBaWkpCxcuZJ999uGee+6p8fjKAWro0KHxi1qj6JNPPqFjx460aNFit57nlltu4c0332T69Om7/VypSrYsyC9/+UvKysooLS3lzDPP5Oabb65y3sknnxzPqPHKK6/QqlWreK5ECC4+TucyIZEMUO6+2N3z3D0P6AVsBKaHu++K7XP35wHM7ChgONANGAz82cxqzowp0gCtXLmS/fbbL/5Ft99++8UzLCT+Mm/Tpg3XXXcdxxxzDH379o1n2P7oo4/o27cvRx99NNdff33SX8Jbt27ll7/8Jb1796ZHjx785S9/2Wm9+vXrF8/OPW/ePPr160fPnj05/vjjWbx4MZs2beLGG29k2rRp5OXlMW3atB1aREuXLuWUU06hR48eDBgwoNqs6MnceeeddO/ene7du/OHP/whXv7rX/+aww8/nP79+3PBBRfssARJTGypkvz8fLp27cpzzz0X3/fCCy/Es0VMnTqVrl270qdPH374wx+m3JK74447mDlzJs8++yx77LFHyu8pE9q1axd/XF5evtOktk888QSnnXYarVq1Arb/Xfzud79LW50iGaAqGQB85O41JQU7C3jU3Svc/RNgCdCnTmonsrvSuN5GYWEhy5Yto2vXrvz4xz/mtddeS3pceXk5ffv25d133+Wkk06KX0D7s5/9jJ/97Ge899575ObmJj33r3/9K3vuuSfz589n/vz53H///fF8fsls3bqVl19+maFDhwJwxBFH8MYbb/DOO+9w8803M27cOJo3b87NN9/M+eefT2lpaTyzecxPfvITRo4cSVlZGSNGjIhnNd+ZkpISpk6dyttvv83cuXO5//77eeedd5g/fz5PPvkk7777LjNnzqyxS23p0qXMmzePf/7zn4wePTqeXDcWoFauXMlNN93Em2++yZw5c1LuRn3zzTe57777mDlzZq26xNavX580iW1eXl5Kr13dsiAA1113HZ06deLhhx9O2oJK9Oijj3LBBRfEt++++26GDh0aX1okHepDgBoOPJKwfbWZlZnZFDOL5e7vCCxLOGZ5WLYDMxtlZsVmVpy4IJpI1sTW27jnnuB+N4NUmzZtKCkpYdKkSbRv357zzz+fBx54oMpxzZs3j4+B9OrVK75g4VtvvcW5554LwIUXXlhNlYv429/+Rl5eHscddxxr1qzhww8/rHLcN998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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "source": [],
   "outputs": [],
   "metadata": {}
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "sr",
   "language": "python",
   "name": "sr"
  },
  "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.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}