{
 "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",
    "from torchvision import transforms as tf\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": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n_img 3\n"
     ]
    }
   ],
   "source": [
    "# Hyperparameters for backbone/encoder\n",
    "NUM_INPUT_CHANNELS = 1\n",
    "NUM_CHARS = 9\n",
    "NUM_CLASSES = 300\n",
    "N_DIMS=1\n",
    "DIM_OUTPUT = 10\n",
    "DROPOUT = 0.20\n",
    "# Hyperparameters for linear classifier training\n",
    "BS = 500\n",
    "NUM_EPOCHS = 100\n",
    "SEED = 21\n",
    "LR = 1e-3\n",
    "DEVICE = 'cuda:0' if torch.cuda.is_available() else 'cpu'\n",
    "MAX_NUM_CLASSES = 55\n",
    "ALPHAS = get_dim_mix_masks((1, 84, 84))"
   ]
  },
  {
   "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)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Alphas set to tensor([[[[[0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           ...,\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.]]],\n",
      "\n",
      "\n",
      "         [[[1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           ...,\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.]]]],\n",
      "\n",
      "\n",
      "\n",
      "        [[[[0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           ...,\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.]]],\n",
      "\n",
      "\n",
      "         [[[1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           ...,\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.]]]],\n",
      "\n",
      "\n",
      "\n",
      "        [[[[0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           ...,\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.]]],\n",
      "\n",
      "\n",
      "         [[[1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           ...,\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.]]]],\n",
      "\n",
      "\n",
      "\n",
      "        ...,\n",
      "\n",
      "\n",
      "\n",
      "        [[[[0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           ...,\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.]]],\n",
      "\n",
      "\n",
      "         [[[1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           ...,\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.]]]],\n",
      "\n",
      "\n",
      "\n",
      "        [[[[0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           ...,\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.]]],\n",
      "\n",
      "\n",
      "         [[[1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           ...,\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.]]]],\n",
      "\n",
      "\n",
      "\n",
      "        [[[[0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           [0., 0., 0.,  ..., 0., 0., 0.],\n",
      "           ...,\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.],\n",
      "           [0., 0., 0.,  ..., 1., 1., 1.]]],\n",
      "\n",
      "\n",
      "         [[[1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           [1., 1., 1.,  ..., 1., 1., 1.],\n",
      "           ...,\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.],\n",
      "           [1., 1., 1.,  ..., 0., 0., 0.]]]]])\n",
      "35.394439697265625\n"
     ]
    }
   ],
   "source": [
    "# Define model\n",
    "model = RatioCriticImageBilinearCoB(dim_input=N_DIMS, dim_output=DIM_OUTPUT, num_input_channels=1, dropout=DROPOUT, alphas=ALPHAS, num_classes=NUM_CLASSES)\n",
    "\n",
    "# Load model weights\n",
    "checkpoint = torch.load('../results/omniglot_ch9_K8_lr0.001_annealFalse_bs500_final300_sameencTrue/models/model_best.pth')\n",
    "model.load_state_dict(checkpoint['model_state_dict'])\n",
    "print(checkpoint['kl_est'])\n",
    "# Define backbone from model\n",
    "backbone = model.g"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define transforms\n",
    "transforms = None # tf.Compose([tf.ToTensor()])\n",
    "\n",
    "# Define dataset & dataloader\n",
    "train_ds = SpatialOmniDataset('../data/omniglot/multiomniglot_trn_{}.npz'.format(NUM_CHARS), transforms=transforms)\n",
    "\n",
    "val_ds = SpatialOmniDataset('../data/omniglot/multiomniglot_val_{}.npz'.format(NUM_CHARS), transforms=transforms)\n",
    "\n",
    "test_ds = SpatialOmniDataset('../data/omniglot/multiomniglot_tst_{}.npz'.format(NUM_CHARS), transforms=transforms)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "NUM_SAMPLES = len(train_ds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define dataloader\n",
    "train_dl = DataLoader(train_ds, batch_size=BS, shuffle=True)\n",
    "val_dl = DataLoader(val_ds, batch_size=BS, shuffle=False)\n",
    "test_dl = DataLoader(test_ds, batch_size=BS*2, shuffle=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "# For masking out logits / over-parametrization\n",
    "labels = train_ds.labels\n",
    "label_shape = labels.shape\n",
    "num_classes_per_problem = np.array([len(np.unique(labels[:, i])) for i in range(label_shape[1])])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([55, 52, 48, 47, 46, 43, 42, 41, 41])"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "num_classes_per_problem"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define linear classifier\n",
    "clfs = [torch.nn.Linear(model.g.fc_out_features, out_dim) for out_dim in num_classes_per_problem]\n",
    "\n",
    "# Define optimizer\n",
    "# optim = torch.optim.LBFGS(clf.parameters(), lr=LR)\n",
    "optims = [torch.optim.Adam(clfs[j].parameters(), lr=LR) for j in range(len(clfs))]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1440x360 with 4 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Set up viz\n",
    "fig, [ax1,ax2,ax3,ax4] = plt.subplots(1, 4,figsize=(20,5))\n",
    "\n",
    "train_loss_plt, = ax1.plot([0,1],[0,1],label='Test Loss')\n",
    "test_loss_plt, = ax2.plot([0,1],[0,1],label='Train Loss')\n",
    "train_acc_plt, = ax3.plot([0,1],[0,1],label='Train Accuracy')\n",
    "test_acc_plt, = ax4.plot([0,1],[0,1],label='Test Accuracy')\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,20])\n",
    "\n",
    "ax2.set_xlabel(\"Iteration\")\n",
    "ax2.set_ylabel(\"Test Loss\")\n",
    "ax2.set_xlim([0,NUM_EPOCHS*(NUM_SAMPLES//BS)])\n",
    "ax2.set_ylim([0,20])\n",
    "\n",
    "ax3.set_xlabel(\"Iteration\")\n",
    "ax3.set_ylabel(\"Train Accuracy\")\n",
    "ax3.set_xlim([0,NUM_EPOCHS*(NUM_SAMPLES//BS)])\n",
    "ax3.set_ylim([0,1])\n",
    "\n",
    "ax4.set_xlabel(\"Iteration\")\n",
    "ax4.set_ylabel(\"Test Accuracy\")\n",
    "ax4.set_xlim([0,NUM_EPOCHS*(NUM_SAMPLES//BS)])\n",
    "ax4.set_ylim([0,1])\n",
    "\n",
    "plt.tight_layout()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "def models_train(models):\n",
    "    for m in models:\n",
    "        m.train()\n",
    "        \n",
    "def models_eval(models):\n",
    "    for m in models:\n",
    "        m.eval()\n",
    "        \n",
    "# Training functions\n",
    "def accuracy_fn1(log_probs, true_labels):\n",
    "    accuracy = 0.0\n",
    "    \n",
    "    for i in range(len(log_probs)):\n",
    "        log_prob = log_probs[i]\n",
    "        labels = true_labels[i]\n",
    "        predictions = torch.argmax(log_prob, dim=-1)\n",
    "        accuracy += (torch.sum(predictions == labels) / float(labels.numel()))\n",
    "    \n",
    "    return accuracy / float(len(log_probs))\n",
    "\n",
    "def accuracy_fn(log_probs, true_labels):\n",
    "    masked_log_probs = log_probs.clone()\n",
    "    if NUM_CHARS > 1:\n",
    "        masked_log_probs = mask_extra_dims(masked_log_probs, zero_mask_output=False)\n",
    "\n",
    "    predictions = torch.argmax(masked_log_probs, dim=-1) # (n, num_classification_problems)\n",
    "    accuracy = torch.sum(predictions == true_labels) / torch.numel(true_labels)\n",
    "    \n",
    "    \n",
    "    return accuracy\n",
    "\n",
    "def nll_loss(log_probs, one_hot_labels):\n",
    "    masked_log_probs = log_probs.clone()\n",
    "     \n",
    "    if NUM_CHARS > 1:\n",
    "        masked_log_probs = mask_extra_dims(masked_log_probs, zero_mask_output=True)\n",
    "    \n",
    "    \n",
    "    log_likelihood = torch.sum(one_hot_labels * masked_log_probs, axis=-1)\n",
    "    av_nll = -torch.mean(log_likelihood)\n",
    "    \n",
    "    return av_nll\n",
    "\n",
    "def mask_extra_dims(log_probs, zero_mask_output):\n",
    "    \"\"\"For each classification problem, we are outputting the same number of logits, despite the fact\n",
    "    that different problems have different number of classes. Thus, for each problem,\n",
    "    we mask out unnecessary logits (and renormalize). This leads to (increased) over-parameterisation\n",
    "    but given that the usual softmax cross-entropy loss is overparameterised to start with, it's\n",
    "    probably not a big deal.\n",
    "    \"\"\"\n",
    "    \n",
    "    class_mask = torch.zeros((NUM_CHARS, MAX_NUM_CLASSES), dtype=torch.float32, device=log_probs.device)\n",
    "    neg_infs = torch.zeros_like(class_mask, device=log_probs.device)\n",
    "    \n",
    "    for i in range(NUM_CHARS):\n",
    "        class_mask[i, :num_classes_per_problem[i]] = 1.0\n",
    "        neg_infs[i, num_classes_per_problem[i]:] = -np.inf\n",
    "    \n",
    "    # renormalise the non-masked entries\n",
    "    masked_log_probs = (class_mask * log_probs.clone().unsqueeze(1)) + neg_infs  # (n, num_classification_problems, max_num_classes)\n",
    "    masked_log_probs = masked_log_probs - (torch.logsumexp(masked_log_probs, dim=-1, keepdim=True) * class_mask)\n",
    "    \n",
    "    if zero_mask_output:\n",
    "        masked_log_probs = torch.where(torch.isinf(masked_log_probs), torch.zeros_like(masked_log_probs), masked_log_probs)\n",
    "    \n",
    "#     return log_probs\n",
    "    return masked_log_probs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1440x360 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [49:38<00:00, 29.79s/it]\n"
     ]
    }
   ],
   "source": [
    "backbone.eval()\n",
    "\n",
    "train_loss_store = []\n",
    "test_loss_store = []\n",
    "train_acc_store = []\n",
    "test_acc_store = []\n",
    "\n",
    "if torch.cuda.is_available():\n",
    "    backbone = backbone.to(DEVICE)\n",
    "    clfs = [clf.to(DEVICE) for clf in clfs]\n",
    "    \n",
    "i = 0\n",
    "# Define custom NLL loss\n",
    "loss_crit = torch.nn.CrossEntropyLoss()\n",
    "\n",
    "\n",
    "for epoch in trange(NUM_EPOCHS):\n",
    "    models_train(clfs)\n",
    "    for (u, _, labels) in iter(train_dl):\n",
    "        i += 1\n",
    "        if torch.cuda.is_available():\n",
    "            u, labels = u.to(DEVICE), labels.to(DEVICE)\n",
    "#         one_hot_labels = torch.nn.functional.one_hot(labels, MAX_NUM_CLASSES)\n",
    "        \n",
    "        # Splitting labels for the grid\n",
    "        labels = [labels[:,i] for i in range(labels.shape[-1])]\n",
    "#         labels = labels.unbind(dim=-1)\n",
    "        \n",
    "        for optim in optims:\n",
    "            optim.zero_grad()\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            g_u = backbone(u)\n",
    "            \n",
    "        outputs = [clf(g_u) for clf in clfs]\n",
    "        loss = sum(list(map(loss_crit, outputs, labels)))\n",
    "        loss.backward()\n",
    "        \n",
    "        for optim in optims:\n",
    "            optim.step()\n",
    "        \n",
    "        accuracy = accuracy_fn1(outputs, labels)\n",
    "#         loss = nll_loss(outputs, one_hot_labels)\n",
    "        train_loss_store.append(loss.item())\n",
    "        train_acc_store.append(accuracy.item())\n",
    "    \n",
    "    # Validation\n",
    "    with torch.no_grad():\n",
    "        models_eval(clfs)\n",
    "        test_epoch_loss = []\n",
    "        test_epoch_acc = []\n",
    "        \n",
    "        for (u, _, labels) in iter(val_dl):\n",
    "            # CUDA\n",
    "            if torch.cuda.is_available():\n",
    "                u, labels = u.to(DEVICE), labels.to(DEVICE)\n",
    "\n",
    "            # Splitting labels for the grid\n",
    "            labels = labels.unbind(dim=-1)\n",
    "            g_u = backbone(u)\n",
    "            outputs = [clf(g_u) for clf in clfs]\n",
    "            loss = sum(list(map(loss_crit, outputs, labels)))\n",
    "\n",
    "            accuracy = accuracy_fn1(outputs, labels)\n",
    "            \n",
    "            test_epoch_loss.append(loss.item())\n",
    "            test_epoch_acc.append(accuracy.item())\n",
    "        \n",
    "        test_loss_store.append(np.mean(test_epoch_loss))\n",
    "        test_acc_store.append(np.mean(test_epoch_acc))\n",
    "        \n",
    "    train_loss_plt.set_data(range(len(train_loss_store)), train_loss_store)\n",
    "    ax1.set_xlim(0, len(train_loss_store))\n",
    "    train_acc_plt.set_data(range(len(train_acc_store)), train_acc_store)\n",
    "    ax3.set_xlim(0, len(train_acc_store))\n",
    "    ax3.set_ylim(0, 1)\n",
    "    test_loss_plt.set_data(range(len(test_loss_store)), test_loss_store)\n",
    "    ax2.set_xlim(0, len(test_loss_store))\n",
    "    test_acc_plt.set_data(range(len(test_acc_store)), test_acc_store)\n",
    "    ax4.set_xlim(0, len(test_acc_store))\n",
    "    ax4.set_ylim(0, 1)\n",
    "    \n",
    "    clear_output(wait=True)\n",
    "    display(fig)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.0 0.9999666690826416\n"
     ]
    }
   ],
   "source": [
    "print(train_acc_store[-1], test_acc_store[-1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.10518804714083671\n",
      "0.9999777913093567\n"
     ]
    }
   ],
   "source": [
    "with torch.no_grad():\n",
    "    models_eval(clfs)\n",
    "    test_epoch_loss = []\n",
    "    test_epoch_acc = []\n",
    "\n",
    "    for (u, _, labels) in iter(test_dl):\n",
    "        # CUDA\n",
    "        if torch.cuda.is_available():\n",
    "            u, labels = u.to(DEVICE), labels.to(DEVICE)\n",
    "\n",
    "        # Splitting labels for the grid\n",
    "        labels = labels.unbind(dim=-1)\n",
    "        g_u = backbone(u)\n",
    "        outputs = [clf(g_u) for clf in clfs]\n",
    "        loss = sum(list(map(loss_crit, outputs, labels)))\n",
    "\n",
    "        accuracy = accuracy_fn1(outputs, labels)\n",
    "\n",
    "        test_epoch_loss.append(loss.item())\n",
    "        test_epoch_acc.append(accuracy.item())\n",
    "\n",
    "    print(np.mean(test_epoch_loss))\n",
    "    print(np.mean(test_epoch_acc))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "#### LBFGS #####\n",
    "\n",
    "backbone.eval()\n",
    "\n",
    "train_loss_store = []\n",
    "test_loss_store = []\n",
    "train_acc_store = []\n",
    "test_acc_store = []\n",
    "\n",
    "if torch.cuda.is_available():\n",
    "    backbone = backbone.to(DEVICE)\n",
    "    clfs = [clf.to(DEVICE) for clf in clfs]\n",
    "    \n",
    "i = 0\n",
    "\n",
    "for epoch in trange(NUM_EPOCHS):\n",
    "    clf.train()\n",
    "    for (u, _, labels) in iter(train_dl):\n",
    "        i += 1\n",
    "        if torch.cuda.is_available():\n",
    "            u, labels = u.to(DEVICE), labels.to(DEVICE)\n",
    "        one_hot_labels = torch.nn.functional.one_hot(labels, MAX_NUM_CLASSES)        \n",
    "        \n",
    "        def closure():\n",
    "            optim.zero_grad()\n",
    "            with torch.no_grad():\n",
    "                g_u = backbone(u)\n",
    "            outputs = clf(g_u)\n",
    "            loss = nll_loss(outputs, one_hot_labels)\n",
    "            loss.backward()\n",
    "            \n",
    "            return loss\n",
    "        \n",
    "        optim.step(closure)\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            g_u = backbone(u)\n",
    "            outputs = clf(g_u)\n",
    "            loss = nll_loss(outputs, one_hot_labels)\n",
    "            accuracy = accuracy_fn(outputs, labels)\n",
    "            \n",
    "        train_loss_store.append(loss.item())\n",
    "        train_acc_store.append(accuracy.item())\n",
    "    \n",
    "    # Validation\n",
    "    with torch.no_grad():\n",
    "        models_eval(clfs)\n",
    "        test_epoch_loss = []\n",
    "        test_epoch_acc = []\n",
    "        \n",
    "        for (u, _, labels) in iter(val_dl):\n",
    "            # CUDA\n",
    "            if torch.cuda.is_available():\n",
    "                u, labels = u.to(DEVICE), labels.to(DEVICE)\n",
    "\n",
    "            g_u = backbone(u)\n",
    "            outputs = clf(g_u)\n",
    "            loss = nll_loss(outputs, one_hot_labels)\n",
    "            accuracy = accuracy_fn1(outputs, labels)\n",
    "            \n",
    "            test_epoch_loss.append(loss.item())\n",
    "            test_epoch_acc.append(accuracy.item())\n",
    "        \n",
    "        test_loss_store.append(np.mean(test_epoch_loss))\n",
    "        test_acc_store.append(np.mean(test_epoch_acc))\n",
    "        \n",
    "    train_loss_plt.set_data(range(len(train_loss_store)), train_loss_store)\n",
    "    ax1.set_xlim(0, len(train_loss_store))\n",
    "    train_acc_plt.set_data(range(len(train_acc_store)), train_acc_store)\n",
    "    ax3.set_xlim(0, len(train_acc_store))\n",
    "    ax3.set_ylim(0, 1)\n",
    "    test_loss_plt.set_data(range(len(test_loss_store)), test_loss_store)\n",
    "    ax2.set_xlim(0, len(test_loss_store))\n",
    "    test_acc_plt.set_data(range(len(test_acc_store)), test_acc_store)\n",
    "    ax4.set_xlim(0, len(test_acc_store))\n",
    "    ax4.set_ylim(0, 1)\n",
    "    \n",
    "    clear_output(wait=True)\n",
    "    display(fig)"
   ]
  }
 ],
 "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
}
