{
 "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 = 500\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.1, scale_q=0.2, 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": "iVBORw0KGgoAAAANSUhEUgAABDAAAAEYCAYAAACqUwbqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8QVMy6AAAACXBIWXMAAAsTAAALEwEAmpwYAAAxCUlEQVR4nO3de7RdZX3v//enCRLuAgYEgiYi6gkoKOEmFhE8IVqPyBGUnFLjpb8cPXhBqy3o6QHPELXaWouiR7wBBbmUQqFU5KaBAUVCQJSbURCUCELAFhDllnx/f6wZ2En2TnYua625136/xphjzfnM2/dZ7Dzz4bvmfGaqCkmSJEmSpDb7o34HIEmSJEmStDomMCRJkiRJUuuZwJAkSZIkSa1nAkOSJEmSJLWeCQxJkiRJktR6JjAkSZIkSVLrdS2BkeRbSR5IcsuQsq2SXJbk583nlt06vySNV0nuTnJzkpuSLGjKRmx/kxyb5I4kC5McPKR8j+Y4dyQ5MUn6UR9Jkn1rSYLu3oFxCjBrhbJjgCuqamfgimZZkrT+va6qdq+qGc3ysO1vkunAEcAudNrsrySZ0OzzVWAusHMzrdimS5J65xTsW0sa57qWwKiqq4DfrlB8CHBqM38q8JZunV+StJyR2t9DgLOq6omqugu4A9gryXbA5lV1bVUVcBq22ZLUN/atJQkm9vh821bVfQBVdV+SbUbaMMlcOr/8sckmm+zxspe9rEchShovbrjhhgeranK/4+iCAi5NUsDXqupkRm5/dwB+OGTfRU3ZU838iuXLsa2W1G0D3FavD/atJbVCr9rqXicwRq3pcJ8MMGPGjFqwYEGfI5I0aJL8st8xdMl+VXVv05G9LMlPV7HtcONa1CrKly+wrZbUZQPcVveU7bWkbupVW93rt5Dc39yWTPP5QI/PL0kDr6rubT4fAM4H9mLk9ncRsOOQ3acA9zblU4YplyS1h31rSeNKrxMYFwJzmvk5wAU9Pr8kDbQkmyTZbNk8MBO4hZHb3wuBI5JsmGQancE65ze3JD+aZJ/m7SPvwDZbktrGvrWkcaVrj5AkORM4AHhekkXAccBngXOSvAf4FXB4t84vSePUtsD5zRtPJwLfqarvJbmeYdrfqro1yTnAbcDTwFFVtaQ51vvojHq/EXBxM0mS+sC+tSR1MYFRVbNHWHVQt84pSeNdVf0C2G2Y8ocYof2tqhOAE4YpXwDsur5jlMaTp556ikWLFvH444/3O5TWmzRpElOmTGGDDTbodyitZN9aklo8iKckSdJYt2jRIjbbbDOmTp1Kc2eUhlFVPPTQQyxatIhp06b1OxxJUkv1egwMSZKkcePxxx9n6623NnmxGknYeuutvVNFkrRKJjAkSZK6yOTF6Pg9SZJWxwSGJEmSJElqPRMYkiRJA+qhhx5i9913Z/fdd+f5z38+O+ywwzPLTz75ZNfPf9999zFz5syun0eSND44iKckSdKA2nrrrbnpppsAOP7449l000356Ec/+sz6p59+mokTu9cd/N73vsfBBx/cteNLksYXExiSJEnjyDvf+U622morfvSjH/GqV72KzTbbbLnExq677spFF13E1KlTOf300znxxBN58skn2XvvvfnKV77ChAkTljve1KlTefvb384PfvADAL7zne/w4he/GOgkMI477jiqig984AN8//vfZ9q0aVQV7373uznssMN6W3lJ0pjmIySSJEktcuml8PGPdz675Wc/+xmXX345f/d3fzfiNrfffjtnn30211xzDTfddBMTJkzgjDPOGHbbzTffnPnz5/P+97+fo48+GoAlS5awcOFCpk+fzvnnn8/ChQu5+eab+frXv86///u/d6NakqQB5x0YkiRJLXHppXDkkfDEE/CNb8Dpp0M3hpA4/PDDV7qTYkVXXHEFN9xwA3vuuScAf/jDH9hmm22G3Xb27NnPfH74wx8G4LrrrmPvvfcG4KqrrmL27NlMmDCB7bffngMPPHB9VUWSNI6YwJAkSWqJefM6yYtNNoHHHussdyOBsckmmzwzP3HiRJYuXfrM8uOPPw5AVTFnzhw+85nPrPZ4Q1+Bumz+4osvZtasWcNuI0nS2vAREkmSpJY44ADYcMNO8mLDDTvL3TZ16lRuvPFGAG688UbuuusuAA466CDOPfdcHnjgAQB++9vf8stf/nLYY5x99tnPfO67775A5w6Ogw46CID999+fs846iyVLlnDfffc9M16GJElrwjswJEmSWmLmzM5jI/PmdZIXvXgD6Vvf+lZOO+00dt99d/bcc09e8pKXADB9+nQ+9alPMXPmTJYuXcoGG2zASSedxAtf+MKVjvHEE0+w9957s3TpUs4880wWL17MpEmT2HzzzQE49NBD+f73v8/LX/5yXvKSl/Da1762+xWTJA0cExiSJEktMnNmdxIXxx9//LDlG220EZeOMGLo29/+dt7+9rev9thHHXUUxx133DPLp59+OjOHVCIJX/7yl59Zfuc73zm6oCVJGsIEhiRJktarI488st8hSJIGkAkMSZIkrbW77757jfc55ZRT1nsckqTB5yCekiRJkiSp9UxgSJIkSZKk1jOBIUmSJEmSWs8EhiRJkiRJaj0TGJIkSQPsN7/5DUcccQQ77bQT06dP541vfCM/+9nPRtz+7rvvZqONNmL33Xdnt91249WvfjULFy5c4/N+5jOf4YwzzliX0CVJWo4JDEmSpAFVVRx66KEccMAB3Hnnndx22218+tOf5v7771/lfjvttBM33XQTP/7xj5kzZw6f/vSn1/jcl156KTNnzlzb0CVJWokJDEmSpAH1gx/8gA022ID3vve9z5Ttvvvu/PEf/zFVxcc+9jF23XVXXv7yl3P22WcPe4xHHnmELbfccqXyefPmsf/++3PooYcyffp03vve97J06dJn9nnyySeZPHkyd911F/vuuy977rknf/3Xf82mm27ancpKkgbexH4HIEmSpCEuvRTmzYMDDoB1vIPhlltuYY899hh23XnnnffMXRYPPvgge+65J/vvvz8Ad955J7vvvjuPPvoov//977nuuuuGPcb8+fO57bbbeOELX8isWbM477zzOOyww7j88ss56KCDAPjQhz7E+973Pt7xjndw0kknrVN9JEnjm3dgSJIktcWll8KRR8JJJ3U+L720a6e6+uqrmT17NhMmTGDbbbflta99Lddffz3w7CMkd955J1/84heZO3fusMfYa6+9eNGLXsSECROYPXs2V199NQDf+973eMMb3gDANddcw+zZswH4sz/7s67VR5I0+ExgSJIktcW8efDEE7DJJp3PefPW6XC77LILN9xww7DrqmpUx3jzm9/MVVddNey6JMMuz58/n7322mvE7SRJWhsmMCRJktrigANgww3hscc6nwccsE6HO/DAA3niiSf4+te//kzZ9ddfz5VXXsn+++/P2WefzZIlS1i8eDFXXXXVckmHZa6++mp22mmnYY8/f/587rrrLpYuXcrZZ5/Na17zGm699VZe9rKXMWHCBAD2228/zjrrLADfSiJJWicmMCRJktpi5kw4/XQ46qjO5zqOgZGE888/n8suu4yddtqJXXbZ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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": [],
   "source": [
    "NUM_EPOCHS=1500"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1080x288 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  6%|▌         | 92/1500 [05:13<1:20:03,  3.41s/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-10-b4a08f41935e>\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/torch1.8/lib/python3.8/site-packages/torch/tensor.py\u001b[0m in \u001b[0;36mbackward\u001b[0;34m(self, gradient, retain_graph, create_graph, inputs)\u001b[0m\n\u001b[1;32m    243\u001b[0m                 \u001b[0mcreate_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[1;32m    244\u001b[0m                 inputs=inputs)\n\u001b[0;32m--> 245\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[0minputs\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0minputs\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    246\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    247\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/torch1.8/lib/python3.8/site-packages/torch/autograd/__init__.py\u001b[0m in \u001b[0;36mbackward\u001b[0;34m(tensors, grad_tensors, retain_graph, create_graph, grad_variables, inputs)\u001b[0m\n\u001b[1;32m    143\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[0m\n\u001b[1;32m    144\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 145\u001b[0;31m     Variable._execution_engine.run_backward(\n\u001b[0m\u001b[1;32m    146\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[0minputs\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    147\u001b[0m         allow_unreachable=True, accumulate_grad=True)  # allow_unreachable flag\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( -50., 50. )\n",
    "                    ax2.set_ylim( -500, 200)\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": [
    "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",
    "        u_batch1 = torch.FloatTensor((500, 1)).uniform_(-5, 5).to(DEVICE)\n",
    "        u_batch2 = torch.FloatTensor((500, 1)).uniform_(-10, 10).to(DEVICE)\n",
    "        u_batch3 = torch.FloatTensor((500, 1)).uniform_(-20, 20).to(DEVICE)\n",
    "        \n",
    "    logP = model(p_batch)\n",
    "    logQ = model(q_batch)\n",
    "    logM = model(m_batch)\n",
    "    logU1 = model(u_batch1)\n",
    "    logU2 = model(u_batch2)\n",
    "    logU3 = model(u_batch3)\n",
    "\n",
    "    log_ratio_p_q_from_cob_p = logP[:, 0] - logP[:, 1]\n",
    "    kl_from_cob = torch.mean(log_ratio_p_q_from_cob_p)\n",
    "\n",
    "    log_ratio_p_q_from_cob_m = logM[:, 0] - logM[:, 1]\n",
    "    log_ratio_p_q_from_cob_q = logQ[:, 0] - logQ[:, 1]\n",
    "    \n",
    "    log_ratio_p_q_from_cob_u1 = logU1[:, 0] - logU1[:, 1]\n",
    "    log_ratio_p_q_from_cob_u2 = logU2[:, 0] - logU2[:, 1]\n",
    "    log_ratio_p_q_from_cob_u3 = logU3[:, 0] - logU3[:, 1]\n",
    "    \n",
    "    print('iteration: ',i)\n",
    "    print('True KLD: ', true_kl_p_q)\n",
    "    print('CoB from p samples: ', kl_from_cob)\n",
    "    print('CoB from q samples: ', log_ratio_p_q_from_cob_q.mean())\n",
    "    print('CoB from m samples: ', log_ratio_p_q_from_cob_m.mean())\n",
    "    print('CoB (-5, 5): ', log_ratio_p_q_from_cob_u1.mean())\n",
    "    print('CoB (-10, 10): ', log_ratio_p_q_from_cob_u2.mean())\n",
    "    print('CoB (-20, 20): ', log_ratio_p_q_from_cob_u3.mean())\n",
    "\n",
    "#     clear_output(wait=True)\n",
    "    display(fig)\n",
    "    break\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "torch1.8",
   "language": "python",
   "name": "torch1.8"
  },
  "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.8.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
