{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "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 *"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "N_DIMS = 1\n",
    "NUM_SAMPLES = 100000\n",
    "BS = 500\n",
    "NUM_EPOCHS = 800\n",
    "SEED = 10\n",
    "LR = 1e-2\n",
    "DROPOUT = 0.20\n",
    "DEVICE = 'cuda:0' if torch.cuda.is_available() else 'cpu'\n",
    "\n",
    "\n",
    "# Break by changing num datapoints, scales, means, or to 2D"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Setting seed for reproducibility\n",
    "random.seed(SEED)\n",
    "torch.manual_seed(SEED)\n",
    "np.random.seed(SEED)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "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",
    "# Define distributions\n",
    "p, q, m = get_dists_1d(mu1=-2., mu2=2., mu3=0, scale_p=0.08, scale_q=0.15, scale_m=1.0)\n",
    "\n",
    "# -5, 5, m_var=3.0\n",
    "# -10, 10, m_var=3.0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling p\n",
      "Sampling q\n",
      "Cauchy(loc: 0.0, scale: 1.0)\n",
      "Sampling m\n",
      "torch.Size([100000])\n",
      "torch.Size([100000])\n",
      "torch.Size([100000])\n",
      "Sampling p\n",
      "Sampling q\n",
      "Cauchy(loc: 0.0, scale: 1.0)\n",
      "Sampling m\n",
      "torch.Size([100000])\n",
      "torch.Size([100000])\n",
      "torch.Size([100000])\n"
     ]
    }
   ],
   "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 "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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lFMalq+11lpc32TE0EMPPZRDDz20x20ff/zxnH766cvnL7zwQibXvYmI4Hvf+97y+SOOOKJ3QUuSVMcEhiRJkgbU4YcfXnYIkqQOZAJDkiRJfbZgwYLVfs1555034HFIkjqfg3hKkiRJkqTKM4EhSZIkSZIqzwSGJEmSJEmqPBMYkiRJkiSp8kxgSJIkdbAnn3ySww47jK222opddtmFAw44gD/84Q9N11+wYAFrrbUW48ePZ6edduI973kP8+bNW+16TzvtNC666KL+hC5J0gpMYEiSJHWozOTggw9m4sSJPPzww8ydO5evf/3r/PnPf17l67baaivuuusu7r77bqZNm8a///u/r3bd1113HZMnT+5r6JIkrcQEhiRJUoe6+eabGTFiBEcdddTysp122on3vve9ZCbHH38822+/PTvssAMzZ85suI3nnnuODTfccKXyWbNmsffee3PggQeyzTbbcNRRR7Fs2bLlr3n11VcZNWoUjz76KHvssQc77LADX/7yl1lnnXVa82YlSR1veNkBSJIkqc7118OsWTBxIvTzDIb77ruPXXbZpeGyyy+/fPlZFn/5y1/Ydddd2XvvvQF4+OGHGT9+PH/729948cUXuf322xtuY/bs2TzwwAO87W1vY8qUKVx++eUccsgh3Hjjjeyzzz4AHHvssXzqU5/i4x//OGeffXa/3o8kaWjzDAxJkqSquP56OPxwOPvs2vP117esqltvvZWpU6cybNgwNtlkE/7hH/6BO+64A3jjEpKHH36Ys846i+nTpzfcxm677cbb3/52hg0bxtSpU7n11lsBuPbaa9l///0BuO2225g6dSoAH/vYx1r2fiRJnc8EhiRJUlXMmgWvvAJrr117njWrX5vbbrvtmDt3br+28cEPfpBbbrml4bKIaDg/e/Zsdtttt6brSZLUFyYwJEmSqmLiRFhzTXjhhdrzxIn92tykSZN45ZVXmDFjxvKye+65h1/96le8973vZebMmSxdupTFixdzyy23rJB06HLrrbey1VZbNdz+7NmzefTRR1m2bBkzZ85kr7324v7772fbbbdl2LBhAOy5555ccsklAN6VRJLULyYwJEmSqmLyZLjwQjj66NpzP8fAiAiuuOIKbrzxRrbaaiu22247TjrpJDbddFMOPvhgdtxxR3baaScmTZrEN77xDTbddFPgjTEwdtpp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\n",
      "text/plain": [
       "<Figure size 1080x288 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "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 = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABB0AAAEYCAYAAAAQ8I6KAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/d3fzzAAAACXBIWXMAAAsTAAALEwEAmpwYAABRgUlEQVR4nO3deZwU5bX/8c/pnoVtWAVBFkHEfSGCuJtxyYgaNSZelxuNSnKJRu9NTDQRkvxibozE7Bvxxqu4ROMavVETZdxG4y4QRESJICggCIIwLMOs5/dH1UDPMPv0Ut39fb9e/aqup6urT8301FSffp7zmLsjIiIiIiIiIpJssUwHICIiIiIiIiK5SUkHEREREREREUkJJR1EREREREREJCWUdBARERERERGRlFDSQURERERERERSQkkHEREREREREUmJlCUdzGyWma01s4UJbQPN7EkzezdcDkjV64uI5KPOnHst8FszW2JmC8zssITnXBxu/66ZXZyJYxERkZ10bS0i2SqVPR1uByY3a7sWeNrdxwFPh+siIpI8t9Pxc++pwLjwNhW4CYKLWOAHwBHAJOAHupAVEcm429G1tYhkoZQlHdz9eWBDs+azgDvC+3cAn0vV64uI5KNOnnvPAu70wCtAfzMbBpwCPOnuG9z9E+BJdr3QFRGRNNK1tYhkq4I0v97u7r46vL8G2L21Dc1sKsE3b/Tu3XvCfvvtl4bwJFMWrtqEt/LYwcP7pTUWyR9z58792N0HZzqONGjt3DscWJGw3cqwrbX2XehcLSKplkfn6q7QtbU0sXDVJnbrU8zQfj0yHYrkodbO1+lOOuzg7m5mrX3OxN1vBm4GmDhxos+ZMydtsUn67fPdx6mpb2jxsTk/OT3N0Ui+MLP3Mx1DurV37u3C/nSuFpGUysdzdVfo2loA9vv+43zpqNFMP23/TIcieai183W6Z6/4KOy6S7hcm+bXFxHJR62de1cBIxO2GxG2tdYuIiLRomtraaIgFqO+IWnfLYgkRbqTDo8AjVXQLwb+mubXl4jyVgdXiEgStHbufQT4UjiLxZHAprCb7mygzMwGhAUky8I2ERGJFl1bSxMxQ0kHiZyUDa8ws3uAUmA3M1tJUAn9J8D9ZvZl4H3g3FS9vohIPurkuffvwGnAEmAbcCmAu28wsx8Br4fb/be7Ny9eJiIiaaRra+mIgrh6Okj0pCzp4O4XtPLQSal6TRGRfNeZc6+7O3BFK/uZBcxKYmgieam2tpaVK1eyffv2TIcSeT169GDEiBEUFhZmOpRI0rW1dETMjDolHSRiMlZIUkRERCTXrVy5kpKSEkaPHo2ZZTqcyHJ31q9fz8qVKxkzZkymwxHJWgUxo0FJB4mYdNd0EGmR69woIiI5aPv27QwaNEgJh3aYGYMGDVKPEJFuisfU00GiR0kHERERkRRSwqFj9HMS6b54zGjQt3kSMUo6iIiIiIiI5IB4zFRIUiJHSQeJBJ0aRUREkm/9+vWMHz+e8ePHM3ToUIYPH75jvaamJuWvv3r1asrKylL+OiIS0JSZEkUqJCkiIiKSowYNGsT8+fMBuO666+jTpw9XX331jsfr6uooKEjd5eATTzzBKaeckrL9i0hTBTFNmSnRo54OIiIiInnkkksu4bLLLuOII47g29/+Ntdddx0///nPdzx+0EEHsXz5cgDuuusuJk2axPjx4/nqV79KfX39LvsbPXo03/72tzn44IOZNGkSS5Ys2fHYE088wamnnoq7c+WVV7Lvvvty8sknc9ppp/Hggw+m/FhF8k1MhSQlgpR0EBEREYmQ8nKYPj1YpsrKlSt56aWX+OUvf9nqNm+//Tb33XcfL774IvPnzycej3P33Xe3uG2/fv148803ufLKK/nGN74BQH19PYsXL+aAAw7g4YcfZvHixSxatIg777yTl156KRWHJZL3ClRIUiJIwyskElwnRxEREcrL4cILoboabrkF7roLUlES4d/+7d+Ix+NtbvP0008zd+5cDj/8cACqqqoYMmRIi9tecMEFO5ZXXXUVAK+++ipHHHEEAM8//zwXXHAB8XicPfbYgxNPPDFZhyIiCdTTQaJISQcRERGRiKioCBIOvXvD1q3BeiqSDr17995xv6CggIaGhh3r27dvB4IvBC6++GJmzJjR7v4Sp7tsvP/4448zefLkZIUsIh1QEDMalHSQiNHwChEREZGIKC2F4uIg4VBcHKyn2ujRo5k3bx4A8+bNY9myZQCcdNJJPPjgg6xduxaADRs28P7777e4j/vuu2/H8qijjgKCnhInn3wyAMcffzz33Xcf9fX1rF69mmeffTalxySSr+Jm1CUkEUWiQD0dRERERCKirCwYUlFRESQc0jHb5Be+8AXuvPNODjzwQI444gj22Wcf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\n",
      "text/plain": [
       "<Figure size 1080x288 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|▏         | 14/800 [00:54<50:35,  3.86s/it]\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-9-25dcf4c74e33>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m     31\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     32\u001b[0m         \u001b[0mloss\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mloss_crit\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mlogP\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mp_label\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0mloss_crit\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mlogQ\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mq_label\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0mloss_crit\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mlogM\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mm_label\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 33\u001b[0;31m         \u001b[0mloss\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbackward\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     34\u001b[0m         \u001b[0moptim\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstep\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     35\u001b[0m         \u001b[0mloss_store\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mappend\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mloss\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mitem\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[0;32m/disk_c/han/ananconda3/envs/sr/lib/python3.7/site-packages/torch/tensor.py\u001b[0m in \u001b[0;36mbackward\u001b[0;34m(self, gradient, retain_graph, create_graph)\u001b[0m\n\u001b[1;32m    183\u001b[0m                 \u001b[0mproducts\u001b[0m\u001b[0;34m.\u001b[0m \u001b[0mDefaults\u001b[0m \u001b[0mto\u001b[0m\u001b[0;31m \u001b[0m\u001b[0;31m`\u001b[0m\u001b[0;31m`\u001b[0m\u001b[0;32mFalse\u001b[0m\u001b[0;31m`\u001b[0m\u001b[0;31m`\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    184\u001b[0m         \"\"\"\n\u001b[0;32m--> 185\u001b[0;31m         \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mautograd\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbackward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mgradient\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mretain_graph\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcreate_graph\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    186\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    187\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mregister_hook\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mhook\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/ananconda3/envs/sr/lib/python3.7/site-packages/torch/autograd/__init__.py\u001b[0m in \u001b[0;36mbackward\u001b[0;34m(tensors, grad_tensors, retain_graph, create_graph, grad_variables)\u001b[0m\n\u001b[1;32m    125\u001b[0m     Variable._execution_engine.run_backward(\n\u001b[1;32m    126\u001b[0m         \u001b[0mtensors\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mgrad_tensors\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mretain_graph\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcreate_graph\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 127\u001b[0;31m         allow_unreachable=True)  # allow_unreachable flag\n\u001b[0m\u001b[1;32m    128\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    129\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "## CONFIRM q_list_test in validation/visualization in Akash's 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",
    "        logQ = model(q_batch)\n",
    "        logM = model(m_batch)\n",
    "        \n",
    "        p_label = torch.empty(p_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(0)\n",
    "        q_label = torch.empty(q_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(1)\n",
    "        m_label = torch.empty(m_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(2)\n",
    "        \n",
    "        loss = loss_crit(logP, p_label) + loss_crit(logQ, q_label) + loss_crit(logM, m_label)\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",
    "                    logQ = model(q_batch)\n",
    "                    logM = model(m_batch)\n",
    "\n",
    "                    log_ratio_p_q_from_cob = logP[:, 0] - logP[:, 1]\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_(0)\n",
    "                    q_label = torch.empty(q_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(1)\n",
    "                    m_label = torch.empty(m_batch.shape[0], dtype=torch.long, device=DEVICE).fill_(2)\n",
    "                    \n",
    "                    test_loss = loss_crit(logP, p_label) + loss_crit(logQ, q_label) + loss_crit(logM, m_label)\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( -25., 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()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "                    print('iteration: ',i)\n",
    "                    print('KLD: ', true_kl_p_q)\n",
    "                    print('CoB: ', kl_from_cob)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "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='CoB 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/cob_mu2.png')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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
}
