{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 14,
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "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": 16,
   "source": [
    "N_DIMS = 1\n",
    "NUM_SAMPLES = 50000\n",
    "BS = 500\n",
    "NUM_EPOCHS = 400\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": 17,
   "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": 18,
   "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=-2., mu2=2., mu3=0, scale_p=.08, scale_q=.15, 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": 19,
   "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",
      "Sampling m\n",
      "Sampling p\n",
      "Sampling q\n",
      "Sampling m\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "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": 21,
   "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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uDj55JN7Ld9ggw24qo8ZQ4844giOOOKINdb96U9/mi9/+csr188//3ymNb2JiOBf//VfV65/8IMfHFjQkqSuZwJDkiRJLXPUUUdVHYIkqUOYwJAkSdKQeOihh9b6Neecc86QxyFJ6kxO4ilJkiRJkmrPBIYkSZIkSao9ExiSJEmSJKn2TGBIkiRJkqTaM4EhSZLUJR577DGOPPJItttuO/bYYw8OPvhg/vu//7vP/R966CE22GADJk6cyG677cbb3vY27r333rVu99RTT+WCCy4YTOiSJJnAkCRJ6gaZyWGHHcaUKVO4//77WbBgAV/60pf47W9/2+/rtttuOxYuXMhtt93GzJkz+ad/+qe1bvvKK69k2rRp6xq6JEmACQxJkqSucN111zFq1CiOPvrolWW77bYbb3/728lMPv3pT7Pzzjuzyy67MGfOnF7rePrpp9l8881fUT537lz2228/DjnkEHbYYQeOPvpoVqxYsfI1L7zwAmPGjOHBBx9k7733ZpddduHv//7v2WijjVrzZiVJHWlk1QFIkiSpD1ddBXPnwpQpMMgRDHfeeSd77LFHr9suvfTSlaMsfve737Hnnnuy3377AXD//fczceJE/vCHP/DHP/6Rm2++udc65s2bx913380b3vAGpk+fzqWXXsrhhx/ONddcw/777w/Acccdx0c/+lE+8IEPcOaZZw7q/UiSuo8jMCRJkuroqqvgqKPgzDMbz1dd1bKmbrjhBmbMmMGIESPYcsst+Yu/+AtuueUW4OVLSO6//37OOOMMZs2a1WsdkydP5o1vfCMjRoxgxowZ3HDDDQBcccUVHHTQQQDceOONzJgxA4D3v//9LXs/kqTOZAJDkiSpjubOheefhw03bDzPnTuo6nbaaScWLFgwqDre9a53cf311/e6LSJ6XZ83bx6TJ0/ucz9JkgbKBIYkSVIdTZkC668Pzz7beJ4yZVDVTZ06leeff57Zs2evLLv99tv52c9+xtvf/nbmzJnD8uXLWbp0Kddff/0qSYceN9xwA9ttt12v9c+bN48HH3yQFStWMGfOHPbdd1/uuusudtxxR0aMGAHAPvvsw0UXXQTgXUkkSWvNBIYkSVIdTZsG558PxxzTeB7kHBgRwWWXXcY111zDdtttx0477cSJJ57IVlttxWGHHcauu+7KbrvtxtSpU/nKV77CVlttBbw8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",
      "text/plain": [
       "<Figure size 1080x288 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "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": "stream",
     "name": "stderr",
     "text": [
      "  2%|▏         | 8/400 [00:17<14:20,  2.19s/it]\n"
     ]
    },
    {
     "output_type": "error",
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-45-6d3ec4e6b57b>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m     85\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     86\u001b[0m                     \u001b[0mclear_output\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mwait\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mTrue\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 87\u001b[0;31m                     \u001b[0mdisplay\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfig\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     88\u001b[0m                     \u001b[0;32mbreak\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     89\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/IPython/core/display.py\u001b[0m in \u001b[0;36mdisplay\u001b[0;34m(include, exclude, metadata, transient, display_id, *objs, **kwargs)\u001b[0m\n\u001b[1;32m    311\u001b[0m             \u001b[0mpublish_display_data\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mdata\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mobj\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmetadata\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mmetadata\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    312\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 313\u001b[0;31m             \u001b[0mformat_dict\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmd_dict\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mformat\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mobj\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0minclude\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0minclude\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexclude\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mexclude\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    314\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mformat_dict\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    315\u001b[0m                 \u001b[0;31m# nothing to display (e.g. _ipython_display_ took over)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/IPython/core/formatters.py\u001b[0m in \u001b[0;36mformat\u001b[0;34m(self, obj, include, exclude)\u001b[0m\n\u001b[1;32m    178\u001b[0m             \u001b[0mmd\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    179\u001b[0m             \u001b[0;32mtry\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 180\u001b[0;31m                 \u001b[0mdata\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mformatter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mobj\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    181\u001b[0m             \u001b[0;32mexcept\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    182\u001b[0m                 \u001b[0;31m# FIXME: log the exception\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m<decorator-gen-2>\u001b[0m in \u001b[0;36m__call__\u001b[0;34m(self, obj)\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/IPython/core/formatters.py\u001b[0m in \u001b[0;36mcatch_format_error\u001b[0;34m(method, self, *args, **kwargs)\u001b[0m\n\u001b[1;32m    222\u001b[0m     \u001b[0;34m\"\"\"show traceback on failed format call\"\"\"\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    223\u001b[0m     \u001b[0;32mtry\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 224\u001b[0;31m         \u001b[0mr\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mmethod\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    225\u001b[0m     \u001b[0;32mexcept\u001b[0m \u001b[0mNotImplementedError\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    226\u001b[0m         \u001b[0;31m# don't warn on NotImplementedErrors\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/IPython/core/formatters.py\u001b[0m in \u001b[0;36m__call__\u001b[0;34m(self, obj)\u001b[0m\n\u001b[1;32m    339\u001b[0m                 \u001b[0;32mpass\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    340\u001b[0m             \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 341\u001b[0;31m                 \u001b[0;32mreturn\u001b[0m \u001b[0mprinter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mobj\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    342\u001b[0m             \u001b[0;31m# Finally look for special method names\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    343\u001b[0m             \u001b[0mmethod\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_real_method\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mobj\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mprint_method\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/IPython/core/pylabtools.py\u001b[0m in \u001b[0;36m<lambda>\u001b[0;34m(fig)\u001b[0m\n\u001b[1;32m    246\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    247\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0;34m'png'\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mformats\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 248\u001b[0;31m         \u001b[0mpng_formatter\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfor_type\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mFigure\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;32mlambda\u001b[0m \u001b[0mfig\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mprint_figure\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfig\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'png'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    249\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0;34m'retina'\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mformats\u001b[0m \u001b[0;32mor\u001b[0m \u001b[0;34m'png2x'\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mformats\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    250\u001b[0m         \u001b[0mpng_formatter\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfor_type\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mFigure\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;32mlambda\u001b[0m \u001b[0mfig\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mretina_figure\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfig\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/IPython/core/pylabtools.py\u001b[0m in \u001b[0;36mprint_figure\u001b[0;34m(fig, fmt, bbox_inches, **kwargs)\u001b[0m\n\u001b[1;32m    130\u001b[0m         \u001b[0mFigureCanvasBase\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfig\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    131\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 132\u001b[0;31m     \u001b[0mfig\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcanvas\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mprint_figure\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mbytes_io\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkw\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    133\u001b[0m     \u001b[0mdata\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mbytes_io\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mgetvalue\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    134\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0mfmt\u001b[0m \u001b[0;34m==\u001b[0m \u001b[0;34m'svg'\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/backend_bases.py\u001b[0m in \u001b[0;36mprint_figure\u001b[0;34m(self, filename, dpi, facecolor, edgecolor, orientation, format, bbox_inches, pad_inches, bbox_extra_artists, backend, **kwargs)\u001b[0m\n\u001b[1;32m   2215\u001b[0m                     \u001b[0morientation\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0morientation\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2216\u001b[0m                     \u001b[0mbbox_inches_restore\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0m_bbox_inches_restore\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2217\u001b[0;31m                     **kwargs)\n\u001b[0m\u001b[1;32m   2218\u001b[0m             \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2219\u001b[0m                 \u001b[0;32mif\u001b[0m \u001b[0mbbox_inches\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0mrestore_bbox\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/backend_bases.py\u001b[0m in \u001b[0;36mwrapper\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m   1637\u001b[0m             \u001b[0mkwargs\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mpop\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0marg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1638\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1639\u001b[0;31m         \u001b[0;32mreturn\u001b[0m \u001b[0mfunc\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1640\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1641\u001b[0m     \u001b[0;32mreturn\u001b[0m \u001b[0mwrapper\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/backends/backend_agg.py\u001b[0m in \u001b[0;36mprint_png\u001b[0;34m(self, filename_or_obj, metadata, pil_kwargs, *args)\u001b[0m\n\u001b[1;32m    507\u001b[0m             \u001b[0;34m*\u001b[0m\u001b[0mmetadata\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mincluding\u001b[0m \u001b[0mthe\u001b[0m \u001b[0mdefault\u001b[0m \u001b[0;34m'Software'\u001b[0m \u001b[0mkey\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    508\u001b[0m         \"\"\"\n\u001b[0;32m--> 509\u001b[0;31m         \u001b[0mFigureCanvasAgg\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    510\u001b[0m         mpl.image.imsave(\n\u001b[1;32m    511\u001b[0m             \u001b[0mfilename_or_obj\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbuffer_rgba\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mformat\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m\"png\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0morigin\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m\"upper\"\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/backends/backend_agg.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    405\u001b[0m              (self.toolbar._wait_cursor_for_draw_cm() if self.toolbar\n\u001b[1;32m    406\u001b[0m               else nullcontext()):\n\u001b[0;32m--> 407\u001b[0;31m             \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfigure\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    408\u001b[0m             \u001b[0;31m# A GUI class may be need to update a window using this draw, so\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    409\u001b[0m             \u001b[0;31m# don't forget to call the superclass.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/artist.py\u001b[0m in \u001b[0;36mdraw_wrapper\u001b[0;34m(artist, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m     39\u001b[0m                 \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstart_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     40\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 41\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0martist\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     42\u001b[0m         \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     43\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0martist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_agg_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/figure.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self, renderer)\u001b[0m\n\u001b[1;32m   1862\u001b[0m             \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mpatch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1863\u001b[0m             mimage._draw_list_compositing_images(\n\u001b[0;32m-> 1864\u001b[0;31m                 renderer, self, artists, self.suppressComposite)\n\u001b[0m\u001b[1;32m   1865\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1866\u001b[0m             \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mclose_group\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'figure'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/image.py\u001b[0m in \u001b[0;36m_draw_list_compositing_images\u001b[0;34m(renderer, parent, artists, suppress_composite)\u001b[0m\n\u001b[1;32m    129\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0mnot_composite\u001b[0m \u001b[0;32mor\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mhas_images\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    130\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0ma\u001b[0m \u001b[0;32min\u001b[0m \u001b[0martists\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 131\u001b[0;31m             \u001b[0ma\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    132\u001b[0m     \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    133\u001b[0m         \u001b[0;31m# Composite any adjacent images together\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/artist.py\u001b[0m in \u001b[0;36mdraw_wrapper\u001b[0;34m(artist, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m     39\u001b[0m                 \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstart_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     40\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 41\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0martist\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     42\u001b[0m         \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     43\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0martist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_agg_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/cbook/deprecation.py\u001b[0m in \u001b[0;36mwrapper\u001b[0;34m(*inner_args, **inner_kwargs)\u001b[0m\n\u001b[1;32m    409\u001b[0m                          \u001b[0;32melse\u001b[0m \u001b[0mdeprecation_addendum\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    410\u001b[0m                 **kwargs)\n\u001b[0;32m--> 411\u001b[0;31m         \u001b[0;32mreturn\u001b[0m \u001b[0mfunc\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0minner_args\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0minner_kwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    412\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    413\u001b[0m     \u001b[0;32mreturn\u001b[0m \u001b[0mwrapper\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/axes/_base.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self, renderer, inframe)\u001b[0m\n\u001b[1;32m   2745\u001b[0m             \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstop_rasterizing\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2746\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2747\u001b[0;31m         \u001b[0mmimage\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_draw_list_compositing_images\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0martists\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2748\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2749\u001b[0m         \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mclose_group\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'axes'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/image.py\u001b[0m in \u001b[0;36m_draw_list_compositing_images\u001b[0;34m(renderer, parent, artists, suppress_composite)\u001b[0m\n\u001b[1;32m    129\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0mnot_composite\u001b[0m \u001b[0;32mor\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mhas_images\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    130\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0ma\u001b[0m \u001b[0;32min\u001b[0m \u001b[0martists\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 131\u001b[0;31m             \u001b[0ma\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    132\u001b[0m     \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    133\u001b[0m         \u001b[0;31m# Composite any adjacent images together\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/artist.py\u001b[0m in \u001b[0;36mdraw_wrapper\u001b[0;34m(artist, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m     39\u001b[0m                 \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstart_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     40\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 41\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0martist\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     42\u001b[0m         \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     43\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0martist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_agg_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/axis.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1174\u001b[0m         \u001b[0;31m# the actual bbox\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1175\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1176\u001b[0;31m         \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_update_label_position\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1177\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1178\u001b[0m         \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mlabel\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/axis.py\u001b[0m in \u001b[0;36m_update_label_position\u001b[0;34m(self, renderer)\u001b[0m\n\u001b[1;32m   2056\u001b[0m         \u001b[0;31m# get bounding boxes for this axis and any siblings\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2057\u001b[0m         \u001b[0;31m# that have been set by `fig.align_xlabels()`\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2058\u001b[0;31m         \u001b[0mbboxes\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mbboxes2\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_get_tick_boxes_siblings\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2059\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2060\u001b[0m         \u001b[0mx\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mlabel\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_position\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/axis.py\u001b[0m in \u001b[0;36m_get_tick_boxes_siblings\u001b[0;34m(self, renderer)\u001b[0m\n\u001b[1;32m   2041\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0mnn\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0maxx\u001b[0m \u001b[0;32min\u001b[0m \u001b[0menumerate\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mgrp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_siblings\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0maxes\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2042\u001b[0m             \u001b[0mticks_to_draw\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0maxx\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mxaxis\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_update_ticks\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2043\u001b[0;31m             \u001b[0mtlb\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtlb2\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0maxx\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mxaxis\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_get_tick_bboxes\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mticks_to_draw\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2044\u001b[0m             \u001b[0mbboxes\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mextend\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtlb\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2045\u001b[0m             \u001b[0mbboxes2\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mextend\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtlb2\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/axis.py\u001b[0m in \u001b[0;36m_get_tick_bboxes\u001b[0;34m(self, ticks, renderer)\u001b[0m\n\u001b[1;32m   1090\u001b[0m         \u001b[0;34m\"\"\"Return lists of bboxes for ticks' label1's and label2's.\"\"\"\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1091\u001b[0m         return ([tick.label1.get_window_extent(renderer)\n\u001b[0;32m-> 1092\u001b[0;31m                  for tick in ticks if tick.label1.get_visible()],\n\u001b[0m\u001b[1;32m   1093\u001b[0m                 [tick.label2.get_window_extent(renderer)\n\u001b[1;32m   1094\u001b[0m                  for tick in ticks if tick.label2.get_visible()])\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/axis.py\u001b[0m in \u001b[0;36m<listcomp>\u001b[0;34m(.0)\u001b[0m\n\u001b[1;32m   1090\u001b[0m         \u001b[0;34m\"\"\"Return lists of bboxes for ticks' label1's and label2's.\"\"\"\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1091\u001b[0m         return ([tick.label1.get_window_extent(renderer)\n\u001b[0;32m-> 1092\u001b[0;31m                  for tick in ticks if tick.label1.get_visible()],\n\u001b[0m\u001b[1;32m   1093\u001b[0m                 [tick.label2.get_window_extent(renderer)\n\u001b[1;32m   1094\u001b[0m                  for tick in ticks if tick.label2.get_visible()])\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/text.py\u001b[0m in \u001b[0;36mget_window_extent\u001b[0;34m(self, renderer, dpi)\u001b[0m\n\u001b[1;32m    900\u001b[0m             \u001b[0mbbox\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0minfo\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdescent\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_get_layout\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_renderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    901\u001b[0m             \u001b[0mx\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_unitless_position\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 902\u001b[0;31m             \u001b[0mx\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_transform\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtransform\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    903\u001b[0m             \u001b[0mbbox\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mbbox\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtranslated\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    904\u001b[0m             \u001b[0;32mreturn\u001b[0m \u001b[0mbbox\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/transforms.py\u001b[0m in \u001b[0;36mtransform\u001b[0;34m(self, values)\u001b[0m\n\u001b[1;32m   1420\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1421\u001b[0m         \u001b[0;31m# Transform the values\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1422\u001b[0;31m         \u001b[0mres\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtransform_affine\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtransform_non_affine\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mvalues\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1423\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1424\u001b[0m         \u001b[0;31m# Convert the result back to the shape of the input values.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/transforms.py\u001b[0m in \u001b[0;36mtransform_affine\u001b[0;34m(self, points)\u001b[0m\n\u001b[1;32m   2342\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mtransform_affine\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mpoints\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2343\u001b[0m         \u001b[0;31m# docstring inherited\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2344\u001b[0;31m         \u001b[0;32mreturn\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_affine\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtransform\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mpoints\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2345\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2346\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mtransform_non_affine\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mpoints\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/anaconda3/envs/sr/lib/python3.7/site-packages/matplotlib/transforms.py\u001b[0m in \u001b[0;36mget_affine\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m   2370\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2371\u001b[0m             return Affine2D(np.dot(self._b.get_affine().get_matrix(),\n\u001b[0;32m-> 2372\u001b[0;31m                                    self._a.get_affine().get_matrix()))\n\u001b[0m\u001b[1;32m   2373\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2374\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0minverted\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m<__array_function__ internals>\u001b[0m in \u001b[0;36mdot\u001b[0;34m(*args, **kwargs)\u001b[0m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "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:  251\n",
      "KLD:  tensor(355.8264)\n",
      "CoB:  tensor(18.1069, device='cuda:1')\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "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_mu2.png')"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAagAAAEYCAYAAAAJeGK1AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8vihELAAAACXBIWXMAAAsTAAALEwEAmpwYAAA8cElEQVR4nO3deXiU5dX48e8h7Jss4saiWMCyGAJEFkWNoMPSilot4IpQS7Xat8XX11fUt4BFU391aW0VFwStK2hFQUGiWKhaEAhiZJEKAhIERFDACCGQ8/vjfibMJJM9k3lm5nyua66Z555nZu6ZLGfu7dyiqhhjjDF+UyfWFTDGGGMisQBljDHGlyxAGWOM8SULUMYYY3zJApQxxhhfsgBljDHGl2IaoERkhoh8LSJrQspaicg7IvK5d93SKxcReURENopIjoj0DnnMGO/8z0VkTCzeizHGmJoV6xbUM8DQYmV3AItUtTOwyDsGGAZ09i7jgWngAhowCegH9AUmBYOaMcaY+BXTAKWq/wL2Fiu+BHjWu/0scGlI+d/VWQa0EJGTgSHAO6q6V1W/Bd6hZNAzxhgTZ+rGugIRnKiqO7zbO4ETvdttgW0h5+V6ZaWVlyAi43GtL5o0adLnxz/+cQ1W2xhjTFVkZ2d/o6ptipf7MUAVUVUVkRrLxaSqTwJPAqSnp+vKlStr6qmNMcZUkYhsjVQe6zGoSHZ5XXd411975duB9iHntfPKSis3xhgTx/wYoOYCwZl4Y4A3Qsqv82bz9Qf2eV2BC4GAiLT0JkcEvDJjjDFxLKZdfCLyEpABHC8iubjZeH8EZovIL4CtwEjv9PnAcGAj8AMwFkBV94rIH4AV3nn3qGrxiRfGGGPijCTrdhs2BmVqQ0FBAbm5uRw6dCjWVTEm5ho2bEi7du2oV69eWLmIZKtqevHzfT1Jwph4l5ubS7NmzTjttNMQkVhXx5iYUVX27NlDbm4uHTt2rNBj/DgGZUzCOHToEK1bt7bgZJKeiNC6detK9SZYgDImyiw4GeNU9m/BApQxxhhfsgBlTALbs2cPaWlppKWlcdJJJ9G2bdui48OHD9fIa2RkZFCbE46WLVvGL3/5yyo99vrrr+fVV18FYO/evfTq1YuZM2eyZcsWevToUWN1PHToEH379qVnz550796dSZMmhdWhY8eORT+H1atXA7B48WKOO+64ovJ77rkn4nMvWrSI3r17k5aWxsCBA9m4cSMAjz/+OGeeeWZR+bp162rs/cSKTZIwJoG1bt266B/g5MmTadq0KbfddlvR/UeOHKFu3fj6N7BgwQKGDq1eus19+/YxZMgQxo8fz9ixY9myZUvNVM7ToEED3nvvPZo2bUpBQQEDBw5k2LBh9O/fH4A//elPXHHFFSUed+655/Lmm2+W+dw33XQTb7zxBl27duWxxx5j6tSpPPPMM1x11VXceOONAMydO5dbb72Vt99+u0bfV22zFpQxSeb666/nxhtvpF+/ftx+++1MnjyZBx54oOj+Hj16FP3Dfv755+nbty9paWn86le/4ujRoxV6jb1793LppZeSmppK//79ycnJAWD37t1cdNFFdO/enRtuuIFTTz2Vb775psTjmzZtyoQJE+jevTuDBw9m9+7dRfctWrSICy+8kIMHDzJ69Gi6du3KZZddRr9+/SrUkvv+++8ZNmwYV111FTfddFOF3k9liQhNmzYF3FKDgoKCGhuLFBH2798PuEB7yimnANC8efOic/Ly8hJi7NMClDE+k5UFd97prqMlNzeXf//73zz00EOlnrN+/XpmzZrFhx9+yOrVq0lJSeGFF16o0PNPmjSJXr16kZOTw3333cd1110HwJQpUxg0aBBr167liiuu4Msvv4z4+Ly8PNLT01m7di3nn38+U6ZMAeCbb76hXr16HHfccUybNo3GjRuzfv16pkyZQnZ2doXqduuttzJw4EAmTJhQofODXnjhhaLut9BLpJYQwNGjR0lLS+OEE07goosuol+/fkX33XXXXaSmpjJhwgTy8/OLypcuXUrPnj0ZNmwYa9eujfi806dPZ/jw4bRr147nnnuOO+64o+i+Rx99lB/96EfcfvvtPPLII5V6f35kAcoYH8nKgmuugUcfddfRClI///nPSUlJKfOcRYsWkZ2dzVlnnUVaWhqLFi3iiy++qNDzf/DBB1x77bUADBo0iD179rB//34++OADRo8eDcDQoUNp2TLy1m116tRh1KhRAFxzzTV88MEHAGRlZREIBAD417/+xTXXXANAamoqqampFarboEGDeOONN/j666/LPznE1VdfzerVq0tcgmNaxaWkpLB69Wpyc3NZvnw5a9a4fVkzMzP57LPPWLFiBXv37uX+++8HoHfv3mzdupVPPvmE3/zmN1x66aURn/fhhx9m/vz55ObmMnbsWG699dai+26++WY2bdrE/fffz9SpUyv1/vzIApQxPrJ4MeTnQ5Mm7nrx4ui8TpMmTYpu161bl8LCwqLj4DoVVWXMmDFF/4g3bNjA5MmTo1OhcgS7q2pi/Gn06NHceOONDB8+nAMHDlT4cZVtQQW1aNGCCy64oGg86OSTT0ZEaNCgAWPHjmX58uWA66ILdgsOHz6cgoKCEt2fu3fv5pNPPilqjY0aNYp///vfEd/j66+/XuH35lcWoIzxkYwMaNAA8vLcdUZG9F/ztNNOY9WqVQCsWrWKzZs3AzB48GBeffXVopbG3r172bo14q4IJZx77rlF3YGLFy/m+OOPp3nz5pxzzjnMnj0bcK2hb7/9NuLjCwsLi1omL774IgMHDkRVycnJIS0tDYDzzjuPF198EYA1a9YUjXMBXHfddUX/+COZMGECgwcP5mc/+1mFZzNWpgW1e/duvvvuOwAOHjzIO++8Q3D/uR073HZ3qsrrr79eNHtw586dBFPPLV++nMLCQlq3bh32vC1btmTfvn385z//AeCdd96ha9euAHz++edF57311lt07ty5Qu/Lz+Jr+o4xCS4QgOefdy2njAx3HG2XX345f//73+nevTv9+vWjS5cuAHTr1o2pU6cSCAQoLCykXr16PProo5x66qklnuMnP/lJUX61AQMG8MQTTzBu3DhSU1Np3Lgxzz7rNsmeNGkSV155Jc899xwDBgzgpJNOolmzZiWer0mTJixfvpypU6dywgknMGvWLLKzs+nVq1dRa+qmm25i7NixdO3ala5du9KnT5+ix+fk5BRNHijN/fffz9ixY7n22mvJzMxkw4YNtGvXruj+hx9+mJ///OeV/DSdHTt2MGbMGI4ePUphYSEjR47kpz/9KeAC3e7du1FV0tLSePzxxwF49dVXmTZtGnXr1qVRo0a8/PLLRe91+PDhTJ8+nVNOOYWnnnqKyy+/nDp16tCyZUtmzJgBwN/+9jfeffdd6tWrR8uWLYs+83hmyWKNiaL169cXfcM1kJ+fT0pKCnXr1mXp0qXcdNNNRdPgQzVt2pTvv/8+rGzq1Kl06tSpaAyruIyMDB544AG6dOnCL37xC1555ZVovAVTTZH+JixZrDEm5r788ktGjhxJYWEh9evX56mnnqrwY+++++4Knde8eXMLTgnCApQxptZ07tyZjz/+uNzzireeKmJxtGaUmJixSRLGGGN8yQKUMcYYX7IAZYwxxpcsQBljjPElC1DGJLh7772X7t27k5qaSlpaGh999BEAN9xwQ5W3ZKjK9hQpKSmkpaXRo0cPLr744qKFrKVZvXo18+fPLzqeO3cuf/zjHyv8es888wy33HJLpepYXX369AnLrVdRixcvLlonBW7G4tChQ8nPz6/x7UxeeeUVunfvTp06dcKet6CggDFjxnDmmWfStWtXMjMzIz5+8+bN9OvXj06dOjFq1Kiihc5bt25l8ODBpKamkpGRQW5ubrXr6ssAJSJniMjqkMt+EfmdiEwWke0h5cNDHjNRRDaKyAYRGRLL+hvjF0uXLuXNN99k1apV5OTk8O6779K+fXvAJR3t1q1brdWlUaNGrF69mjVr1tCqVSseffTRMs8vHqBGjBgRlhjVbzZv3kzbtm1p0KBBtZ5n6tSpfPjhh8yZM6fazxVJjx49eO211zjvvPPCyl955RXy8/P59NNPyc7O5oknnoi4Dcn//u//MmHCBDZu3EjLli15+umnAbjtttu47rrryMnJ4fe//z0TJ06sdl19GaBUdYOqpqlqGtAH+AGY4939cPA+VZ0PICLdgNFAd2Ao8JiIlJ0J05gksGPHDo4//viif3THH398UYaF0G/mTZs25a677qJnz57079+fXbt2AbBp0yb69+/PmWeeyd13312UKy7U0aNH+Z//+R/OOussUlNTeeKJJ8qt14ABA9i+fTvg0voMGDCAXr16cfbZZ7NhwwYOHz7M73//e2bNmkVaWhqzZs0KaxFt2bKFQYMGkZqayuDBg0vNih7JQw89RI8ePejRowd//vOfi8r/8Ic/cMYZZzBw4ECuvPLKsC1IgoJblaSnp9OlS5ewvZvefvvtojyBM2fOpEuXLvTt25df/vKXFW7JPfjggyxYsIB58+bRqFGjCr+nyujatStnnHFGiXIRIS8vjyNHjnDw4EHq168ftoUHuPRM7733XlH+wTFjxhTl/Fu3bh2DBg0C4IILLuCNN96odl19GaCKGQxsUtWykoBdArysqvmquhnYCPStldoZU9NqcL+NQCDAtm3b6NKlC7/+9a9ZsmRJxPPy8vLo378/n3zyCeedd17RAtrf/va3/Pa3v+XTTz8NSwMU6umnn+a4445jxYoVrFixgqeeeqoo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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
}