{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "olympic-maximum",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "import torch.nn as nn\n",
    "import matplotlib.pyplot as plt\n",
    "from torchvision import transforms\n",
    "from torchvision.datasets import MNIST\n",
    "from copy import deepcopy\n",
    "from torchmetrics import Accuracy\n",
    "from adaptive import AdaptiveSelection, MaskLayer, MaskingPretrainer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "fixed-decline",
   "metadata": {},
   "outputs": [],
   "source": [
    "device = torch.device('cuda', 5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "refined-store",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load data\n",
    "class Flatten(object):\n",
    "    def __call__(self, pic):\n",
    "        return torch.flatten(pic)\n",
    "    \n",
    "mnist_dataset = MNIST('/tmp/mnist/', download=True, train=True,\n",
    "                      transform=transforms.Compose([transforms.ToTensor(), Flatten()]))\n",
    "images = mnist_dataset.data\n",
    "targets = mnist_dataset.targets\n",
    "np.random.seed(0)\n",
    "val_inds = np.sort(np.random.choice(len(images), size=10000, replace=False))\n",
    "train_inds = np.setdiff1d(np.arange(len(images)), val_inds)\n",
    "\n",
    "# Training dataset\n",
    "train_dataset = torch.utils.data.Subset(mnist_dataset, train_inds)\n",
    "\n",
    "# Validation dataset\n",
    "val_dataset = torch.utils.data.Subset(mnist_dataset, val_inds)\n",
    "\n",
    "# Test dataset\n",
    "test_dataset = MNIST('/tmp/mnist/', download=True, train=False,\n",
    "                     transform=transforms.Compose([transforms.ToTensor(), Flatten()]))\n",
    "\n",
    "# Set input/output dimensions\n",
    "d_in = 784\n",
    "d_out = 10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "earlier-coffee",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Number of features to select\n",
    "max_features = 25"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "dd3b1f67-8e29-46f8-8c6f-1556925e33c1",
   "metadata": {},
   "source": [
    "# Set up architectures"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "52f26224-44f4-40f7-ba38-8e87bd761dc9",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Predictor\n",
    "predictor = nn.Sequential(\n",
    "    nn.Linear(d_in * 2, 512),\n",
    "    nn.ReLU(),\n",
    "    nn.Linear(512, 512),\n",
    "    nn.ReLU(),\n",
    "    nn.Linear(512, d_out))\n",
    "\n",
    "# Selector\n",
    "selector = nn.Sequential(\n",
    "    nn.Linear(d_in * 2, 512),\n",
    "    nn.ReLU(),\n",
    "    nn.Linear(512, 512),\n",
    "    nn.ReLU(),\n",
    "    nn.Linear(512, d_in))\n",
    "\n",
    "# Tie weights\n",
    "selector[0] = predictor[0]\n",
    "selector[2] = predictor[2]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "perceived-blind",
   "metadata": {},
   "source": [
    "# Pretrain predictor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "separate-going",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--------Epoch 1--------\n",
      "Val loss = 0.4093\n",
      "\n",
      "--------Epoch 2--------\n",
      "Val loss = 0.4430\n",
      "\n",
      "--------Epoch 3--------\n",
      "Val loss = 0.4466\n",
      "\n",
      "--------Epoch 4--------\n",
      "Val loss = 0.4777\n",
      "\n",
      "--------Epoch 5--------\n",
      "Val loss = 0.4711\n",
      "\n",
      "--------Epoch 6--------\n",
      "Val loss = 0.4792\n",
      "\n",
      "--------Epoch 7--------\n",
      "Val loss = 0.4865\n",
      "\n",
      "--------Epoch 8--------\n",
      "Val loss = 0.4898\n",
      "\n",
      "--------Epoch 9--------\n",
      "Val loss = 0.4894\n",
      "\n",
      "--------Epoch 10--------\n",
      "Val loss = 0.4921\n",
      "\n",
      "--------Epoch 11--------\n",
      "Val loss = 0.5021\n",
      "\n",
      "--------Epoch 12--------\n",
      "Val loss = 0.5107\n",
      "\n",
      "--------Epoch 13--------\n",
      "Val loss = 0.4962\n",
      "\n",
      "--------Epoch 14--------\n",
      "Val loss = 0.5034\n",
      "\n",
      "--------Epoch 15--------\n",
      "Val loss = 0.5185\n",
      "\n",
      "--------Epoch 16--------\n",
      "Val loss = 0.5098\n",
      "\n",
      "--------Epoch 17--------\n",
      "Val loss = 0.5186\n",
      "\n",
      "--------Epoch 18--------\n",
      "Val loss = 0.5058\n",
      "\n",
      "--------Epoch 19--------\n",
      "Val loss = 0.5198\n",
      "\n",
      "--------Epoch 20--------\n",
      "Val loss = 0.5290\n",
      "\n",
      "--------Epoch 21--------\n",
      "Val loss = 0.5206\n",
      "\n",
      "--------Epoch 22--------\n",
      "Val loss = 0.5215\n",
      "\n",
      "--------Epoch 23--------\n",
      "Val loss = 0.5271\n",
      "\n",
      "Epoch    23: reducing learning rate of group 0 to 2.0000e-04.\n",
      "--------Epoch 24--------\n",
      "Val loss = 0.5375\n",
      "\n",
      "--------Epoch 25--------\n",
      "Val loss = 0.5461\n",
      "\n",
      "--------Epoch 26--------\n",
      "Val loss = 0.5455\n",
      "\n",
      "--------Epoch 27--------\n",
      "Val loss = 0.5457\n",
      "\n",
      "--------Epoch 28--------\n",
      "Val loss = 0.5462\n",
      "\n",
      "--------Epoch 29--------\n",
      "Val loss = 0.5609\n",
      "\n",
      "--------Epoch 30--------\n",
      "Val loss = 0.5550\n",
      "\n",
      "--------Epoch 31--------\n",
      "Val loss = 0.5454\n",
      "\n",
      "--------Epoch 32--------\n",
      "Val loss = 0.5531\n",
      "\n",
      "Epoch    32: reducing learning rate of group 0 to 4.0000e-05.\n",
      "--------Epoch 33--------\n",
      "Val loss = 0.5551\n",
      "\n",
      "Stopping early at epoch 33\n"
     ]
    }
   ],
   "source": [
    "# Set up predictor pretraining\n",
    "mask_layer = MaskLayer(append=True)\n",
    "pretrain = MaskingPretrainer(predictor, mask_layer).to(device)\n",
    "\n",
    "# Pretrain\n",
    "pretrain.fit(train_dataset,\n",
    "             val_dataset,\n",
    "             mbsize=128,\n",
    "             lr=1e-3,\n",
    "             nepochs=100,\n",
    "             max_features=max_features,\n",
    "             loss_fn=nn.CrossEntropyLoss(),\n",
    "             val_loss_fn=Accuracy(),\n",
    "             val_loss_mode='max')\n",
    "\n",
    "# Save model\n",
    "predictor.cpu()\n",
    "torch.save(predictor, 'mnist_predictor_pretrained.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "abc41843-29c5-42c2-9a9a-eb0227eacd5e",
   "metadata": {},
   "outputs": [],
   "source": [
    "# # Load model\n",
    "# predictor = torch.load('mnist_predictor_pretrained.pt')\n",
    "# predictor.to(device)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ethical-terry",
   "metadata": {},
   "source": [
    "# Adaptive selection"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "identical-single",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Starting training with temp = 1.0000\n",
      "\n",
      "--------Epoch 1 (1 total)--------\n",
      "Val loss = 0.7609, Zero-temp loss = 0.7076\n",
      "\n",
      "--------Epoch 2 (2 total)--------\n",
      "Val loss = 0.7830, Zero-temp loss = 0.7225\n",
      "\n",
      "--------Epoch 3 (3 total)--------\n",
      "Val loss = 0.7970, Zero-temp loss = 0.7309\n",
      "\n",
      "--------Epoch 4 (4 total)--------\n",
      "Val loss = 0.8048, Zero-temp loss = 0.7321\n",
      "\n",
      "--------Epoch 5 (5 total)--------\n",
      "Val loss = 0.8100, Zero-temp loss = 0.7307\n",
      "\n",
      "--------Epoch 6 (6 total)--------\n",
      "Val loss = 0.8173, Zero-temp loss = 0.7296\n",
      "\n",
      "--------Epoch 7 (7 total)--------\n",
      "Val loss = 0.8253, Zero-temp loss = 0.7345\n",
      "\n",
      "--------Epoch 8 (8 total)--------\n",
      "Val loss = 0.8266, Zero-temp loss = 0.7289\n",
      "\n",
      "--------Epoch 9 (9 total)--------\n",
      "Val loss = 0.8315, Zero-temp loss = 0.7268\n",
      "\n",
      "--------Epoch 10 (10 total)--------\n",
      "Val loss = 0.8375, Zero-temp loss = 0.7265\n",
      "\n",
      "--------Epoch 11 (11 total)--------\n",
      "Val loss = 0.8380, Zero-temp loss = 0.7232\n",
      "\n",
      "--------Epoch 12 (12 total)--------\n",
      "Val loss = 0.8435, Zero-temp loss = 0.7209\n",
      "\n",
      "--------Epoch 13 (13 total)--------\n",
      "Val loss = 0.8457, Zero-temp loss = 0.7226\n",
      "\n",
      "--------Epoch 14 (14 total)--------\n",
      "Val loss = 0.8499, Zero-temp loss = 0.7187\n",
      "\n",
      "--------Epoch 15 (15 total)--------\n",
      "Val loss = 0.8551, Zero-temp loss = 0.7204\n",
      "\n",
      "--------Epoch 16 (16 total)--------\n",
      "Val loss = 0.8556, Zero-temp loss = 0.7165\n",
      "\n",
      "--------Epoch 17 (17 total)--------\n",
      "Val loss = 0.8618, Zero-temp loss = 0.7185\n",
      "\n",
      "--------Epoch 18 (18 total)--------\n",
      "Val loss = 0.8627, Zero-temp loss = 0.7182\n",
      "\n",
      "--------Epoch 19 (19 total)--------\n",
      "Val loss = 0.8651, Zero-temp loss = 0.7149\n",
      "\n",
      "--------Epoch 20 (20 total)--------\n",
      "Val loss = 0.8650, Zero-temp loss = 0.7110\n",
      "\n",
      "--------Epoch 21 (21 total)--------\n",
      "Val loss = 0.8703, Zero-temp loss = 0.7143\n",
      "\n",
      "--------Epoch 22 (22 total)--------\n",
      "Val loss = 0.8709, Zero-temp loss = 0.7060\n",
      "\n",
      "--------Epoch 23 (23 total)--------\n",
      "Val loss = 0.8694, Zero-temp loss = 0.7028\n",
      "\n",
      "--------Epoch 24 (24 total)--------\n",
      "Val loss = 0.8727, Zero-temp loss = 0.6993\n",
      "\n",
      "--------Epoch 25 (25 total)--------\n",
      "Val loss = 0.8734, Zero-temp loss = 0.6996\n",
      "\n",
      "--------Epoch 26 (26 total)--------\n",
      "Val loss = 0.8739, Zero-temp loss = 0.6991\n",
      "\n",
      "--------Epoch 27 (27 total)--------\n",
      "Val loss = 0.8771, Zero-temp loss = 0.6996\n",
      "\n",
      "--------Epoch 28 (28 total)--------\n",
      "Val loss = 0.8802, Zero-temp loss = 0.6976\n",
      "\n",
      "--------Epoch 29 (29 total)--------\n",
      "Val loss = 0.8801, Zero-temp loss = 0.6907\n",
      "\n",
      "--------Epoch 30 (30 total)--------\n",
      "Val loss = 0.8825, Zero-temp loss = 0.6935\n",
      "\n",
      "--------Epoch 31 (31 total)--------\n",
      "Val loss = 0.8851, Zero-temp loss = 0.6928\n",
      "\n",
      "--------Epoch 32 (32 total)--------\n",
      "Val loss = 0.8856, Zero-temp loss = 0.6921\n",
      "\n",
      "--------Epoch 33 (33 total)--------\n",
      "Val loss = 0.8861, Zero-temp loss = 0.6917\n",
      "\n",
      "--------Epoch 34 (34 total)--------\n",
      "Val loss = 0.8870, Zero-temp loss = 0.6882\n",
      "\n",
      "--------Epoch 35 (35 total)--------\n",
      "Val loss = 0.8886, Zero-temp loss = 0.6869\n",
      "\n",
      "--------Epoch 36 (36 total)--------\n",
      "Val loss = 0.8885, Zero-temp loss = 0.6848\n",
      "\n",
      "--------Epoch 37 (37 total)--------\n",
      "Val loss = 0.8897, Zero-temp loss = 0.6820\n",
      "\n",
      "--------Epoch 38 (38 total)--------\n",
      "Val loss = 0.8905, Zero-temp loss = 0.6856\n",
      "\n",
      "--------Epoch 39 (39 total)--------\n",
      "Val loss = 0.8918, Zero-temp loss = 0.6808\n",
      "\n",
      "--------Epoch 40 (40 total)--------\n",
      "Val loss = 0.8936, Zero-temp loss = 0.6776\n",
      "\n",
      "--------Epoch 41 (41 total)--------\n",
      "Val loss = 0.8929, Zero-temp loss = 0.6765\n",
      "\n",
      "--------Epoch 42 (42 total)--------\n",
      "Val loss = 0.8942, Zero-temp loss = 0.6783\n",
      "\n",
      "--------Epoch 43 (43 total)--------\n",
      "Val loss = 0.8930, Zero-temp loss = 0.6758\n",
      "\n",
      "--------Epoch 44 (44 total)--------\n",
      "Val loss = 0.8972, Zero-temp loss = 0.6799\n",
      "\n",
      "--------Epoch 45 (45 total)--------\n",
      "Val loss = 0.8953, Zero-temp loss = 0.6689\n",
      "\n",
      "--------Epoch 46 (46 total)--------\n",
      "Val loss = 0.8969, Zero-temp loss = 0.6748\n",
      "\n",
      "--------Epoch 47 (47 total)--------\n",
      "Val loss = 0.8987, Zero-temp loss = 0.6774\n",
      "\n",
      "--------Epoch 48 (48 total)--------\n",
      "Val loss = 0.8986, Zero-temp loss = 0.6691\n",
      "\n",
      "--------Epoch 49 (49 total)--------\n",
      "Val loss = 0.8990, Zero-temp loss = 0.6699\n",
      "\n",
      "--------Epoch 50 (50 total)--------\n",
      "Val loss = 0.9031, Zero-temp loss = 0.6738\n",
      "\n",
      "--------Epoch 51 (51 total)--------\n",
      "Val loss = 0.9031, Zero-temp loss = 0.6733\n",
      "\n",
      "--------Epoch 52 (52 total)--------\n",
      "Val loss = 0.9012, Zero-temp loss = 0.6688\n",
      "\n",
      "--------Epoch 53 (53 total)--------\n",
      "Val loss = 0.9033, Zero-temp loss = 0.6691\n",
      "\n",
      "--------Epoch 54 (54 total)--------\n",
      "Val loss = 0.9022, Zero-temp loss = 0.6611\n",
      "\n",
      "--------Epoch 55 (55 total)--------\n",
      "Val loss = 0.9041, Zero-temp loss = 0.6664\n",
      "\n",
      "--------Epoch 56 (56 total)--------\n",
      "Val loss = 0.9053, Zero-temp loss = 0.6662\n",
      "\n",
      "--------Epoch 57 (57 total)--------\n",
      "Val loss = 0.9064, Zero-temp loss = 0.6642\n",
      "\n",
      "--------Epoch 58 (58 total)--------\n",
      "Val loss = 0.9074, Zero-temp loss = 0.6665\n",
      "\n",
      "--------Epoch 59 (59 total)--------\n",
      "Val loss = 0.9049, Zero-temp loss = 0.6582\n",
      "\n",
      "--------Epoch 60 (60 total)--------\n",
      "Val loss = 0.9051, Zero-temp loss = 0.6598\n",
      "\n",
      "--------Epoch 61 (61 total)--------\n",
      "Val loss = 0.9087, Zero-temp loss = 0.6616\n",
      "\n",
      "--------Epoch 62 (62 total)--------\n",
      "Val loss = 0.9066, Zero-temp loss = 0.6592\n",
      "\n",
      "--------Epoch 63 (63 total)--------\n",
      "Val loss = 0.9060, Zero-temp loss = 0.6514\n",
      "\n",
      "--------Epoch 64 (64 total)--------\n",
      "Val loss = 0.9076, Zero-temp loss = 0.6580\n",
      "\n",
      "Epoch    64: reducing learning rate of group 0 to 2.0000e-04.\n",
      "--------Epoch 65 (65 total)--------\n",
      "Val loss = 0.9144, Zero-temp loss = 0.6643\n",
      "\n",
      "--------Epoch 66 (66 total)--------\n",
      "Val loss = 0.9175, Zero-temp loss = 0.6657\n",
      "\n",
      "--------Epoch 67 (67 total)--------\n",
      "Val loss = 0.9182, Zero-temp loss = 0.6695\n",
      "\n",
      "--------Epoch 68 (68 total)--------\n",
      "Val loss = 0.9190, Zero-temp loss = 0.6623\n",
      "\n",
      "--------Epoch 69 (69 total)--------\n",
      "Val loss = 0.9189, Zero-temp loss = 0.6646\n",
      "\n",
      "--------Epoch 70 (70 total)--------\n",
      "Val loss = 0.9205, Zero-temp loss = 0.6673\n",
      "\n",
      "--------Epoch 71 (71 total)--------\n",
      "Val loss = 0.9203, Zero-temp loss = 0.6614\n",
      "\n",
      "--------Epoch 72 (72 total)--------\n",
      "Val loss = 0.9211, Zero-temp loss = 0.6645\n",
      "\n",
      "--------Epoch 73 (73 total)--------\n",
      "Val loss = 0.9217, Zero-temp loss = 0.6634\n",
      "\n",
      "--------Epoch 74 (74 total)--------\n",
      "Val loss = 0.9217, Zero-temp loss = 0.6637\n",
      "\n",
      "--------Epoch 75 (75 total)--------\n",
      "Val loss = 0.9212, Zero-temp loss = 0.6631\n",
      "\n",
      "--------Epoch 76 (76 total)--------\n",
      "Val loss = 0.9222, Zero-temp loss = 0.6623\n",
      "\n",
      "--------Epoch 77 (77 total)--------\n",
      "Val loss = 0.9235, Zero-temp loss = 0.6666\n",
      "\n",
      "--------Epoch 78 (78 total)--------\n",
      "Val loss = 0.9221, Zero-temp loss = 0.6674\n",
      "\n",
      "--------Epoch 79 (79 total)--------\n",
      "Val loss = 0.9226, Zero-temp loss = 0.6628\n",
      "\n",
      "--------Epoch 80 (80 total)--------\n",
      "Val loss = 0.9242, Zero-temp loss = 0.6643\n",
      "\n",
      "--------Epoch 81 (81 total)--------\n",
      "Val loss = 0.9241, Zero-temp loss = 0.6644\n",
      "\n",
      "--------Epoch 82 (82 total)--------\n",
      "Val loss = 0.9254, Zero-temp loss = 0.6634\n",
      "\n",
      "--------Epoch 83 (83 total)--------\n",
      "Val loss = 0.9247, Zero-temp loss = 0.6579\n",
      "\n",
      "--------Epoch 84 (84 total)--------\n",
      "Val loss = 0.9236, Zero-temp loss = 0.6583\n",
      "\n",
      "--------Epoch 85 (85 total)--------\n",
      "Val loss = 0.9248, Zero-temp loss = 0.6546\n",
      "\n",
      "Epoch    85: reducing learning rate of group 0 to 4.0000e-05.\n",
      "--------Epoch 86 (86 total)--------\n",
      "Val loss = 0.9249, Zero-temp loss = 0.6592\n",
      "\n",
      "Stopping temp = 1.0000 at epoch 86\n",
      "\n",
      "Starting training with temp = 0.5623\n",
      "\n",
      "--------Epoch 1 (87 total)--------\n",
      "Val loss = 0.8546, Zero-temp loss = 0.7229\n",
      "\n",
      "--------Epoch 2 (88 total)--------\n",
      "Val loss = 0.8553, Zero-temp loss = 0.7240\n",
      "\n",
      "--------Epoch 3 (89 total)--------\n",
      "Val loss = 0.8582, Zero-temp loss = 0.7275\n",
      "\n",
      "--------Epoch 4 (90 total)--------\n",
      "Val loss = 0.8577, Zero-temp loss = 0.7291\n",
      "\n",
      "--------Epoch 5 (91 total)--------\n",
      "Val loss = 0.8587, Zero-temp loss = 0.7271\n",
      "\n",
      "--------Epoch 6 (92 total)--------\n",
      "Val loss = 0.8577, Zero-temp loss = 0.7240\n",
      "\n",
      "--------Epoch 7 (93 total)--------\n",
      "Val loss = 0.8587, Zero-temp loss = 0.7289\n",
      "\n",
      "--------Epoch 8 (94 total)--------\n",
      "Val loss = 0.8613, Zero-temp loss = 0.7301\n",
      "\n",
      "--------Epoch 9 (95 total)--------\n",
      "Val loss = 0.8622, Zero-temp loss = 0.7277\n",
      "\n",
      "--------Epoch 10 (96 total)--------\n",
      "Val loss = 0.8616, Zero-temp loss = 0.7287\n",
      "\n",
      "--------Epoch 11 (97 total)--------\n",
      "Val loss = 0.8634, Zero-temp loss = 0.7318\n",
      "\n",
      "--------Epoch 12 (98 total)--------\n",
      "Val loss = 0.8640, Zero-temp loss = 0.7320\n",
      "\n",
      "--------Epoch 13 (99 total)--------\n",
      "Val loss = 0.8639, Zero-temp loss = 0.7304\n",
      "\n",
      "--------Epoch 14 (100 total)--------\n",
      "Val loss = 0.8627, Zero-temp loss = 0.7294\n",
      "\n",
      "--------Epoch 15 (101 total)--------\n",
      "Val loss = 0.8641, Zero-temp loss = 0.7313\n",
      "\n",
      "--------Epoch 16 (102 total)--------\n",
      "Val loss = 0.8652, Zero-temp loss = 0.7274\n",
      "\n",
      "--------Epoch 17 (103 total)--------\n",
      "Val loss = 0.8656, Zero-temp loss = 0.7316\n",
      "\n",
      "--------Epoch 18 (104 total)--------\n",
      "Val loss = 0.8640, Zero-temp loss = 0.7272\n",
      "\n",
      "--------Epoch 19 (105 total)--------\n",
      "Val loss = 0.8650, Zero-temp loss = 0.7292\n",
      "\n",
      "--------Epoch 20 (106 total)--------\n",
      "Val loss = 0.8645, Zero-temp loss = 0.7296\n",
      "\n",
      "Epoch    20: reducing learning rate of group 0 to 2.0000e-04.\n",
      "--------Epoch 21 (107 total)--------\n",
      "Val loss = 0.8757, Zero-temp loss = 0.7389\n",
      "\n",
      "--------Epoch 22 (108 total)--------\n",
      "Val loss = 0.8782, Zero-temp loss = 0.7407\n",
      "\n",
      "--------Epoch 23 (109 total)--------\n",
      "Val loss = 0.8762, Zero-temp loss = 0.7368\n",
      "\n",
      "--------Epoch 24 (110 total)--------\n",
      "Val loss = 0.8790, Zero-temp loss = 0.7405\n",
      "\n",
      "--------Epoch 25 (111 total)--------\n",
      "Val loss = 0.8808, Zero-temp loss = 0.7411\n",
      "\n",
      "--------Epoch 26 (112 total)--------\n",
      "Val loss = 0.8824, Zero-temp loss = 0.7433\n",
      "\n",
      "--------Epoch 27 (113 total)--------\n",
      "Val loss = 0.8826, Zero-temp loss = 0.7394\n",
      "\n",
      "--------Epoch 28 (114 total)--------\n",
      "Val loss = 0.8803, Zero-temp loss = 0.7379\n",
      "\n",
      "--------Epoch 29 (115 total)--------\n",
      "Val loss = 0.8827, Zero-temp loss = 0.7403\n",
      "\n",
      "--------Epoch 30 (116 total)--------\n",
      "Val loss = 0.8826, Zero-temp loss = 0.7393\n",
      "\n",
      "--------Epoch 31 (117 total)--------\n",
      "Val loss = 0.8812, Zero-temp loss = 0.7377\n",
      "\n",
      "--------Epoch 32 (118 total)--------\n",
      "Val loss = 0.8838, Zero-temp loss = 0.7405\n",
      "\n",
      "--------Epoch 33 (119 total)--------\n",
      "Val loss = 0.8840, Zero-temp loss = 0.7381\n",
      "\n",
      "--------Epoch 34 (120 total)--------\n",
      "Val loss = 0.8828, Zero-temp loss = 0.7370\n",
      "\n",
      "--------Epoch 35 (121 total)--------\n",
      "Val loss = 0.8820, Zero-temp loss = 0.7359\n",
      "\n",
      "--------Epoch 36 (122 total)--------\n",
      "Val loss = 0.8845, Zero-temp loss = 0.7399\n",
      "\n",
      "--------Epoch 37 (123 total)--------\n",
      "Val loss = 0.8844, Zero-temp loss = 0.7402\n",
      "\n",
      "--------Epoch 38 (124 total)--------\n",
      "Val loss = 0.8864, Zero-temp loss = 0.7398\n",
      "\n",
      "--------Epoch 39 (125 total)--------\n",
      "Val loss = 0.8838, Zero-temp loss = 0.7378\n",
      "\n",
      "--------Epoch 40 (126 total)--------\n",
      "Val loss = 0.8834, Zero-temp loss = 0.7369\n",
      "\n",
      "--------Epoch 41 (127 total)--------\n",
      "Val loss = 0.8841, Zero-temp loss = 0.7365\n",
      "\n",
      "Epoch    41: reducing learning rate of group 0 to 4.0000e-05.\n",
      "--------Epoch 42 (128 total)--------\n",
      "Val loss = 0.8887, Zero-temp loss = 0.7427\n",
      "\n",
      "--------Epoch 43 (129 total)--------\n",
      "Val loss = 0.8869, Zero-temp loss = 0.7391\n",
      "\n",
      "--------Epoch 44 (130 total)--------\n",
      "Val loss = 0.8864, Zero-temp loss = 0.7396\n",
      "\n",
      "--------Epoch 45 (131 total)--------\n",
      "Val loss = 0.8867, Zero-temp loss = 0.7415\n",
      "\n",
      "Epoch    45: reducing learning rate of group 0 to 1.0000e-05.\n",
      "--------Epoch 46 (132 total)--------\n",
      "Val loss = 0.8883, Zero-temp loss = 0.7401\n",
      "\n",
      "Stopping temp = 0.5623 at epoch 46\n",
      "\n",
      "Starting training with temp = 0.3162\n",
      "\n",
      "--------Epoch 1 (133 total)--------\n",
      "Val loss = 0.8254, Zero-temp loss = 0.7683\n",
      "\n",
      "--------Epoch 2 (134 total)--------\n",
      "Val loss = 0.8250, Zero-temp loss = 0.7680\n",
      "\n",
      "--------Epoch 3 (135 total)--------\n",
      "Val loss = 0.8223, Zero-temp loss = 0.7680\n",
      "\n",
      "--------Epoch 4 (136 total)--------\n",
      "Val loss = 0.8260, Zero-temp loss = 0.7705\n",
      "\n",
      "--------Epoch 5 (137 total)--------\n",
      "Val loss = 0.8265, Zero-temp loss = 0.7721\n",
      "\n",
      "--------Epoch 6 (138 total)--------\n",
      "Val loss = 0.8264, Zero-temp loss = 0.7704\n",
      "\n",
      "--------Epoch 7 (139 total)--------\n",
      "Val loss = 0.8277, Zero-temp loss = 0.7745\n",
      "\n",
      "--------Epoch 8 (140 total)--------\n",
      "Val loss = 0.8277, Zero-temp loss = 0.7739\n",
      "\n",
      "--------Epoch 9 (141 total)--------\n",
      "Val loss = 0.8282, Zero-temp loss = 0.7738\n",
      "\n",
      "--------Epoch 10 (142 total)--------\n",
      "Val loss = 0.8271, Zero-temp loss = 0.7742\n",
      "\n",
      "--------Epoch 11 (143 total)--------\n",
      "Val loss = 0.8292, Zero-temp loss = 0.7758\n",
      "\n",
      "--------Epoch 12 (144 total)--------\n",
      "Val loss = 0.8276, Zero-temp loss = 0.7736\n",
      "\n",
      "--------Epoch 13 (145 total)--------\n",
      "Val loss = 0.8279, Zero-temp loss = 0.7775\n",
      "\n",
      "--------Epoch 14 (146 total)--------\n",
      "Val loss = 0.8276, Zero-temp loss = 0.7741\n",
      "\n",
      "Epoch    14: reducing learning rate of group 0 to 2.0000e-04.\n",
      "--------Epoch 15 (147 total)--------\n",
      "Val loss = 0.8364, Zero-temp loss = 0.7814\n",
      "\n",
      "--------Epoch 16 (148 total)--------\n",
      "Val loss = 0.8388, Zero-temp loss = 0.7826\n",
      "\n",
      "--------Epoch 17 (149 total)--------\n",
      "Val loss = 0.8407, Zero-temp loss = 0.7841\n",
      "\n",
      "--------Epoch 18 (150 total)--------\n",
      "Val loss = 0.8387, Zero-temp loss = 0.7808\n",
      "\n",
      "--------Epoch 19 (151 total)--------\n",
      "Val loss = 0.8383, Zero-temp loss = 0.7805\n",
      "\n",
      "--------Epoch 20 (152 total)--------\n",
      "Val loss = 0.8398, Zero-temp loss = 0.7819\n",
      "\n",
      "Epoch    20: reducing learning rate of group 0 to 4.0000e-05.\n",
      "--------Epoch 21 (153 total)--------\n",
      "Val loss = 0.8436, Zero-temp loss = 0.7854\n",
      "\n",
      "--------Epoch 22 (154 total)--------\n",
      "Val loss = 0.8439, Zero-temp loss = 0.7856\n",
      "\n",
      "--------Epoch 23 (155 total)--------\n",
      "Val loss = 0.8423, Zero-temp loss = 0.7842\n",
      "\n",
      "--------Epoch 24 (156 total)--------\n",
      "Val loss = 0.8426, Zero-temp loss = 0.7836\n",
      "\n",
      "--------Epoch 25 (157 total)--------\n",
      "Val loss = 0.8424, Zero-temp loss = 0.7849\n",
      "\n",
      "Epoch    25: reducing learning rate of group 0 to 1.0000e-05.\n",
      "--------Epoch 26 (158 total)--------\n",
      "Val loss = 0.8436, Zero-temp loss = 0.7863\n",
      "\n",
      "Stopping temp = 0.3162 at epoch 26\n",
      "\n",
      "Starting training with temp = 0.1778\n",
      "\n",
      "--------Epoch 1 (159 total)--------\n",
      "Val loss = 0.8120, Zero-temp loss = 0.7898\n",
      "\n",
      "--------Epoch 2 (160 total)--------\n",
      "Val loss = 0.8119, Zero-temp loss = 0.7918\n",
      "\n",
      "--------Epoch 3 (161 total)--------\n",
      "Val loss = 0.8134, Zero-temp loss = 0.7933\n",
      "\n",
      "--------Epoch 4 (162 total)--------\n",
      "Val loss = 0.8133, Zero-temp loss = 0.7930\n",
      "\n",
      "--------Epoch 5 (163 total)--------\n",
      "Val loss = 0.8132, Zero-temp loss = 0.7934\n",
      "\n",
      "--------Epoch 6 (164 total)--------\n",
      "Val loss = 0.8134, Zero-temp loss = 0.7939\n",
      "\n",
      "Epoch     6: reducing learning rate of group 0 to 2.0000e-04.\n",
      "--------Epoch 7 (165 total)--------\n",
      "Val loss = 0.8202, Zero-temp loss = 0.7992\n",
      "\n",
      "--------Epoch 8 (166 total)--------\n",
      "Val loss = 0.8214, Zero-temp loss = 0.8002\n",
      "\n",
      "--------Epoch 9 (167 total)--------\n",
      "Val loss = 0.8222, Zero-temp loss = 0.8015\n",
      "\n",
      "--------Epoch 10 (168 total)--------\n",
      "Val loss = 0.8205, Zero-temp loss = 0.7999\n",
      "\n",
      "--------Epoch 11 (169 total)--------\n",
      "Val loss = 0.8246, Zero-temp loss = 0.8033\n",
      "\n",
      "--------Epoch 12 (170 total)--------\n",
      "Val loss = 0.8235, Zero-temp loss = 0.8024\n",
      "\n",
      "--------Epoch 13 (171 total)--------\n",
      "Val loss = 0.8247, Zero-temp loss = 0.8040\n",
      "\n",
      "--------Epoch 14 (172 total)--------\n",
      "Val loss = 0.8257, Zero-temp loss = 0.8041\n",
      "\n",
      "--------Epoch 15 (173 total)--------\n",
      "Val loss = 0.8255, Zero-temp loss = 0.8038\n",
      "\n",
      "--------Epoch 16 (174 total)--------\n",
      "Val loss = 0.8253, Zero-temp loss = 0.8037\n",
      "\n",
      "--------Epoch 17 (175 total)--------\n",
      "Val loss = 0.8236, Zero-temp loss = 0.8019\n",
      "\n",
      "Epoch    17: reducing learning rate of group 0 to 4.0000e-05.\n",
      "--------Epoch 18 (176 total)--------\n",
      "Val loss = 0.8253, Zero-temp loss = 0.8030\n",
      "\n",
      "Stopping temp = 0.1778 at epoch 18\n",
      "\n",
      "Starting training with temp = 0.1000\n",
      "\n",
      "--------Epoch 1 (177 total)--------\n",
      "Val loss = 0.8076, Zero-temp loss = 0.7991\n",
      "\n",
      "--------Epoch 2 (178 total)--------\n",
      "Val loss = 0.8094, Zero-temp loss = 0.8015\n",
      "\n",
      "--------Epoch 3 (179 total)--------\n",
      "Val loss = 0.8059, Zero-temp loss = 0.7978\n",
      "\n",
      "--------Epoch 4 (180 total)--------\n",
      "Val loss = 0.8091, Zero-temp loss = 0.8024\n",
      "\n",
      "--------Epoch 5 (181 total)--------\n",
      "Val loss = 0.8100, Zero-temp loss = 0.8030\n",
      "\n",
      "--------Epoch 6 (182 total)--------\n",
      "Val loss = 0.8062, Zero-temp loss = 0.7988\n",
      "\n",
      "--------Epoch 7 (183 total)--------\n",
      "Val loss = 0.8098, Zero-temp loss = 0.8030\n",
      "\n",
      "--------Epoch 8 (184 total)--------\n",
      "Val loss = 0.8065, Zero-temp loss = 0.8003\n",
      "\n",
      "Epoch     8: reducing learning rate of group 0 to 2.0000e-04.\n",
      "--------Epoch 9 (185 total)--------\n",
      "Val loss = 0.8148, Zero-temp loss = 0.8082\n",
      "\n",
      "--------Epoch 10 (186 total)--------\n",
      "Val loss = 0.8159, Zero-temp loss = 0.8093\n",
      "\n",
      "--------Epoch 11 (187 total)--------\n",
      "Val loss = 0.8175, Zero-temp loss = 0.8108\n",
      "\n",
      "--------Epoch 12 (188 total)--------\n",
      "Val loss = 0.8190, Zero-temp loss = 0.8116\n",
      "\n",
      "--------Epoch 13 (189 total)--------\n",
      "Val loss = 0.8197, Zero-temp loss = 0.8122\n",
      "\n",
      "--------Epoch 14 (190 total)--------\n",
      "Val loss = 0.8207, Zero-temp loss = 0.8134\n",
      "\n",
      "--------Epoch 15 (191 total)--------\n",
      "Val loss = 0.8197, Zero-temp loss = 0.8123\n",
      "\n",
      "--------Epoch 16 (192 total)--------\n",
      "Val loss = 0.8211, Zero-temp loss = 0.8135\n",
      "\n",
      "--------Epoch 17 (193 total)--------\n",
      "Val loss = 0.8189, Zero-temp loss = 0.8114\n",
      "\n",
      "--------Epoch 18 (194 total)--------\n",
      "Val loss = 0.8210, Zero-temp loss = 0.8138\n",
      "\n",
      "--------Epoch 19 (195 total)--------\n",
      "Val loss = 0.8208, Zero-temp loss = 0.8130\n",
      "\n",
      "Epoch    19: reducing learning rate of group 0 to 4.0000e-05.\n",
      "--------Epoch 20 (196 total)--------\n",
      "Val loss = 0.8231, Zero-temp loss = 0.8156\n",
      "\n",
      "--------Epoch 21 (197 total)--------\n",
      "Val loss = 0.8212, Zero-temp loss = 0.8134\n",
      "\n",
      "--------Epoch 22 (198 total)--------\n",
      "Val loss = 0.8223, Zero-temp loss = 0.8145\n",
      "\n",
      "--------Epoch 23 (199 total)--------\n",
      "Val loss = 0.8210, Zero-temp loss = 0.8130\n",
      "\n",
      "Epoch    23: reducing learning rate of group 0 to 1.0000e-05.\n",
      "--------Epoch 24 (200 total)--------\n",
      "Val loss = 0.8231, Zero-temp loss = 0.8156\n",
      "\n",
      "Stopping temp = 0.1000 at epoch 24\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# Set up adaptive selection\n",
    "mask_layer = MaskLayer(append=True)\n",
    "gafs = AdaptiveSelection(selector, deepcopy(predictor), mask_layer).to(device)\n",
    "\n",
    "# Train\n",
    "gafs.fit(train_dataset,\n",
    "         val_dataset,\n",
    "         mbsize=128,\n",
    "         lr=1e-3,\n",
    "         nepochs=250,\n",
    "         max_features=max_features,\n",
    "         loss_fn=nn.CrossEntropyLoss(),\n",
    "         val_loss_fn=Accuracy(),\n",
    "         val_loss_mode='max')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9e68b4eb-0d5e-4433-a27c-e84c4d122edc",
   "metadata": {},
   "source": [
    "# Plot results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "10b4daf1-bdf7-4476-a441-74681d21b9e1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Acc (5) = 70.83\n",
      "Acc (10) = 88.37\n",
      "Acc (15) = 93.46\n",
      "Acc (20) = 94.88\n",
      "Acc (25) = 95.76\n"
     ]
    }
   ],
   "source": [
    "# Get accuracy\n",
    "num_features = (5, 10, 15, 20, 25)\n",
    "acc_list = []\n",
    "for num in num_features:\n",
    "    test_acc = gafs.evaluate(test_dataset, num, Accuracy(), 1024)\n",
    "    print(f'Acc ({num}) = {100*test_acc:.2f}')\n",
    "    acc_list.append(test_acc)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "efc1a8ff-ccb2-4fac-b08e-b18df180dd92",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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m5fz+DGP06SRXmIRERHZnloAM8z3hd8zZsYYjuh2BAd0PiDscERERKQBUgKYYUZNG0VpeSk35d8UdigiIiLSQCkBzCAzVs1gwtcTuKDvBXRr0y3scERERKSBUgKYIcq9nJGTRrJT8524bMBlYYcjIiIiDZgSwAwxceFEZqyewfWDr6dlTsuwwxEREZEGTAlgBigsKeSBqQ/Qv2N/Tt795LDDERERkQZOCWAGeGLmE3y/5Xtu3f9WGpleUhEREamasoU0t2zTMsbNHseJvU5k0C6Dwg5HRERE0oASwDR339T7aNyoMdcPvj7sUERERCRNKAFMY5NWTOIf3/yDS/pfQqeWncIOR0RERNKEEsA0VVpeysjJI+nSsgtD+w0NOxwRERFJI0oA09Qr81/hq3VfcWP+jeQ2yQ07HBEREUkjSgDT0IbiDTz4+YPk75rP0d2PDjscERERSTNKANPQ6Omj2bh1I7fufytmFnY4IiIikmaUAKaZr9d/zfNfPs/pe5zO3h32DjscERERSUNKANOIu3PX5Lto0aQF1+x7TdjhiIiISJpSAphG/rXsX3z67adcOehKOuR2CDscERERSVNKANNESVkJd02+i55te/KLvX8RdjgiIiKSxpqEHYAk5tm5z7Jk0xIePepRchrlhB2OiIiIpDG1AKaB1VtWM3rGaH6y2084uOvBYYcjIiIiaU4JYBp48PMHKS4t5ub8m8MORURERDKAEsAGbvaa2bw6/1XO7XMuPdr2CDscERERyQBKABswd2fkpJG0z23P5QMvDzscERERyRBKABuwdxa/w+fff851+15H66atww5HREREMoQSwAZqS+kW7p1yL3069OG03qeFHY6IiIhkECWADdTTs57mu8LvuHX/W2ncqHHY4YiIiEgGUQLYAK3YvIKnZj3FcT2OY79d9ws7HBEREckwSgAboPum3gfADfvdEHIkIiIikomUADYwU7+byjuL3+Hi/hfTuVXnsMMRERGRDKQEsAEpKy9j5KSR7NpiVy7qf1HY4YiIiEiGUgLYgLy24DXmrp3Ljfk30rxJ87DDERERkQylBLCB2LR1E3/5/C/su8u+HNfjuLDDERERkQzWJOwAJPDY9MdYV7SOR456BDMLOxwRERHJYEoAQzRx4URGTRvFyoKVOE7+rvn069gv7LBEREQkw6kLOCQTF05kxKcjWFGw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\n",
      "text/plain": [
       "<Figure size 648x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure(figsize=(9, 6))\n",
    "\n",
    "# Plot\n",
    "plt.plot(num_features, acc_list, color='tab:green', marker='o')\n",
    "\n",
    "# Labels\n",
    "plt.xlabel('# Features', fontsize=18)\n",
    "plt.ylabel('Accuracy', fontsize=18)\n",
    "plt.tick_params(labelsize=16)\n",
    "plt.title('MNIST Feature Selection', fontsize=20)\n",
    "\n",
    "# Axis spines\n",
    "plt.gca().spines['right'].set_visible(False)\n",
    "plt.gca().spines['top'].set_visible(False)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "697ee1c2-65a5-4a89-98cc-6fbe617c46df",
   "metadata": {},
   "source": [
    "# Check calibration"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "d15bca2b-f56d-402c-be55-c3a9fe4470de",
   "metadata": {},
   "outputs": [],
   "source": [
    "from torchmetrics.functional.classification.calibration_error import _ce_update, _binning_bucketize"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "88b7a790-5b8d-423c-9ee2-f7a39c7ad499",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Set feature budget\n",
    "num = 10\n",
    "\n",
    "# Prepare inputs\n",
    "x = torch.stack([x for x, y in test_dataset])\n",
    "y = torch.tensor(np.array([y for x, y in test_dataset]))\n",
    "\n",
    "# Generate predictions\n",
    "with torch.no_grad():\n",
    "    pred = gafs.forward(x.to(device), max_features=num).softmax(dim=1).cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "426aa45b-9295-41c0-9c41-e572167246d7",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Acc = 88.37\n",
      "\n"
     ]
    }
   ],
   "source": [
    "test_acc = gafs.evaluate(test_dataset, num, Accuracy(), 1024)\n",
    "print(f'Acc = {100*test_acc:.2f}\\n')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "abfd0c31-1c79-41e9-abc0-d0ad1b0b3578",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Calibration error = 0.0138\n"
     ]
    }
   ],
   "source": [
    "# Calculate per-bin calibration results\n",
    "bin_boundaries = torch.linspace(0, 1, 20)\n",
    "confidences, accuracies = _ce_update(pred, y)\n",
    "acc_bin, conf_bin, prop_bin = _binning_bucketize(confidences, accuracies, bin_boundaries)\n",
    "\n",
    "# Calculate l2 metric\n",
    "ce = torch.sqrt(torch.sum(torch.pow(acc_bin - conf_bin, 2) * prop_bin))\n",
    "print(f'Calibration error = {ce:.4f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "72630242-cf6d-42b2-b9c5-bbe34961da28",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 648x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure(figsize=(9, 6))\n",
    "\n",
    "# Plot\n",
    "plt.plot(conf_bin, acc_bin, marker='o')\n",
    "plt.plot([0, 1], [0, 1], color='black', linestyle=':')\n",
    "\n",
    "# Labels\n",
    "plt.xlabel('Confidence', fontsize=18)\n",
    "plt.ylabel('Accuracy', fontsize=18)\n",
    "plt.tick_params(labelsize=16)\n",
    "plt.title('Calibration', fontsize=20)\n",
    "\n",
    "# Axis spines\n",
    "plt.gca().spines['right'].set_visible(False)\n",
    "plt.gca().spines['top'].set_visible(False)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "970af148-b5ba-48bc-8631-dbe1bba76c41",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.6.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
