{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import sys\n",
    "import torch\n",
    "import numpy as np\n",
    "import torch.nn as nn\n",
    "import torch.backends.cudnn as cudnn\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "cwd = os.getcwd()\n",
    "module_path = \"/\".join(cwd.split('/')[0:-1])\n",
    "if module_path not in sys.path:\n",
    "    sys.path.append(module_path)\n",
    "\n",
    "from sklearn import datasets\n",
    "from utils import gmm_utils\n",
    "from net.spectral_normalization.spectral_norm_fc import spectral_norm_fc\n",
    "from metrics.uncertainty_confidence import entropy\n",
    "from utils.train_utils import train_single_epoch, test_single_epoch\n",
    "from utils.ensemble_utils import ensemble_forward_pass"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "_ACTIVATIONS = {\n",
    "    'relu': nn.ReLU(),\n",
    "    'elu': nn.ELU()\n",
    "}\n",
    "\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "class FCResNet(nn.Module):\n",
    "    def __init__(\n",
    "        self,\n",
    "        input_dim,\n",
    "        features,\n",
    "        depth,\n",
    "        spectral_normalization,\n",
    "        coeff=0.95,\n",
    "        n_power_iterations=1,\n",
    "        dropout_rate=0.01,\n",
    "        activation='relu'\n",
    "    ):\n",
    "        super().__init__()\n",
    "\n",
    "        # ResFNN architecture from google paper: https://arxiv.org/abs/2006.10108\n",
    "        self.first = nn.Linear(input_dim, features)\n",
    "        self.residuals = nn.ModuleList(\n",
    "            [nn.Linear(features, features) for i in range(depth)]\n",
    "        )\n",
    "        self.dropout = nn.Dropout(dropout_rate)\n",
    "\n",
    "        if spectral_normalization:\n",
    "            self.first = spectral_norm_fc(\n",
    "                self.first, coeff=coeff, n_power_iterations=n_power_iterations\n",
    "            )\n",
    "\n",
    "            for i in range(len(self.residuals)):\n",
    "                self.residuals[i] = spectral_norm_fc(\n",
    "                    self.residuals[i],\n",
    "                    coeff=coeff,\n",
    "                    n_power_iterations=n_power_iterations,\n",
    "                )\n",
    "        self.last = nn.Linear(features, 2)\n",
    "\n",
    "        self.activation = _ACTIVATIONS.get(activation)\n",
    "        if self.activation is None:\n",
    "            raise NotImplementedError(f\"{activation} is not a valid option. Choose from ['relu, 'elu']\")\n",
    "\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = self.first(x)\n",
    "        for residual in self.residuals:\n",
    "            x = x + self.dropout(self.activation(residual(x)))\n",
    "        self.feature = x\n",
    "        x = self.last(x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "np.random.seed(0)\n",
    "torch.manual_seed(0)\n",
    "\n",
    "noise = 0.1\n",
    "batch_size = 64\n",
    "\n",
    "X_train, y_train = datasets.make_moons(n_samples=2000, noise=noise)\n",
    "X_test, y_test = datasets.make_moons(n_samples=200, noise=noise)\n",
    "\n",
    "ds_train = torch.utils.data.TensorDataset(torch.from_numpy(X_train).float(), torch.from_numpy(y_train))\n",
    "dl_train_loader = torch.utils.data.DataLoader(ds_train, batch_size=batch_size, shuffle=True, drop_last=True)\n",
    "\n",
    "dl_train_no_drop_loader = torch.utils.data.DataLoader(ds_train, batch_size=batch_size, shuffle=False, drop_last=False)\n",
    "\n",
    "ds_test = torch.utils.data.TensorDataset(torch.from_numpy(X_test).float(), torch.from_numpy(y_test))\n",
    "dl_test_loader = torch.utils.data.DataLoader(ds_test, batch_size=200, shuffle=False)\n",
    "\n",
    "input_dim = X_train.shape[1]\n",
    "num_classes = 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "FCResNet(\n",
      "  (first): Linear(in_features=2, out_features=128, bias=True)\n",
      "  (residuals): ModuleList(\n",
      "    (0): Linear(in_features=128, out_features=128, bias=True)\n",
      "    (1): Linear(in_features=128, out_features=128, bias=True)\n",
      "    (2): Linear(in_features=128, out_features=128, bias=True)\n",
      "    (3): Linear(in_features=128, out_features=128, bias=True)\n",
      "  )\n",
      "  (dropout): Dropout(p=0.0, inplace=False)\n",
      "  (last): Linear(in_features=128, out_features=2, bias=True)\n",
      "  (activation): ReLU()\n",
      ")\n"
     ]
    }
   ],
   "source": [
    "features = 128\n",
    "depth = 4\n",
    "coeff = 0.65\n",
    "n_power_iterations = 5\n",
    "\n",
    "spectral_normalization = False\n",
    "dropout_rate = 0.0\n",
    "\n",
    "n_ensemble = 3\n",
    "\n",
    "model_ensemble = [FCResNet(\n",
    "    input_dim=input_dim,\n",
    "    features=features,\n",
    "    depth=depth,\n",
    "    spectral_normalization=spectral_normalization,\n",
    "    coeff=coeff,\n",
    "    n_power_iterations=n_power_iterations,\n",
    "    dropout_rate=dropout_rate\n",
    ").to(device) for i in range(n_ensemble)]\n",
    "\n",
    "print (model_ensemble[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "loss_fn = nn.CrossEntropyLoss()\n",
    "optimizers = [\n",
    "    torch.optim.Adam(model_ensemble[i].parameters()) for i in range(n_ensemble)\n",
    "]\n",
    "num_epochs = 5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Ensemble model: 0\n",
      "Train Epoch: 0 [0/1984 (0%)]\tLoss: 1.141410\n",
      "Train Epoch: 0 [640/1984 (32%)]\tLoss: 0.224662\n",
      "Train Epoch: 0 [1280/1984 (65%)]\tLoss: 0.186688\n",
      "Train Epoch: 0 [1920/1984 (97%)]\tLoss: 0.128561\n",
      "====> Epoch: 0 Average loss: 0.0044\n",
      "Ensemble model: 1\n",
      "Train Epoch: 0 [0/1984 (0%)]\tLoss: 0.763293\n",
      "Train Epoch: 0 [640/1984 (32%)]\tLoss: 0.178700\n",
      "Train Epoch: 0 [1280/1984 (65%)]\tLoss: 0.311694\n",
      "Train Epoch: 0 [1920/1984 (97%)]\tLoss: 0.115386\n",
      "====> Epoch: 0 Average loss: 0.0039\n",
      "Ensemble model: 2\n",
      "Train Epoch: 0 [0/1984 (0%)]\tLoss: 0.587230\n",
      "Train Epoch: 0 [640/1984 (32%)]\tLoss: 0.177205\n",
      "Train Epoch: 0 [1280/1984 (65%)]\tLoss: 0.111804\n",
      "Train Epoch: 0 [1920/1984 (97%)]\tLoss: 0.066701\n",
      "====> Epoch: 0 Average loss: 0.0032\n",
      "Ensemble model: 0\n",
      "Train Epoch: 1 [0/1984 (0%)]\tLoss: 0.133269\n",
      "Train Epoch: 1 [640/1984 (32%)]\tLoss: 0.106636\n",
      "Train Epoch: 1 [1280/1984 (65%)]\tLoss: 0.045631\n",
      "Train Epoch: 1 [1920/1984 (97%)]\tLoss: 0.029579\n",
      "====> Epoch: 1 Average loss: 0.0013\n",
      "Ensemble model: 1\n",
      "Train Epoch: 1 [0/1984 (0%)]\tLoss: 0.216867\n",
      "Train Epoch: 1 [640/1984 (32%)]\tLoss: 0.114994\n",
      "Train Epoch: 1 [1280/1984 (65%)]\tLoss: 0.055918\n",
      "Train Epoch: 1 [1920/1984 (97%)]\tLoss: 0.028313\n",
      "====> Epoch: 1 Average loss: 0.0013\n",
      "Ensemble model: 2\n",
      "Train Epoch: 1 [0/1984 (0%)]\tLoss: 0.032473\n",
      "Train Epoch: 1 [640/1984 (32%)]\tLoss: 0.022703\n",
      "Train Epoch: 1 [1280/1984 (65%)]\tLoss: 0.034912\n",
      "Train Epoch: 1 [1920/1984 (97%)]\tLoss: 0.010492\n",
      "====> Epoch: 1 Average loss: 0.0006\n",
      "Ensemble model: 0\n",
      "Train Epoch: 2 [0/1984 (0%)]\tLoss: 0.055079\n",
      "Train Epoch: 2 [640/1984 (32%)]\tLoss: 0.009132\n",
      "Train Epoch: 2 [1280/1984 (65%)]\tLoss: 0.010504\n",
      "Train Epoch: 2 [1920/1984 (97%)]\tLoss: 0.012479\n",
      "====> Epoch: 2 Average loss: 0.0003\n",
      "Ensemble model: 1\n",
      "Train Epoch: 2 [0/1984 (0%)]\tLoss: 0.041390\n",
      "Train Epoch: 2 [640/1984 (32%)]\tLoss: 0.012658\n",
      "Train Epoch: 2 [1280/1984 (65%)]\tLoss: 0.014210\n",
      "Train Epoch: 2 [1920/1984 (97%)]\tLoss: 0.004497\n",
      "====> Epoch: 2 Average loss: 0.0003\n",
      "Ensemble model: 2\n",
      "Train Epoch: 2 [0/1984 (0%)]\tLoss: 0.019942\n",
      "Train Epoch: 2 [640/1984 (32%)]\tLoss: 0.008627\n",
      "Train Epoch: 2 [1280/1984 (65%)]\tLoss: 0.004756\n",
      "Train Epoch: 2 [1920/1984 (97%)]\tLoss: 0.003219\n",
      "====> Epoch: 2 Average loss: 0.0001\n",
      "Ensemble model: 0\n",
      "Train Epoch: 3 [0/1984 (0%)]\tLoss: 0.010550\n",
      "Train Epoch: 3 [640/1984 (32%)]\tLoss: 0.002562\n",
      "Train Epoch: 3 [1280/1984 (65%)]\tLoss: 0.002943\n",
      "Train Epoch: 3 [1920/1984 (97%)]\tLoss: 0.016156\n",
      "====> Epoch: 3 Average loss: 0.0001\n",
      "Ensemble model: 1\n",
      "Train Epoch: 3 [0/1984 (0%)]\tLoss: 0.005916\n",
      "Train Epoch: 3 [640/1984 (32%)]\tLoss: 0.006873\n",
      "Train Epoch: 3 [1280/1984 (65%)]\tLoss: 0.002731\n",
      "Train Epoch: 3 [1920/1984 (97%)]\tLoss: 0.007805\n",
      "====> Epoch: 3 Average loss: 0.0001\n",
      "Ensemble model: 2\n",
      "Train Epoch: 3 [0/1984 (0%)]\tLoss: 0.005344\n",
      "Train Epoch: 3 [640/1984 (32%)]\tLoss: 0.003451\n",
      "Train Epoch: 3 [1280/1984 (65%)]\tLoss: 0.008923\n",
      "Train Epoch: 3 [1920/1984 (97%)]\tLoss: 0.010197\n",
      "====> Epoch: 3 Average loss: 0.0001\n",
      "Ensemble model: 0\n",
      "Train Epoch: 4 [0/1984 (0%)]\tLoss: 0.001270\n",
      "Train Epoch: 4 [640/1984 (32%)]\tLoss: 0.001069\n",
      "Train Epoch: 4 [1280/1984 (65%)]\tLoss: 0.001395\n",
      "Train Epoch: 4 [1920/1984 (97%)]\tLoss: 0.002075\n",
      "====> Epoch: 4 Average loss: 0.0001\n",
      "Ensemble model: 1\n",
      "Train Epoch: 4 [0/1984 (0%)]\tLoss: 0.002662\n",
      "Train Epoch: 4 [640/1984 (32%)]\tLoss: 0.005976\n",
      "Train Epoch: 4 [1280/1984 (65%)]\tLoss: 0.003095\n",
      "Train Epoch: 4 [1920/1984 (97%)]\tLoss: 0.001802\n",
      "====> Epoch: 4 Average loss: 0.0001\n",
      "Ensemble model: 2\n",
      "Train Epoch: 4 [0/1984 (0%)]\tLoss: 0.003191\n",
      "Train Epoch: 4 [640/1984 (32%)]\tLoss: 0.003526\n",
      "Train Epoch: 4 [1280/1984 (65%)]\tLoss: 0.003285\n",
      "Train Epoch: 4 [1920/1984 (97%)]\tLoss: 0.003339\n",
      "====> Epoch: 4 Average loss: 0.0001\n"
     ]
    }
   ],
   "source": [
    "# Training the ensemble of FC ResNets\n",
    "for epoch in range(num_epochs):\n",
    "    for i, model in enumerate(model_ensemble):\n",
    "        print (\"Ensemble model: \" + str(i))\n",
    "        train_loss = train_single_epoch(\n",
    "            epoch, model, dl_train_loader, optimizers[i], device\n",
    "        )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Visualising softmax entropy\n",
    "for model in model_ensemble:\n",
    "    model.eval()\n",
    "\n",
    "single_model = model_ensemble[1]\n",
    "\n",
    "domain = 3\n",
    "x_lin = np.linspace(-domain+0.5, domain+0.5, 100)\n",
    "y_lin = np.linspace(-domain, domain, 100)\n",
    "\n",
    "xx, yy = np.meshgrid(x_lin, y_lin)\n",
    "X_grid = np.column_stack([xx.flatten(), yy.flatten()])\n",
    "\n",
    "X_vis, y_vis = datasets.make_moons(n_samples=1000, noise=noise)\n",
    "mask = y_vis.astype(bool)\n",
    "\n",
    "X_grid = torch.from_numpy(X_grid).float().to(device)\n",
    "\n",
    "with torch.no_grad():\n",
    "    output = single_model(X_grid)\n",
    "    uncertainty = entropy(output)\n",
    "\n",
    "z = - uncertainty.cpu().numpy().reshape(xx.shape)\n",
    "\n",
    "plt.figure()\n",
    "plt.contourf(x_lin, y_lin, z, cmap='cividis')\n",
    "\n",
    "plt.scatter(X_vis[mask,0], X_vis[mask,1], s=10)\n",
    "plt.scatter(X_vis[~mask,0], X_vis[~mask,1], s=10)\n",
    "plt.savefig(\"two_moons_softmax_entropy_no_sn.png\", bbox_inches=\"tight\")\n",
    "plt.savefig(\"two_moons_softmax_entropy_no_sn.pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Visualising predictive entropy\n",
    "domain = 3\n",
    "x_lin = np.linspace(-domain+0.5, domain+0.5, 100)\n",
    "y_lin = np.linspace(-domain, domain, 100)\n",
    "\n",
    "xx, yy = np.meshgrid(x_lin, y_lin)\n",
    "X_grid = np.column_stack([xx.flatten(), yy.flatten()])\n",
    "\n",
    "X_vis, y_vis = datasets.make_moons(n_samples=1000, noise=noise)\n",
    "mask = y_vis.astype(bool)\n",
    "\n",
    "X_grid = torch.from_numpy(X_grid).float().to(device)\n",
    "\n",
    "with torch.no_grad():\n",
    "    output, uncertainty, _ = ensemble_forward_pass(model_ensemble, X_grid)\n",
    "\n",
    "z = - uncertainty.cpu().numpy().reshape(xx.shape)\n",
    "\n",
    "plt.figure()\n",
    "plt.contourf(x_lin, y_lin, z, cmap='cividis')\n",
    "\n",
    "plt.scatter(X_vis[mask,0], X_vis[mask,1], s=10)\n",
    "plt.scatter(X_vis[~mask,0], X_vis[~mask,1], s=10)\n",
    "\n",
    "plt.savefig(\"two_moons_predictive_entropy_no_sn.png\", bbox_inches=\"tight\")\n",
    "plt.savefig(\"two_moons_predictive_entropy_no_sn.pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "FCResNet(\n",
      "  (first): Linear(in_features=2, out_features=128, bias=True)\n",
      "  (residuals): ModuleList(\n",
      "    (0): Linear(in_features=128, out_features=128, bias=True)\n",
      "    (1): Linear(in_features=128, out_features=128, bias=True)\n",
      "    (2): Linear(in_features=128, out_features=128, bias=True)\n",
      "    (3): Linear(in_features=128, out_features=128, bias=True)\n",
      "  )\n",
      "  (dropout): Dropout(p=0.0, inplace=False)\n",
      "  (last): Linear(in_features=128, out_features=2, bias=True)\n",
      "  (activation): ReLU()\n",
      ")\n"
     ]
    }
   ],
   "source": [
    "# Train FC ResNet with spectral normalization\n",
    "\n",
    "features = 128\n",
    "depth = 4\n",
    "coeff = 0.65\n",
    "n_power_iterations = 5\n",
    "\n",
    "spectral_normalization = True\n",
    "dropout_rate = 0.0\n",
    "\n",
    "model = FCResNet(\n",
    "    input_dim=input_dim,\n",
    "    features=features,\n",
    "    depth=depth,\n",
    "    spectral_normalization=spectral_normalization,\n",
    "    coeff=coeff,\n",
    "    n_power_iterations=n_power_iterations,\n",
    "    dropout_rate=dropout_rate\n",
    ").to(device)\n",
    "\n",
    "print (model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "loss_fn = nn.CrossEntropyLoss()\n",
    "optimizer = torch.optim.Adam(model.parameters())\n",
    "num_epochs = 150"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train Epoch: 0 [0/1984 (0%)]\tLoss: 0.660564\n",
      "Train Epoch: 0 [640/1984 (32%)]\tLoss: 0.583091\n",
      "Train Epoch: 0 [1280/1984 (65%)]\tLoss: 0.449884\n",
      "Train Epoch: 0 [1920/1984 (97%)]\tLoss: 0.435066\n",
      "====> Epoch: 0 Average loss: 0.0084\n",
      "Train Epoch: 1 [0/1984 (0%)]\tLoss: 0.378749\n",
      "Train Epoch: 1 [640/1984 (32%)]\tLoss: 0.350785\n",
      "Train Epoch: 1 [1280/1984 (65%)]\tLoss: 0.406387\n",
      "Train Epoch: 1 [1920/1984 (97%)]\tLoss: 0.314573\n",
      "====> Epoch: 1 Average loss: 0.0055\n",
      "Train Epoch: 2 [0/1984 (0%)]\tLoss: 0.328401\n",
      "Train Epoch: 2 [640/1984 (32%)]\tLoss: 0.287965\n",
      "Train Epoch: 2 [1280/1984 (65%)]\tLoss: 0.348256\n",
      "Train Epoch: 2 [1920/1984 (97%)]\tLoss: 0.255123\n",
      "====> Epoch: 2 Average loss: 0.0048\n",
      "Train Epoch: 3 [0/1984 (0%)]\tLoss: 0.250969\n",
      "Train Epoch: 3 [640/1984 (32%)]\tLoss: 0.288830\n",
      "Train Epoch: 3 [1280/1984 (65%)]\tLoss: 0.261940\n",
      "Train Epoch: 3 [1920/1984 (97%)]\tLoss: 0.252808\n",
      "====> Epoch: 3 Average loss: 0.0044\n",
      "Train Epoch: 4 [0/1984 (0%)]\tLoss: 0.261587\n",
      "Train Epoch: 4 [640/1984 (32%)]\tLoss: 0.351348\n",
      "Train Epoch: 4 [1280/1984 (65%)]\tLoss: 0.295351\n",
      "Train Epoch: 4 [1920/1984 (97%)]\tLoss: 0.206104\n",
      "====> Epoch: 4 Average loss: 0.0043\n",
      "Train Epoch: 5 [0/1984 (0%)]\tLoss: 0.479941\n",
      "Train Epoch: 5 [640/1984 (32%)]\tLoss: 0.317206\n",
      "Train Epoch: 5 [1280/1984 (65%)]\tLoss: 0.318592\n",
      "Train Epoch: 5 [1920/1984 (97%)]\tLoss: 0.227633\n",
      "====> Epoch: 5 Average loss: 0.0042\n",
      "Train Epoch: 6 [0/1984 (0%)]\tLoss: 0.188505\n",
      "Train Epoch: 6 [640/1984 (32%)]\tLoss: 0.203877\n",
      "Train Epoch: 6 [1280/1984 (65%)]\tLoss: 0.162480\n",
      "Train Epoch: 6 [1920/1984 (97%)]\tLoss: 0.384095\n",
      "====> Epoch: 6 Average loss: 0.0040\n",
      "Train Epoch: 7 [0/1984 (0%)]\tLoss: 0.310069\n",
      "Train Epoch: 7 [640/1984 (32%)]\tLoss: 0.313008\n",
      "Train Epoch: 7 [1280/1984 (65%)]\tLoss: 0.264644\n",
      "Train Epoch: 7 [1920/1984 (97%)]\tLoss: 0.254294\n",
      "====> Epoch: 7 Average loss: 0.0041\n",
      "Train Epoch: 8 [0/1984 (0%)]\tLoss: 0.245252\n",
      "Train Epoch: 8 [640/1984 (32%)]\tLoss: 0.357011\n",
      "Train Epoch: 8 [1280/1984 (65%)]\tLoss: 0.201150\n",
      "Train Epoch: 8 [1920/1984 (97%)]\tLoss: 0.248713\n",
      "====> Epoch: 8 Average loss: 0.0042\n",
      "Train Epoch: 9 [0/1984 (0%)]\tLoss: 0.384584\n",
      "Train Epoch: 9 [640/1984 (32%)]\tLoss: 0.203508\n",
      "Train Epoch: 9 [1280/1984 (65%)]\tLoss: 0.259530\n",
      "Train Epoch: 9 [1920/1984 (97%)]\tLoss: 0.258483\n",
      "====> Epoch: 9 Average loss: 0.0040\n",
      "Train Epoch: 10 [0/1984 (0%)]\tLoss: 0.262665\n",
      "Train Epoch: 10 [640/1984 (32%)]\tLoss: 0.172050\n",
      "Train Epoch: 10 [1280/1984 (65%)]\tLoss: 0.194705\n",
      "Train Epoch: 10 [1920/1984 (97%)]\tLoss: 0.220971\n",
      "====> Epoch: 10 Average loss: 0.0041\n",
      "Train Epoch: 11 [0/1984 (0%)]\tLoss: 0.301671\n",
      "Train Epoch: 11 [640/1984 (32%)]\tLoss: 0.332831\n",
      "Train Epoch: 11 [1280/1984 (65%)]\tLoss: 0.339596\n",
      "Train Epoch: 11 [1920/1984 (97%)]\tLoss: 0.336615\n",
      "====> Epoch: 11 Average loss: 0.0042\n",
      "Train Epoch: 12 [0/1984 (0%)]\tLoss: 0.222075\n",
      "Train Epoch: 12 [640/1984 (32%)]\tLoss: 0.278390\n",
      "Train Epoch: 12 [1280/1984 (65%)]\tLoss: 0.307992\n",
      "Train Epoch: 12 [1920/1984 (97%)]\tLoss: 0.239149\n",
      "====> Epoch: 12 Average loss: 0.0040\n",
      "Train Epoch: 13 [0/1984 (0%)]\tLoss: 0.282133\n",
      "Train Epoch: 13 [640/1984 (32%)]\tLoss: 0.233694\n",
      "Train Epoch: 13 [1280/1984 (65%)]\tLoss: 0.205827\n",
      "Train Epoch: 13 [1920/1984 (97%)]\tLoss: 0.225493\n",
      "====> Epoch: 13 Average loss: 0.0040\n",
      "Train Epoch: 14 [0/1984 (0%)]\tLoss: 0.216341\n",
      "Train Epoch: 14 [640/1984 (32%)]\tLoss: 0.200789\n",
      "Train Epoch: 14 [1280/1984 (65%)]\tLoss: 0.304527\n",
      "Train Epoch: 14 [1920/1984 (97%)]\tLoss: 0.283323\n",
      "====> Epoch: 14 Average loss: 0.0041\n",
      "Train Epoch: 15 [0/1984 (0%)]\tLoss: 0.161046\n",
      "Train Epoch: 15 [640/1984 (32%)]\tLoss: 0.239266\n",
      "Train Epoch: 15 [1280/1984 (65%)]\tLoss: 0.284181\n",
      "Train Epoch: 15 [1920/1984 (97%)]\tLoss: 0.301942\n",
      "====> Epoch: 15 Average loss: 0.0043\n",
      "Train Epoch: 16 [0/1984 (0%)]\tLoss: 0.295668\n",
      "Train Epoch: 16 [640/1984 (32%)]\tLoss: 0.187410\n",
      "Train Epoch: 16 [1280/1984 (65%)]\tLoss: 0.260961\n",
      "Train Epoch: 16 [1920/1984 (97%)]\tLoss: 0.295992\n",
      "====> Epoch: 16 Average loss: 0.0039\n",
      "Train Epoch: 17 [0/1984 (0%)]\tLoss: 0.215882\n",
      "Train Epoch: 17 [640/1984 (32%)]\tLoss: 0.383077\n",
      "Train Epoch: 17 [1280/1984 (65%)]\tLoss: 0.183207\n",
      "Train Epoch: 17 [1920/1984 (97%)]\tLoss: 0.224322\n",
      "====> Epoch: 17 Average loss: 0.0039\n",
      "Train Epoch: 18 [0/1984 (0%)]\tLoss: 0.219554\n",
      "Train Epoch: 18 [640/1984 (32%)]\tLoss: 0.183739\n",
      "Train Epoch: 18 [1280/1984 (65%)]\tLoss: 0.360106\n",
      "Train Epoch: 18 [1920/1984 (97%)]\tLoss: 0.168304\n",
      "====> Epoch: 18 Average loss: 0.0038\n",
      "Train Epoch: 19 [0/1984 (0%)]\tLoss: 0.153726\n",
      "Train Epoch: 19 [640/1984 (32%)]\tLoss: 0.167365\n",
      "Train Epoch: 19 [1280/1984 (65%)]\tLoss: 0.268240\n",
      "Train Epoch: 19 [1920/1984 (97%)]\tLoss: 0.315825\n",
      "====> Epoch: 19 Average loss: 0.0039\n",
      "Train Epoch: 20 [0/1984 (0%)]\tLoss: 0.205548\n",
      "Train Epoch: 20 [640/1984 (32%)]\tLoss: 0.266792\n",
      "Train Epoch: 20 [1280/1984 (65%)]\tLoss: 0.268507\n",
      "Train Epoch: 20 [1920/1984 (97%)]\tLoss: 0.294256\n",
      "====> Epoch: 20 Average loss: 0.0037\n",
      "Train Epoch: 21 [0/1984 (0%)]\tLoss: 0.249654\n",
      "Train Epoch: 21 [640/1984 (32%)]\tLoss: 0.163131\n",
      "Train Epoch: 21 [1280/1984 (65%)]\tLoss: 0.213700\n",
      "Train Epoch: 21 [1920/1984 (97%)]\tLoss: 0.266816\n",
      "====> Epoch: 21 Average loss: 0.0036\n",
      "Train Epoch: 22 [0/1984 (0%)]\tLoss: 0.241297\n",
      "Train Epoch: 22 [640/1984 (32%)]\tLoss: 0.217238\n",
      "Train Epoch: 22 [1280/1984 (65%)]\tLoss: 0.223713\n",
      "Train Epoch: 22 [1920/1984 (97%)]\tLoss: 0.189875\n",
      "====> Epoch: 22 Average loss: 0.0036\n",
      "Train Epoch: 23 [0/1984 (0%)]\tLoss: 0.185825\n",
      "Train Epoch: 23 [640/1984 (32%)]\tLoss: 0.159810\n",
      "Train Epoch: 23 [1280/1984 (65%)]\tLoss: 0.177750\n",
      "Train Epoch: 23 [1920/1984 (97%)]\tLoss: 0.151500\n",
      "====> Epoch: 23 Average loss: 0.0035\n",
      "Train Epoch: 24 [0/1984 (0%)]\tLoss: 0.247679\n",
      "Train Epoch: 24 [640/1984 (32%)]\tLoss: 0.151180\n",
      "Train Epoch: 24 [1280/1984 (65%)]\tLoss: 0.118464\n",
      "Train Epoch: 24 [1920/1984 (97%)]\tLoss: 0.209826\n",
      "====> Epoch: 24 Average loss: 0.0033\n",
      "Train Epoch: 25 [0/1984 (0%)]\tLoss: 0.282110\n",
      "Train Epoch: 25 [640/1984 (32%)]\tLoss: 0.217176\n",
      "Train Epoch: 25 [1280/1984 (65%)]\tLoss: 0.252829\n",
      "Train Epoch: 25 [1920/1984 (97%)]\tLoss: 0.273989\n",
      "====> Epoch: 25 Average loss: 0.0034\n",
      "Train Epoch: 26 [0/1984 (0%)]\tLoss: 0.267453\n",
      "Train Epoch: 26 [640/1984 (32%)]\tLoss: 0.180920\n",
      "Train Epoch: 26 [1280/1984 (65%)]\tLoss: 0.249275\n",
      "Train Epoch: 26 [1920/1984 (97%)]\tLoss: 0.202086\n",
      "====> Epoch: 26 Average loss: 0.0032\n",
      "Train Epoch: 27 [0/1984 (0%)]\tLoss: 0.221863\n",
      "Train Epoch: 27 [640/1984 (32%)]\tLoss: 0.174706\n",
      "Train Epoch: 27 [1280/1984 (65%)]\tLoss: 0.188848\n",
      "Train Epoch: 27 [1920/1984 (97%)]\tLoss: 0.145292\n",
      "====> Epoch: 27 Average loss: 0.0032\n",
      "Train Epoch: 28 [0/1984 (0%)]\tLoss: 0.175693\n",
      "Train Epoch: 28 [640/1984 (32%)]\tLoss: 0.192791\n",
      "Train Epoch: 28 [1280/1984 (65%)]\tLoss: 0.183785\n",
      "Train Epoch: 28 [1920/1984 (97%)]\tLoss: 0.149362\n",
      "====> Epoch: 28 Average loss: 0.0031\n",
      "Train Epoch: 29 [0/1984 (0%)]\tLoss: 0.113800\n",
      "Train Epoch: 29 [640/1984 (32%)]\tLoss: 0.251552\n",
      "Train Epoch: 29 [1280/1984 (65%)]\tLoss: 0.245960\n",
      "Train Epoch: 29 [1920/1984 (97%)]\tLoss: 0.152552\n",
      "====> Epoch: 29 Average loss: 0.0030\n",
      "Train Epoch: 30 [0/1984 (0%)]\tLoss: 0.076352\n",
      "Train Epoch: 30 [640/1984 (32%)]\tLoss: 0.119178\n",
      "Train Epoch: 30 [1280/1984 (65%)]\tLoss: 0.147050\n",
      "Train Epoch: 30 [1920/1984 (97%)]\tLoss: 0.176383\n",
      "====> Epoch: 30 Average loss: 0.0029\n",
      "Train Epoch: 31 [0/1984 (0%)]\tLoss: 0.143766\n",
      "Train Epoch: 31 [640/1984 (32%)]\tLoss: 0.178044\n",
      "Train Epoch: 31 [1280/1984 (65%)]\tLoss: 0.122943\n",
      "Train Epoch: 31 [1920/1984 (97%)]\tLoss: 0.161709\n",
      "====> Epoch: 31 Average loss: 0.0028\n",
      "Train Epoch: 32 [0/1984 (0%)]\tLoss: 0.116576\n",
      "Train Epoch: 32 [640/1984 (32%)]\tLoss: 0.186826\n",
      "Train Epoch: 32 [1280/1984 (65%)]\tLoss: 0.134849\n",
      "Train Epoch: 32 [1920/1984 (97%)]\tLoss: 0.163441\n",
      "====> Epoch: 32 Average loss: 0.0027\n",
      "Train Epoch: 33 [0/1984 (0%)]\tLoss: 0.139948\n",
      "Train Epoch: 33 [640/1984 (32%)]\tLoss: 0.162131\n",
      "Train Epoch: 33 [1280/1984 (65%)]\tLoss: 0.224508\n",
      "Train Epoch: 33 [1920/1984 (97%)]\tLoss: 0.180414\n",
      "====> Epoch: 33 Average loss: 0.0027\n",
      "Train Epoch: 34 [0/1984 (0%)]\tLoss: 0.097609\n",
      "Train Epoch: 34 [640/1984 (32%)]\tLoss: 0.157999\n",
      "Train Epoch: 34 [1280/1984 (65%)]\tLoss: 0.208797\n",
      "Train Epoch: 34 [1920/1984 (97%)]\tLoss: 0.142235\n",
      "====> Epoch: 34 Average loss: 0.0026\n",
      "Train Epoch: 35 [0/1984 (0%)]\tLoss: 0.216955\n",
      "Train Epoch: 35 [640/1984 (32%)]\tLoss: 0.131716\n",
      "Train Epoch: 35 [1280/1984 (65%)]\tLoss: 0.150393\n",
      "Train Epoch: 35 [1920/1984 (97%)]\tLoss: 0.157235\n",
      "====> Epoch: 35 Average loss: 0.0025\n",
      "Train Epoch: 36 [0/1984 (0%)]\tLoss: 0.158054\n",
      "Train Epoch: 36 [640/1984 (32%)]\tLoss: 0.100509\n",
      "Train Epoch: 36 [1280/1984 (65%)]\tLoss: 0.086361\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train Epoch: 36 [1920/1984 (97%)]\tLoss: 0.172579\n",
      "====> Epoch: 36 Average loss: 0.0026\n",
      "Train Epoch: 37 [0/1984 (0%)]\tLoss: 0.220616\n",
      "Train Epoch: 37 [640/1984 (32%)]\tLoss: 0.112979\n",
      "Train Epoch: 37 [1280/1984 (65%)]\tLoss: 0.204004\n",
      "Train Epoch: 37 [1920/1984 (97%)]\tLoss: 0.163981\n",
      "====> Epoch: 37 Average loss: 0.0025\n",
      "Train Epoch: 38 [0/1984 (0%)]\tLoss: 0.190393\n",
      "Train Epoch: 38 [640/1984 (32%)]\tLoss: 0.193478\n",
      "Train Epoch: 38 [1280/1984 (65%)]\tLoss: 0.078215\n",
      "Train Epoch: 38 [1920/1984 (97%)]\tLoss: 0.165409\n",
      "====> Epoch: 38 Average loss: 0.0026\n",
      "Train Epoch: 39 [0/1984 (0%)]\tLoss: 0.123539\n",
      "Train Epoch: 39 [640/1984 (32%)]\tLoss: 0.169357\n",
      "Train Epoch: 39 [1280/1984 (65%)]\tLoss: 0.116744\n",
      "Train Epoch: 39 [1920/1984 (97%)]\tLoss: 0.135234\n",
      "====> Epoch: 39 Average loss: 0.0023\n",
      "Train Epoch: 40 [0/1984 (0%)]\tLoss: 0.142044\n",
      "Train Epoch: 40 [640/1984 (32%)]\tLoss: 0.088272\n",
      "Train Epoch: 40 [1280/1984 (65%)]\tLoss: 0.110089\n",
      "Train Epoch: 40 [1920/1984 (97%)]\tLoss: 0.151809\n",
      "====> Epoch: 40 Average loss: 0.0024\n",
      "Train Epoch: 41 [0/1984 (0%)]\tLoss: 0.150661\n",
      "Train Epoch: 41 [640/1984 (32%)]\tLoss: 0.111781\n",
      "Train Epoch: 41 [1280/1984 (65%)]\tLoss: 0.150260\n",
      "Train Epoch: 41 [1920/1984 (97%)]\tLoss: 0.218862\n",
      "====> Epoch: 41 Average loss: 0.0022\n",
      "Train Epoch: 42 [0/1984 (0%)]\tLoss: 0.110272\n",
      "Train Epoch: 42 [640/1984 (32%)]\tLoss: 0.178919\n",
      "Train Epoch: 42 [1280/1984 (65%)]\tLoss: 0.186034\n",
      "Train Epoch: 42 [1920/1984 (97%)]\tLoss: 0.113733\n",
      "====> Epoch: 42 Average loss: 0.0022\n",
      "Train Epoch: 43 [0/1984 (0%)]\tLoss: 0.152677\n",
      "Train Epoch: 43 [640/1984 (32%)]\tLoss: 0.090699\n",
      "Train Epoch: 43 [1280/1984 (65%)]\tLoss: 0.202616\n",
      "Train Epoch: 43 [1920/1984 (97%)]\tLoss: 0.106838\n",
      "====> Epoch: 43 Average loss: 0.0021\n",
      "Train Epoch: 44 [0/1984 (0%)]\tLoss: 0.109908\n",
      "Train Epoch: 44 [640/1984 (32%)]\tLoss: 0.200746\n",
      "Train Epoch: 44 [1280/1984 (65%)]\tLoss: 0.110526\n",
      "Train Epoch: 44 [1920/1984 (97%)]\tLoss: 0.198399\n",
      "====> Epoch: 44 Average loss: 0.0021\n",
      "Train Epoch: 45 [0/1984 (0%)]\tLoss: 0.126043\n",
      "Train Epoch: 45 [640/1984 (32%)]\tLoss: 0.296325\n",
      "Train Epoch: 45 [1280/1984 (65%)]\tLoss: 0.100648\n",
      "Train Epoch: 45 [1920/1984 (97%)]\tLoss: 0.127129\n",
      "====> Epoch: 45 Average loss: 0.0021\n",
      "Train Epoch: 46 [0/1984 (0%)]\tLoss: 0.155916\n",
      "Train Epoch: 46 [640/1984 (32%)]\tLoss: 0.134433\n",
      "Train Epoch: 46 [1280/1984 (65%)]\tLoss: 0.103474\n",
      "Train Epoch: 46 [1920/1984 (97%)]\tLoss: 0.082103\n",
      "====> Epoch: 46 Average loss: 0.0019\n",
      "Train Epoch: 47 [0/1984 (0%)]\tLoss: 0.144397\n",
      "Train Epoch: 47 [640/1984 (32%)]\tLoss: 0.124308\n",
      "Train Epoch: 47 [1280/1984 (65%)]\tLoss: 0.165303\n",
      "Train Epoch: 47 [1920/1984 (97%)]\tLoss: 0.108260\n",
      "====> Epoch: 47 Average loss: 0.0019\n",
      "Train Epoch: 48 [0/1984 (0%)]\tLoss: 0.069357\n",
      "Train Epoch: 48 [640/1984 (32%)]\tLoss: 0.091756\n",
      "Train Epoch: 48 [1280/1984 (65%)]\tLoss: 0.117390\n",
      "Train Epoch: 48 [1920/1984 (97%)]\tLoss: 0.136283\n",
      "====> Epoch: 48 Average loss: 0.0018\n",
      "Train Epoch: 49 [0/1984 (0%)]\tLoss: 0.104134\n",
      "Train Epoch: 49 [640/1984 (32%)]\tLoss: 0.108518\n",
      "Train Epoch: 49 [1280/1984 (65%)]\tLoss: 0.134530\n",
      "Train Epoch: 49 [1920/1984 (97%)]\tLoss: 0.085818\n",
      "====> Epoch: 49 Average loss: 0.0018\n",
      "Train Epoch: 50 [0/1984 (0%)]\tLoss: 0.111542\n",
      "Train Epoch: 50 [640/1984 (32%)]\tLoss: 0.106348\n",
      "Train Epoch: 50 [1280/1984 (65%)]\tLoss: 0.143946\n",
      "Train Epoch: 50 [1920/1984 (97%)]\tLoss: 0.136190\n",
      "====> Epoch: 50 Average loss: 0.0018\n",
      "Train Epoch: 51 [0/1984 (0%)]\tLoss: 0.104067\n",
      "Train Epoch: 51 [640/1984 (32%)]\tLoss: 0.102624\n",
      "Train Epoch: 51 [1280/1984 (65%)]\tLoss: 0.107644\n",
      "Train Epoch: 51 [1920/1984 (97%)]\tLoss: 0.098762\n",
      "====> Epoch: 51 Average loss: 0.0018\n",
      "Train Epoch: 52 [0/1984 (0%)]\tLoss: 0.089886\n",
      "Train Epoch: 52 [640/1984 (32%)]\tLoss: 0.064431\n",
      "Train Epoch: 52 [1280/1984 (65%)]\tLoss: 0.109197\n",
      "Train Epoch: 52 [1920/1984 (97%)]\tLoss: 0.094848\n",
      "====> Epoch: 52 Average loss: 0.0017\n",
      "Train Epoch: 53 [0/1984 (0%)]\tLoss: 0.116104\n",
      "Train Epoch: 53 [640/1984 (32%)]\tLoss: 0.108655\n",
      "Train Epoch: 53 [1280/1984 (65%)]\tLoss: 0.056492\n",
      "Train Epoch: 53 [1920/1984 (97%)]\tLoss: 0.076264\n",
      "====> Epoch: 53 Average loss: 0.0016\n",
      "Train Epoch: 54 [0/1984 (0%)]\tLoss: 0.091604\n",
      "Train Epoch: 54 [640/1984 (32%)]\tLoss: 0.147046\n",
      "Train Epoch: 54 [1280/1984 (65%)]\tLoss: 0.143199\n",
      "Train Epoch: 54 [1920/1984 (97%)]\tLoss: 0.054399\n",
      "====> Epoch: 54 Average loss: 0.0017\n",
      "Train Epoch: 55 [0/1984 (0%)]\tLoss: 0.107594\n",
      "Train Epoch: 55 [640/1984 (32%)]\tLoss: 0.092421\n",
      "Train Epoch: 55 [1280/1984 (65%)]\tLoss: 0.110457\n",
      "Train Epoch: 55 [1920/1984 (97%)]\tLoss: 0.070391\n",
      "====> Epoch: 55 Average loss: 0.0016\n",
      "Train Epoch: 56 [0/1984 (0%)]\tLoss: 0.093062\n",
      "Train Epoch: 56 [640/1984 (32%)]\tLoss: 0.134405\n",
      "Train Epoch: 56 [1280/1984 (65%)]\tLoss: 0.120298\n",
      "Train Epoch: 56 [1920/1984 (97%)]\tLoss: 0.081943\n",
      "====> Epoch: 56 Average loss: 0.0015\n",
      "Train Epoch: 57 [0/1984 (0%)]\tLoss: 0.126165\n",
      "Train Epoch: 57 [640/1984 (32%)]\tLoss: 0.068384\n",
      "Train Epoch: 57 [1280/1984 (65%)]\tLoss: 0.102360\n",
      "Train Epoch: 57 [1920/1984 (97%)]\tLoss: 0.095730\n",
      "====> Epoch: 57 Average loss: 0.0015\n",
      "Train Epoch: 58 [0/1984 (0%)]\tLoss: 0.057571\n",
      "Train Epoch: 58 [640/1984 (32%)]\tLoss: 0.068561\n",
      "Train Epoch: 58 [1280/1984 (65%)]\tLoss: 0.079113\n",
      "Train Epoch: 58 [1920/1984 (97%)]\tLoss: 0.074633\n",
      "====> Epoch: 58 Average loss: 0.0014\n",
      "Train Epoch: 59 [0/1984 (0%)]\tLoss: 0.064338\n",
      "Train Epoch: 59 [640/1984 (32%)]\tLoss: 0.107992\n",
      "Train Epoch: 59 [1280/1984 (65%)]\tLoss: 0.089934\n",
      "Train Epoch: 59 [1920/1984 (97%)]\tLoss: 0.070815\n",
      "====> Epoch: 59 Average loss: 0.0015\n",
      "Train Epoch: 60 [0/1984 (0%)]\tLoss: 0.107939\n",
      "Train Epoch: 60 [640/1984 (32%)]\tLoss: 0.094391\n",
      "Train Epoch: 60 [1280/1984 (65%)]\tLoss: 0.059833\n",
      "Train Epoch: 60 [1920/1984 (97%)]\tLoss: 0.083289\n",
      "====> Epoch: 60 Average loss: 0.0014\n",
      "Train Epoch: 61 [0/1984 (0%)]\tLoss: 0.129103\n",
      "Train Epoch: 61 [640/1984 (32%)]\tLoss: 0.054205\n",
      "Train Epoch: 61 [1280/1984 (65%)]\tLoss: 0.079060\n",
      "Train Epoch: 61 [1920/1984 (97%)]\tLoss: 0.047542\n",
      "====> Epoch: 61 Average loss: 0.0013\n",
      "Train Epoch: 62 [0/1984 (0%)]\tLoss: 0.050206\n",
      "Train Epoch: 62 [640/1984 (32%)]\tLoss: 0.133299\n",
      "Train Epoch: 62 [1280/1984 (65%)]\tLoss: 0.095326\n",
      "Train Epoch: 62 [1920/1984 (97%)]\tLoss: 0.093841\n",
      "====> Epoch: 62 Average loss: 0.0013\n",
      "Train Epoch: 63 [0/1984 (0%)]\tLoss: 0.057783\n",
      "Train Epoch: 63 [640/1984 (32%)]\tLoss: 0.051238\n",
      "Train Epoch: 63 [1280/1984 (65%)]\tLoss: 0.085459\n",
      "Train Epoch: 63 [1920/1984 (97%)]\tLoss: 0.050896\n",
      "====> Epoch: 63 Average loss: 0.0013\n",
      "Train Epoch: 64 [0/1984 (0%)]\tLoss: 0.083677\n",
      "Train Epoch: 64 [640/1984 (32%)]\tLoss: 0.060906\n",
      "Train Epoch: 64 [1280/1984 (65%)]\tLoss: 0.079833\n",
      "Train Epoch: 64 [1920/1984 (97%)]\tLoss: 0.077881\n",
      "====> Epoch: 64 Average loss: 0.0012\n",
      "Train Epoch: 65 [0/1984 (0%)]\tLoss: 0.090711\n",
      "Train Epoch: 65 [640/1984 (32%)]\tLoss: 0.071351\n",
      "Train Epoch: 65 [1280/1984 (65%)]\tLoss: 0.070059\n",
      "Train Epoch: 65 [1920/1984 (97%)]\tLoss: 0.055502\n",
      "====> Epoch: 65 Average loss: 0.0013\n",
      "Train Epoch: 66 [0/1984 (0%)]\tLoss: 0.068023\n",
      "Train Epoch: 66 [640/1984 (32%)]\tLoss: 0.096226\n",
      "Train Epoch: 66 [1280/1984 (65%)]\tLoss: 0.055894\n",
      "Train Epoch: 66 [1920/1984 (97%)]\tLoss: 0.067554\n",
      "====> Epoch: 66 Average loss: 0.0012\n",
      "Train Epoch: 67 [0/1984 (0%)]\tLoss: 0.114266\n",
      "Train Epoch: 67 [640/1984 (32%)]\tLoss: 0.111726\n",
      "Train Epoch: 67 [1280/1984 (65%)]\tLoss: 0.087246\n",
      "Train Epoch: 67 [1920/1984 (97%)]\tLoss: 0.037117\n",
      "====> Epoch: 67 Average loss: 0.0012\n",
      "Train Epoch: 68 [0/1984 (0%)]\tLoss: 0.052166\n",
      "Train Epoch: 68 [640/1984 (32%)]\tLoss: 0.085275\n",
      "Train Epoch: 68 [1280/1984 (65%)]\tLoss: 0.053775\n",
      "Train Epoch: 68 [1920/1984 (97%)]\tLoss: 0.091210\n",
      "====> Epoch: 68 Average loss: 0.0012\n",
      "Train Epoch: 69 [0/1984 (0%)]\tLoss: 0.066614\n",
      "Train Epoch: 69 [640/1984 (32%)]\tLoss: 0.090583\n",
      "Train Epoch: 69 [1280/1984 (65%)]\tLoss: 0.056567\n",
      "Train Epoch: 69 [1920/1984 (97%)]\tLoss: 0.039058\n",
      "====> Epoch: 69 Average loss: 0.0012\n",
      "Train Epoch: 70 [0/1984 (0%)]\tLoss: 0.053812\n",
      "Train Epoch: 70 [640/1984 (32%)]\tLoss: 0.073988\n",
      "Train Epoch: 70 [1280/1984 (65%)]\tLoss: 0.084845\n",
      "Train Epoch: 70 [1920/1984 (97%)]\tLoss: 0.068683\n",
      "====> Epoch: 70 Average loss: 0.0011\n",
      "Train Epoch: 71 [0/1984 (0%)]\tLoss: 0.079604\n",
      "Train Epoch: 71 [640/1984 (32%)]\tLoss: 0.053559\n",
      "Train Epoch: 71 [1280/1984 (65%)]\tLoss: 0.080585\n",
      "Train Epoch: 71 [1920/1984 (97%)]\tLoss: 0.057679\n",
      "====> Epoch: 71 Average loss: 0.0011\n",
      "Train Epoch: 72 [0/1984 (0%)]\tLoss: 0.050609\n",
      "Train Epoch: 72 [640/1984 (32%)]\tLoss: 0.079498\n",
      "Train Epoch: 72 [1280/1984 (65%)]\tLoss: 0.060959\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train Epoch: 72 [1920/1984 (97%)]\tLoss: 0.074535\n",
      "====> Epoch: 72 Average loss: 0.0010\n",
      "Train Epoch: 73 [0/1984 (0%)]\tLoss: 0.068140\n",
      "Train Epoch: 73 [640/1984 (32%)]\tLoss: 0.066683\n",
      "Train Epoch: 73 [1280/1984 (65%)]\tLoss: 0.127709\n",
      "Train Epoch: 73 [1920/1984 (97%)]\tLoss: 0.045397\n",
      "====> Epoch: 73 Average loss: 0.0010\n",
      "Train Epoch: 74 [0/1984 (0%)]\tLoss: 0.068157\n",
      "Train Epoch: 74 [640/1984 (32%)]\tLoss: 0.038318\n",
      "Train Epoch: 74 [1280/1984 (65%)]\tLoss: 0.068306\n",
      "Train Epoch: 74 [1920/1984 (97%)]\tLoss: 0.064632\n",
      "====> Epoch: 74 Average loss: 0.0010\n",
      "Train Epoch: 75 [0/1984 (0%)]\tLoss: 0.083699\n",
      "Train Epoch: 75 [640/1984 (32%)]\tLoss: 0.047419\n",
      "Train Epoch: 75 [1280/1984 (65%)]\tLoss: 0.042518\n",
      "Train Epoch: 75 [1920/1984 (97%)]\tLoss: 0.062367\n",
      "====> Epoch: 75 Average loss: 0.0010\n",
      "Train Epoch: 76 [0/1984 (0%)]\tLoss: 0.068141\n",
      "Train Epoch: 76 [640/1984 (32%)]\tLoss: 0.051992\n",
      "Train Epoch: 76 [1280/1984 (65%)]\tLoss: 0.120516\n",
      "Train Epoch: 76 [1920/1984 (97%)]\tLoss: 0.064788\n",
      "====> Epoch: 76 Average loss: 0.0010\n",
      "Train Epoch: 77 [0/1984 (0%)]\tLoss: 0.067179\n",
      "Train Epoch: 77 [640/1984 (32%)]\tLoss: 0.041785\n",
      "Train Epoch: 77 [1280/1984 (65%)]\tLoss: 0.044778\n",
      "Train Epoch: 77 [1920/1984 (97%)]\tLoss: 0.030226\n",
      "====> Epoch: 77 Average loss: 0.0010\n",
      "Train Epoch: 78 [0/1984 (0%)]\tLoss: 0.036482\n",
      "Train Epoch: 78 [640/1984 (32%)]\tLoss: 0.072941\n",
      "Train Epoch: 78 [1280/1984 (65%)]\tLoss: 0.051912\n",
      "Train Epoch: 78 [1920/1984 (97%)]\tLoss: 0.049072\n",
      "====> Epoch: 78 Average loss: 0.0010\n",
      "Train Epoch: 79 [0/1984 (0%)]\tLoss: 0.064259\n",
      "Train Epoch: 79 [640/1984 (32%)]\tLoss: 0.041930\n",
      "Train Epoch: 79 [1280/1984 (65%)]\tLoss: 0.033957\n",
      "Train Epoch: 79 [1920/1984 (97%)]\tLoss: 0.068708\n",
      "====> Epoch: 79 Average loss: 0.0009\n",
      "Train Epoch: 80 [0/1984 (0%)]\tLoss: 0.040957\n",
      "Train Epoch: 80 [640/1984 (32%)]\tLoss: 0.086621\n",
      "Train Epoch: 80 [1280/1984 (65%)]\tLoss: 0.075560\n",
      "Train Epoch: 80 [1920/1984 (97%)]\tLoss: 0.050342\n",
      "====> Epoch: 80 Average loss: 0.0009\n",
      "Train Epoch: 81 [0/1984 (0%)]\tLoss: 0.023106\n",
      "Train Epoch: 81 [640/1984 (32%)]\tLoss: 0.063781\n",
      "Train Epoch: 81 [1280/1984 (65%)]\tLoss: 0.058173\n",
      "Train Epoch: 81 [1920/1984 (97%)]\tLoss: 0.043675\n",
      "====> Epoch: 81 Average loss: 0.0009\n",
      "Train Epoch: 82 [0/1984 (0%)]\tLoss: 0.039376\n",
      "Train Epoch: 82 [640/1984 (32%)]\tLoss: 0.037680\n",
      "Train Epoch: 82 [1280/1984 (65%)]\tLoss: 0.063511\n",
      "Train Epoch: 82 [1920/1984 (97%)]\tLoss: 0.087799\n",
      "====> Epoch: 82 Average loss: 0.0009\n",
      "Train Epoch: 83 [0/1984 (0%)]\tLoss: 0.047954\n",
      "Train Epoch: 83 [640/1984 (32%)]\tLoss: 0.054983\n",
      "Train Epoch: 83 [1280/1984 (65%)]\tLoss: 0.037920\n",
      "Train Epoch: 83 [1920/1984 (97%)]\tLoss: 0.034866\n",
      "====> Epoch: 83 Average loss: 0.0008\n",
      "Train Epoch: 84 [0/1984 (0%)]\tLoss: 0.070809\n",
      "Train Epoch: 84 [640/1984 (32%)]\tLoss: 0.070962\n",
      "Train Epoch: 84 [1280/1984 (65%)]\tLoss: 0.065069\n",
      "Train Epoch: 84 [1920/1984 (97%)]\tLoss: 0.067371\n",
      "====> Epoch: 84 Average loss: 0.0009\n",
      "Train Epoch: 85 [0/1984 (0%)]\tLoss: 0.065497\n",
      "Train Epoch: 85 [640/1984 (32%)]\tLoss: 0.071594\n",
      "Train Epoch: 85 [1280/1984 (65%)]\tLoss: 0.044885\n",
      "Train Epoch: 85 [1920/1984 (97%)]\tLoss: 0.059733\n",
      "====> Epoch: 85 Average loss: 0.0008\n",
      "Train Epoch: 86 [0/1984 (0%)]\tLoss: 0.020548\n",
      "Train Epoch: 86 [640/1984 (32%)]\tLoss: 0.046791\n",
      "Train Epoch: 86 [1280/1984 (65%)]\tLoss: 0.027392\n",
      "Train Epoch: 86 [1920/1984 (97%)]\tLoss: 0.049720\n",
      "====> Epoch: 86 Average loss: 0.0008\n",
      "Train Epoch: 87 [0/1984 (0%)]\tLoss: 0.046302\n",
      "Train Epoch: 87 [640/1984 (32%)]\tLoss: 0.051911\n",
      "Train Epoch: 87 [1280/1984 (65%)]\tLoss: 0.061015\n",
      "Train Epoch: 87 [1920/1984 (97%)]\tLoss: 0.029280\n",
      "====> Epoch: 87 Average loss: 0.0008\n",
      "Train Epoch: 88 [0/1984 (0%)]\tLoss: 0.049157\n",
      "Train Epoch: 88 [640/1984 (32%)]\tLoss: 0.069650\n",
      "Train Epoch: 88 [1280/1984 (65%)]\tLoss: 0.051974\n",
      "Train Epoch: 88 [1920/1984 (97%)]\tLoss: 0.038557\n",
      "====> Epoch: 88 Average loss: 0.0008\n",
      "Train Epoch: 89 [0/1984 (0%)]\tLoss: 0.053131\n",
      "Train Epoch: 89 [640/1984 (32%)]\tLoss: 0.034239\n",
      "Train Epoch: 89 [1280/1984 (65%)]\tLoss: 0.027626\n",
      "Train Epoch: 89 [1920/1984 (97%)]\tLoss: 0.044206\n",
      "====> Epoch: 89 Average loss: 0.0007\n",
      "Train Epoch: 90 [0/1984 (0%)]\tLoss: 0.033390\n",
      "Train Epoch: 90 [640/1984 (32%)]\tLoss: 0.055766\n",
      "Train Epoch: 90 [1280/1984 (65%)]\tLoss: 0.036130\n",
      "Train Epoch: 90 [1920/1984 (97%)]\tLoss: 0.051609\n",
      "====> Epoch: 90 Average loss: 0.0007\n",
      "Train Epoch: 91 [0/1984 (0%)]\tLoss: 0.021920\n",
      "Train Epoch: 91 [640/1984 (32%)]\tLoss: 0.029929\n",
      "Train Epoch: 91 [1280/1984 (65%)]\tLoss: 0.040857\n",
      "Train Epoch: 91 [1920/1984 (97%)]\tLoss: 0.027013\n",
      "====> Epoch: 91 Average loss: 0.0007\n",
      "Train Epoch: 92 [0/1984 (0%)]\tLoss: 0.032455\n",
      "Train Epoch: 92 [640/1984 (32%)]\tLoss: 0.082131\n",
      "Train Epoch: 92 [1280/1984 (65%)]\tLoss: 0.023603\n",
      "Train Epoch: 92 [1920/1984 (97%)]\tLoss: 0.036238\n",
      "====> Epoch: 92 Average loss: 0.0009\n",
      "Train Epoch: 93 [0/1984 (0%)]\tLoss: 0.047966\n",
      "Train Epoch: 93 [640/1984 (32%)]\tLoss: 0.028480\n",
      "Train Epoch: 93 [1280/1984 (65%)]\tLoss: 0.031779\n",
      "Train Epoch: 93 [1920/1984 (97%)]\tLoss: 0.063805\n",
      "====> Epoch: 93 Average loss: 0.0007\n",
      "Train Epoch: 94 [0/1984 (0%)]\tLoss: 0.063024\n",
      "Train Epoch: 94 [640/1984 (32%)]\tLoss: 0.020863\n",
      "Train Epoch: 94 [1280/1984 (65%)]\tLoss: 0.088103\n",
      "Train Epoch: 94 [1920/1984 (97%)]\tLoss: 0.064163\n",
      "====> Epoch: 94 Average loss: 0.0008\n",
      "Train Epoch: 95 [0/1984 (0%)]\tLoss: 0.030671\n",
      "Train Epoch: 95 [640/1984 (32%)]\tLoss: 0.093586\n",
      "Train Epoch: 95 [1280/1984 (65%)]\tLoss: 0.052713\n",
      "Train Epoch: 95 [1920/1984 (97%)]\tLoss: 0.028627\n",
      "====> Epoch: 95 Average loss: 0.0008\n",
      "Train Epoch: 96 [0/1984 (0%)]\tLoss: 0.043683\n",
      "Train Epoch: 96 [640/1984 (32%)]\tLoss: 0.046389\n",
      "Train Epoch: 96 [1280/1984 (65%)]\tLoss: 0.034465\n",
      "Train Epoch: 96 [1920/1984 (97%)]\tLoss: 0.041616\n",
      "====> Epoch: 96 Average loss: 0.0007\n",
      "Train Epoch: 97 [0/1984 (0%)]\tLoss: 0.038998\n",
      "Train Epoch: 97 [640/1984 (32%)]\tLoss: 0.041826\n",
      "Train Epoch: 97 [1280/1984 (65%)]\tLoss: 0.025402\n",
      "Train Epoch: 97 [1920/1984 (97%)]\tLoss: 0.037291\n",
      "====> Epoch: 97 Average loss: 0.0008\n",
      "Train Epoch: 98 [0/1984 (0%)]\tLoss: 0.023161\n",
      "Train Epoch: 98 [640/1984 (32%)]\tLoss: 0.019721\n",
      "Train Epoch: 98 [1280/1984 (65%)]\tLoss: 0.032759\n",
      "Train Epoch: 98 [1920/1984 (97%)]\tLoss: 0.043322\n",
      "====> Epoch: 98 Average loss: 0.0006\n",
      "Train Epoch: 99 [0/1984 (0%)]\tLoss: 0.028067\n",
      "Train Epoch: 99 [640/1984 (32%)]\tLoss: 0.064525\n",
      "Train Epoch: 99 [1280/1984 (65%)]\tLoss: 0.039064\n",
      "Train Epoch: 99 [1920/1984 (97%)]\tLoss: 0.036024\n",
      "====> Epoch: 99 Average loss: 0.0007\n",
      "Train Epoch: 100 [0/1984 (0%)]\tLoss: 0.043949\n",
      "Train Epoch: 100 [640/1984 (32%)]\tLoss: 0.062820\n",
      "Train Epoch: 100 [1280/1984 (65%)]\tLoss: 0.030941\n",
      "Train Epoch: 100 [1920/1984 (97%)]\tLoss: 0.033668\n",
      "====> Epoch: 100 Average loss: 0.0006\n",
      "Train Epoch: 101 [0/1984 (0%)]\tLoss: 0.032641\n",
      "Train Epoch: 101 [640/1984 (32%)]\tLoss: 0.034509\n",
      "Train Epoch: 101 [1280/1984 (65%)]\tLoss: 0.019284\n",
      "Train Epoch: 101 [1920/1984 (97%)]\tLoss: 0.028768\n",
      "====> Epoch: 101 Average loss: 0.0006\n",
      "Train Epoch: 102 [0/1984 (0%)]\tLoss: 0.028731\n",
      "Train Epoch: 102 [640/1984 (32%)]\tLoss: 0.021555\n",
      "Train Epoch: 102 [1280/1984 (65%)]\tLoss: 0.073942\n",
      "Train Epoch: 102 [1920/1984 (97%)]\tLoss: 0.041285\n",
      "====> Epoch: 102 Average loss: 0.0007\n",
      "Train Epoch: 103 [0/1984 (0%)]\tLoss: 0.049837\n",
      "Train Epoch: 103 [640/1984 (32%)]\tLoss: 0.030335\n",
      "Train Epoch: 103 [1280/1984 (65%)]\tLoss: 0.026315\n",
      "Train Epoch: 103 [1920/1984 (97%)]\tLoss: 0.033637\n",
      "====> Epoch: 103 Average loss: 0.0006\n",
      "Train Epoch: 104 [0/1984 (0%)]\tLoss: 0.039949\n",
      "Train Epoch: 104 [640/1984 (32%)]\tLoss: 0.051785\n",
      "Train Epoch: 104 [1280/1984 (65%)]\tLoss: 0.025382\n",
      "Train Epoch: 104 [1920/1984 (97%)]\tLoss: 0.023490\n",
      "====> Epoch: 104 Average loss: 0.0005\n",
      "Train Epoch: 105 [0/1984 (0%)]\tLoss: 0.049790\n",
      "Train Epoch: 105 [640/1984 (32%)]\tLoss: 0.046817\n",
      "Train Epoch: 105 [1280/1984 (65%)]\tLoss: 0.030912\n",
      "Train Epoch: 105 [1920/1984 (97%)]\tLoss: 0.056046\n",
      "====> Epoch: 105 Average loss: 0.0006\n",
      "Train Epoch: 106 [0/1984 (0%)]\tLoss: 0.022544\n",
      "Train Epoch: 106 [640/1984 (32%)]\tLoss: 0.041604\n",
      "Train Epoch: 106 [1280/1984 (65%)]\tLoss: 0.021288\n",
      "Train Epoch: 106 [1920/1984 (97%)]\tLoss: 0.043413\n",
      "====> Epoch: 106 Average loss: 0.0006\n",
      "Train Epoch: 107 [0/1984 (0%)]\tLoss: 0.040032\n",
      "Train Epoch: 107 [640/1984 (32%)]\tLoss: 0.024163\n",
      "Train Epoch: 107 [1280/1984 (65%)]\tLoss: 0.050729\n",
      "Train Epoch: 107 [1920/1984 (97%)]\tLoss: 0.050836\n",
      "====> Epoch: 107 Average loss: 0.0006\n",
      "Train Epoch: 108 [0/1984 (0%)]\tLoss: 0.024077\n",
      "Train Epoch: 108 [640/1984 (32%)]\tLoss: 0.032015\n",
      "Train Epoch: 108 [1280/1984 (65%)]\tLoss: 0.025842\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train Epoch: 108 [1920/1984 (97%)]\tLoss: 0.039204\n",
      "====> Epoch: 108 Average loss: 0.0005\n",
      "Train Epoch: 109 [0/1984 (0%)]\tLoss: 0.028589\n",
      "Train Epoch: 109 [640/1984 (32%)]\tLoss: 0.027494\n",
      "Train Epoch: 109 [1280/1984 (65%)]\tLoss: 0.073814\n",
      "Train Epoch: 109 [1920/1984 (97%)]\tLoss: 0.056628\n",
      "====> Epoch: 109 Average loss: 0.0006\n",
      "Train Epoch: 110 [0/1984 (0%)]\tLoss: 0.064858\n",
      "Train Epoch: 110 [640/1984 (32%)]\tLoss: 0.046731\n",
      "Train Epoch: 110 [1280/1984 (65%)]\tLoss: 0.045407\n",
      "Train Epoch: 110 [1920/1984 (97%)]\tLoss: 0.071925\n",
      "====> Epoch: 110 Average loss: 0.0006\n",
      "Train Epoch: 111 [0/1984 (0%)]\tLoss: 0.037800\n",
      "Train Epoch: 111 [640/1984 (32%)]\tLoss: 0.042021\n",
      "Train Epoch: 111 [1280/1984 (65%)]\tLoss: 0.037903\n",
      "Train Epoch: 111 [1920/1984 (97%)]\tLoss: 0.029890\n",
      "====> Epoch: 111 Average loss: 0.0006\n",
      "Train Epoch: 112 [0/1984 (0%)]\tLoss: 0.014813\n",
      "Train Epoch: 112 [640/1984 (32%)]\tLoss: 0.018630\n",
      "Train Epoch: 112 [1280/1984 (65%)]\tLoss: 0.018287\n",
      "Train Epoch: 112 [1920/1984 (97%)]\tLoss: 0.018812\n",
      "====> Epoch: 112 Average loss: 0.0005\n",
      "Train Epoch: 113 [0/1984 (0%)]\tLoss: 0.051915\n",
      "Train Epoch: 113 [640/1984 (32%)]\tLoss: 0.023641\n",
      "Train Epoch: 113 [1280/1984 (65%)]\tLoss: 0.016384\n",
      "Train Epoch: 113 [1920/1984 (97%)]\tLoss: 0.035246\n",
      "====> Epoch: 113 Average loss: 0.0005\n",
      "Train Epoch: 114 [0/1984 (0%)]\tLoss: 0.015025\n",
      "Train Epoch: 114 [640/1984 (32%)]\tLoss: 0.033220\n",
      "Train Epoch: 114 [1280/1984 (65%)]\tLoss: 0.024830\n",
      "Train Epoch: 114 [1920/1984 (97%)]\tLoss: 0.020164\n",
      "====> Epoch: 114 Average loss: 0.0005\n",
      "Train Epoch: 115 [0/1984 (0%)]\tLoss: 0.028906\n",
      "Train Epoch: 115 [640/1984 (32%)]\tLoss: 0.036409\n",
      "Train Epoch: 115 [1280/1984 (65%)]\tLoss: 0.041712\n",
      "Train Epoch: 115 [1920/1984 (97%)]\tLoss: 0.072820\n",
      "====> Epoch: 115 Average loss: 0.0005\n",
      "Train Epoch: 116 [0/1984 (0%)]\tLoss: 0.023015\n",
      "Train Epoch: 116 [640/1984 (32%)]\tLoss: 0.056049\n",
      "Train Epoch: 116 [1280/1984 (65%)]\tLoss: 0.039847\n",
      "Train Epoch: 116 [1920/1984 (97%)]\tLoss: 0.024491\n",
      "====> Epoch: 116 Average loss: 0.0005\n",
      "Train Epoch: 117 [0/1984 (0%)]\tLoss: 0.028830\n",
      "Train Epoch: 117 [640/1984 (32%)]\tLoss: 0.025584\n",
      "Train Epoch: 117 [1280/1984 (65%)]\tLoss: 0.030052\n",
      "Train Epoch: 117 [1920/1984 (97%)]\tLoss: 0.038960\n",
      "====> Epoch: 117 Average loss: 0.0005\n",
      "Train Epoch: 118 [0/1984 (0%)]\tLoss: 0.032887\n",
      "Train Epoch: 118 [640/1984 (32%)]\tLoss: 0.043449\n",
      "Train Epoch: 118 [1280/1984 (65%)]\tLoss: 0.019924\n",
      "Train Epoch: 118 [1920/1984 (97%)]\tLoss: 0.048517\n",
      "====> Epoch: 118 Average loss: 0.0004\n",
      "Train Epoch: 119 [0/1984 (0%)]\tLoss: 0.027999\n",
      "Train Epoch: 119 [640/1984 (32%)]\tLoss: 0.040309\n",
      "Train Epoch: 119 [1280/1984 (65%)]\tLoss: 0.025440\n",
      "Train Epoch: 119 [1920/1984 (97%)]\tLoss: 0.021280\n",
      "====> Epoch: 119 Average loss: 0.0005\n",
      "Train Epoch: 120 [0/1984 (0%)]\tLoss: 0.033752\n",
      "Train Epoch: 120 [640/1984 (32%)]\tLoss: 0.021350\n",
      "Train Epoch: 120 [1280/1984 (65%)]\tLoss: 0.029842\n",
      "Train Epoch: 120 [1920/1984 (97%)]\tLoss: 0.025320\n",
      "====> Epoch: 120 Average loss: 0.0004\n",
      "Train Epoch: 121 [0/1984 (0%)]\tLoss: 0.021419\n",
      "Train Epoch: 121 [640/1984 (32%)]\tLoss: 0.024011\n",
      "Train Epoch: 121 [1280/1984 (65%)]\tLoss: 0.024762\n",
      "Train Epoch: 121 [1920/1984 (97%)]\tLoss: 0.021892\n",
      "====> Epoch: 121 Average loss: 0.0005\n",
      "Train Epoch: 122 [0/1984 (0%)]\tLoss: 0.022156\n",
      "Train Epoch: 122 [640/1984 (32%)]\tLoss: 0.022931\n",
      "Train Epoch: 122 [1280/1984 (65%)]\tLoss: 0.044778\n",
      "Train Epoch: 122 [1920/1984 (97%)]\tLoss: 0.027223\n",
      "====> Epoch: 122 Average loss: 0.0004\n",
      "Train Epoch: 123 [0/1984 (0%)]\tLoss: 0.042169\n",
      "Train Epoch: 123 [640/1984 (32%)]\tLoss: 0.028047\n",
      "Train Epoch: 123 [1280/1984 (65%)]\tLoss: 0.023051\n",
      "Train Epoch: 123 [1920/1984 (97%)]\tLoss: 0.041274\n",
      "====> Epoch: 123 Average loss: 0.0006\n",
      "Train Epoch: 124 [0/1984 (0%)]\tLoss: 0.042772\n",
      "Train Epoch: 124 [640/1984 (32%)]\tLoss: 0.016289\n",
      "Train Epoch: 124 [1280/1984 (65%)]\tLoss: 0.026810\n",
      "Train Epoch: 124 [1920/1984 (97%)]\tLoss: 0.030033\n",
      "====> Epoch: 124 Average loss: 0.0004\n",
      "Train Epoch: 125 [0/1984 (0%)]\tLoss: 0.033134\n",
      "Train Epoch: 125 [640/1984 (32%)]\tLoss: 0.019614\n",
      "Train Epoch: 125 [1280/1984 (65%)]\tLoss: 0.026381\n",
      "Train Epoch: 125 [1920/1984 (97%)]\tLoss: 0.034066\n",
      "====> Epoch: 125 Average loss: 0.0005\n",
      "Train Epoch: 126 [0/1984 (0%)]\tLoss: 0.030628\n",
      "Train Epoch: 126 [640/1984 (32%)]\tLoss: 0.035293\n",
      "Train Epoch: 126 [1280/1984 (65%)]\tLoss: 0.092584\n",
      "Train Epoch: 126 [1920/1984 (97%)]\tLoss: 0.043776\n",
      "====> Epoch: 126 Average loss: 0.0005\n",
      "Train Epoch: 127 [0/1984 (0%)]\tLoss: 0.053530\n",
      "Train Epoch: 127 [640/1984 (32%)]\tLoss: 0.017546\n",
      "Train Epoch: 127 [1280/1984 (65%)]\tLoss: 0.011335\n",
      "Train Epoch: 127 [1920/1984 (97%)]\tLoss: 0.027105\n",
      "====> Epoch: 127 Average loss: 0.0004\n",
      "Train Epoch: 128 [0/1984 (0%)]\tLoss: 0.020453\n",
      "Train Epoch: 128 [640/1984 (32%)]\tLoss: 0.041675\n",
      "Train Epoch: 128 [1280/1984 (65%)]\tLoss: 0.024666\n",
      "Train Epoch: 128 [1920/1984 (97%)]\tLoss: 0.026463\n",
      "====> Epoch: 128 Average loss: 0.0004\n",
      "Train Epoch: 129 [0/1984 (0%)]\tLoss: 0.015422\n",
      "Train Epoch: 129 [640/1984 (32%)]\tLoss: 0.051326\n",
      "Train Epoch: 129 [1280/1984 (65%)]\tLoss: 0.029329\n",
      "Train Epoch: 129 [1920/1984 (97%)]\tLoss: 0.022253\n",
      "====> Epoch: 129 Average loss: 0.0004\n",
      "Train Epoch: 130 [0/1984 (0%)]\tLoss: 0.028281\n",
      "Train Epoch: 130 [640/1984 (32%)]\tLoss: 0.021080\n",
      "Train Epoch: 130 [1280/1984 (65%)]\tLoss: 0.033694\n",
      "Train Epoch: 130 [1920/1984 (97%)]\tLoss: 0.018915\n",
      "====> Epoch: 130 Average loss: 0.0004\n",
      "Train Epoch: 131 [0/1984 (0%)]\tLoss: 0.017813\n",
      "Train Epoch: 131 [640/1984 (32%)]\tLoss: 0.010844\n",
      "Train Epoch: 131 [1280/1984 (65%)]\tLoss: 0.039467\n",
      "Train Epoch: 131 [1920/1984 (97%)]\tLoss: 0.018035\n",
      "====> Epoch: 131 Average loss: 0.0004\n",
      "Train Epoch: 132 [0/1984 (0%)]\tLoss: 0.022924\n",
      "Train Epoch: 132 [640/1984 (32%)]\tLoss: 0.033801\n",
      "Train Epoch: 132 [1280/1984 (65%)]\tLoss: 0.028964\n",
      "Train Epoch: 132 [1920/1984 (97%)]\tLoss: 0.019793\n",
      "====> Epoch: 132 Average loss: 0.0004\n",
      "Train Epoch: 133 [0/1984 (0%)]\tLoss: 0.044081\n",
      "Train Epoch: 133 [640/1984 (32%)]\tLoss: 0.020844\n",
      "Train Epoch: 133 [1280/1984 (65%)]\tLoss: 0.027205\n",
      "Train Epoch: 133 [1920/1984 (97%)]\tLoss: 0.029657\n",
      "====> Epoch: 133 Average loss: 0.0004\n",
      "Train Epoch: 134 [0/1984 (0%)]\tLoss: 0.023028\n",
      "Train Epoch: 134 [640/1984 (32%)]\tLoss: 0.054511\n",
      "Train Epoch: 134 [1280/1984 (65%)]\tLoss: 0.026027\n",
      "Train Epoch: 134 [1920/1984 (97%)]\tLoss: 0.013536\n",
      "====> Epoch: 134 Average loss: 0.0004\n",
      "Train Epoch: 135 [0/1984 (0%)]\tLoss: 0.030628\n",
      "Train Epoch: 135 [640/1984 (32%)]\tLoss: 0.019815\n",
      "Train Epoch: 135 [1280/1984 (65%)]\tLoss: 0.029060\n",
      "Train Epoch: 135 [1920/1984 (97%)]\tLoss: 0.021869\n",
      "====> Epoch: 135 Average loss: 0.0004\n",
      "Train Epoch: 136 [0/1984 (0%)]\tLoss: 0.020307\n",
      "Train Epoch: 136 [640/1984 (32%)]\tLoss: 0.009891\n",
      "Train Epoch: 136 [1280/1984 (65%)]\tLoss: 0.031101\n",
      "Train Epoch: 136 [1920/1984 (97%)]\tLoss: 0.023863\n",
      "====> Epoch: 136 Average loss: 0.0003\n",
      "Train Epoch: 137 [0/1984 (0%)]\tLoss: 0.021666\n",
      "Train Epoch: 137 [640/1984 (32%)]\tLoss: 0.036785\n",
      "Train Epoch: 137 [1280/1984 (65%)]\tLoss: 0.016773\n",
      "Train Epoch: 137 [1920/1984 (97%)]\tLoss: 0.015128\n",
      "====> Epoch: 137 Average loss: 0.0004\n",
      "Train Epoch: 138 [0/1984 (0%)]\tLoss: 0.023920\n",
      "Train Epoch: 138 [640/1984 (32%)]\tLoss: 0.017447\n",
      "Train Epoch: 138 [1280/1984 (65%)]\tLoss: 0.031689\n",
      "Train Epoch: 138 [1920/1984 (97%)]\tLoss: 0.011528\n",
      "====> Epoch: 138 Average loss: 0.0004\n",
      "Train Epoch: 139 [0/1984 (0%)]\tLoss: 0.026099\n",
      "Train Epoch: 139 [640/1984 (32%)]\tLoss: 0.038621\n",
      "Train Epoch: 139 [1280/1984 (65%)]\tLoss: 0.022406\n",
      "Train Epoch: 139 [1920/1984 (97%)]\tLoss: 0.028016\n",
      "====> Epoch: 139 Average loss: 0.0004\n",
      "Train Epoch: 140 [0/1984 (0%)]\tLoss: 0.024552\n",
      "Train Epoch: 140 [640/1984 (32%)]\tLoss: 0.019044\n",
      "Train Epoch: 140 [1280/1984 (65%)]\tLoss: 0.014400\n",
      "Train Epoch: 140 [1920/1984 (97%)]\tLoss: 0.016849\n",
      "====> Epoch: 140 Average loss: 0.0004\n",
      "Train Epoch: 141 [0/1984 (0%)]\tLoss: 0.044828\n",
      "Train Epoch: 141 [640/1984 (32%)]\tLoss: 0.018509\n",
      "Train Epoch: 141 [1280/1984 (65%)]\tLoss: 0.024492\n",
      "Train Epoch: 141 [1920/1984 (97%)]\tLoss: 0.016940\n",
      "====> Epoch: 141 Average loss: 0.0003\n",
      "Train Epoch: 142 [0/1984 (0%)]\tLoss: 0.020215\n",
      "Train Epoch: 142 [640/1984 (32%)]\tLoss: 0.008939\n",
      "Train Epoch: 142 [1280/1984 (65%)]\tLoss: 0.036311\n",
      "Train Epoch: 142 [1920/1984 (97%)]\tLoss: 0.012459\n",
      "====> Epoch: 142 Average loss: 0.0003\n",
      "Train Epoch: 143 [0/1984 (0%)]\tLoss: 0.030049\n",
      "Train Epoch: 143 [640/1984 (32%)]\tLoss: 0.015871\n",
      "Train Epoch: 143 [1280/1984 (65%)]\tLoss: 0.008530\n",
      "Train Epoch: 143 [1920/1984 (97%)]\tLoss: 0.037335\n",
      "====> Epoch: 143 Average loss: 0.0004\n",
      "Train Epoch: 144 [0/1984 (0%)]\tLoss: 0.017048\n",
      "Train Epoch: 144 [640/1984 (32%)]\tLoss: 0.018615\n",
      "Train Epoch: 144 [1280/1984 (65%)]\tLoss: 0.034288\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train Epoch: 144 [1920/1984 (97%)]\tLoss: 0.048250\n",
      "====> Epoch: 144 Average loss: 0.0003\n",
      "Train Epoch: 145 [0/1984 (0%)]\tLoss: 0.050504\n",
      "Train Epoch: 145 [640/1984 (32%)]\tLoss: 0.018548\n",
      "Train Epoch: 145 [1280/1984 (65%)]\tLoss: 0.020607\n",
      "Train Epoch: 145 [1920/1984 (97%)]\tLoss: 0.016392\n",
      "====> Epoch: 145 Average loss: 0.0003\n",
      "Train Epoch: 146 [0/1984 (0%)]\tLoss: 0.036506\n",
      "Train Epoch: 146 [640/1984 (32%)]\tLoss: 0.016026\n",
      "Train Epoch: 146 [1280/1984 (65%)]\tLoss: 0.030340\n",
      "Train Epoch: 146 [1920/1984 (97%)]\tLoss: 0.021115\n",
      "====> Epoch: 146 Average loss: 0.0004\n",
      "Train Epoch: 147 [0/1984 (0%)]\tLoss: 0.035895\n",
      "Train Epoch: 147 [640/1984 (32%)]\tLoss: 0.019268\n",
      "Train Epoch: 147 [1280/1984 (65%)]\tLoss: 0.026358\n",
      "Train Epoch: 147 [1920/1984 (97%)]\tLoss: 0.014084\n",
      "====> Epoch: 147 Average loss: 0.0003\n",
      "Train Epoch: 148 [0/1984 (0%)]\tLoss: 0.024125\n",
      "Train Epoch: 148 [640/1984 (32%)]\tLoss: 0.012229\n",
      "Train Epoch: 148 [1280/1984 (65%)]\tLoss: 0.017689\n",
      "Train Epoch: 148 [1920/1984 (97%)]\tLoss: 0.017396\n",
      "====> Epoch: 148 Average loss: 0.0003\n",
      "Train Epoch: 149 [0/1984 (0%)]\tLoss: 0.010106\n",
      "Train Epoch: 149 [640/1984 (32%)]\tLoss: 0.017044\n",
      "Train Epoch: 149 [1280/1984 (65%)]\tLoss: 0.012028\n",
      "Train Epoch: 149 [1920/1984 (97%)]\tLoss: 0.028397\n",
      "====> Epoch: 149 Average loss: 0.0003\n"
     ]
    }
   ],
   "source": [
    "# Training the ensemble of FC ResNets\n",
    "for epoch in range(num_epochs):\n",
    "    train_loss = train_single_epoch(\n",
    "        epoch, model, dl_train_loader, optimizer, device\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Visualising softmax entropy\n",
    "model.eval()\n",
    "\n",
    "domain = 3\n",
    "x_lin = np.linspace(-domain+0.5, domain+0.5, 100)\n",
    "y_lin = np.linspace(-domain, domain, 100)\n",
    "\n",
    "xx, yy = np.meshgrid(x_lin, y_lin)\n",
    "X_grid = np.column_stack([xx.flatten(), yy.flatten()])\n",
    "\n",
    "X_vis, y_vis = datasets.make_moons(n_samples=1000, noise=noise)\n",
    "mask = y_vis.astype(bool)\n",
    "\n",
    "X_grid = torch.from_numpy(X_grid).float().to(device)\n",
    "\n",
    "with torch.no_grad():\n",
    "    output = model(X_grid)\n",
    "    uncertainty = entropy(output)\n",
    "\n",
    "z = - uncertainty.cpu().numpy().reshape(xx.shape)\n",
    "\n",
    "plt.figure()\n",
    "plt.contourf(x_lin, y_lin, z, cmap='cividis')\n",
    "\n",
    "plt.scatter(X_vis[mask,0], X_vis[mask,1], s=10)\n",
    "plt.scatter(X_vis[~mask,0], X_vis[~mask,1], s=10)\n",
    "plt.savefig(\"two_moons_softmax_entropy_sn.png\", bbox_inches=\"tight\")\n",
    "plt.savefig(\"two_moons_softmax_entropy_sn.pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 32/32 [00:00<00:00, 494.48it/s]\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "0"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "embeddings, labels = gmm_utils.get_embeddings(model, dl_train_no_drop_loader, features, torch.double, device, device)\n",
    "\n",
    "gda, jitter_eps = gmm_utils.gmm_fit(embeddings, labels, 2)\n",
    "jitter_eps"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 32/32 [00:00<00:00, 403.32it/s]\n"
     ]
    }
   ],
   "source": [
    "train_log_probs_B_Y, train_labels = gmm_utils.gmm_evaluate(model, gda, dl_train_no_drop_loader, device, 2, device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_densities = torch.logsumexp(train_log_probs_B_Y, dim=-1)\n",
    "train_min_density = train_densities.min().item()\n",
    "train_median_density = torch.mean(train_densities).item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "model.eval()\n",
    "\n",
    "domain = 3\n",
    "x_lin = np.linspace(-domain+0.5, domain+0.5, 100)\n",
    "y_lin = np.linspace(-domain, domain, 100)\n",
    "\n",
    "xx, yy = np.meshgrid(x_lin, y_lin)\n",
    "X_grid = np.column_stack([xx.flatten(), yy.flatten()])\n",
    "\n",
    "X_vis, y_vis = datasets.make_moons(n_samples=1000, noise=noise)\n",
    "mask = y_vis.astype(bool)\n",
    "            \n",
    "X_grid = torch.from_numpy(X_grid).float()\n",
    "X_grid = X_grid.cuda() if torch.cuda.is_available() else X_grid\n",
    "\n",
    "with torch.no_grad():\n",
    "    log_probs_B_Y = gmm_utils.gmm_forward(model, gda, X_grid.cuda())\n",
    "    density = torch.logsumexp(log_probs_B_Y, dim=-1) - train_min_density\n",
    "    uncertainty = density\n",
    "        \n",
    "z = uncertainty.cpu().numpy().reshape(xx.shape)\n",
    "\n",
    "z_ood_percentiles = np.percentile(z[z<0],[0, 10, 20, 30, 40, 50, 60, 70, 80, 90])\n",
    "z_id_percentiles = np.percentile(z[z>=0],[0, 10, 20, 30, 40, 50, 60, 70, 80, 90])\n",
    "\n",
    "zmin = z[np.isfinite(z)].min()\n",
    "zmax = z[np.isfinite(z)].max()\n",
    "z[z==np.inf] = zmax\n",
    "z[z==-np.inf] = zmin\n",
    "\n",
    "plt.figure()\n",
    "plt.contourf(x_lin, y_lin, np.clip(z,-1, zmax)/zmax, cmap='cividis')\n",
    "\n",
    "plt.savefig(\"two_moons_ddu_sn.png\", bbox_inches=\"tight\")\n",
    "plt.savefig(\"two_moons_ddu_sn.pdf\", bbox_inches=\"tight\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
