{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "name": "DUQ_DE_rejection.ipynb",
      "provenance": [],
      "collapsed_sections": []
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "accelerator": "GPU"
  },
  "cells": [
    {
      "cell_type": "code",
      "metadata": {
        "id": "7NdrkVE3JhPt",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "d1c1d6b5-b742-4251-c27a-41f3dd6e7de1"
      },
      "source": [
        "from google.colab import drive\n",
        "drive.mount('/content/gdrive')"
      ],
      "execution_count": 11,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "Drive already mounted at /content/gdrive; to attempt to forcibly remount, call drive.mount(\"/content/gdrive\", force_remount=True).\n"
          ],
          "name": "stdout"
        }
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "31kViAtB9PEe",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "134b4380-1557-435c-cb9f-ce0693f707a6"
      },
      "source": [
        "import torch\n",
        "import matplotlib.pyplot as plt \n",
        "import numpy as np\n",
        "from utils.cnn_duq import CNN_DUQ\n",
        "from utils.datasets import all_datasets\n",
        "from utils.cnn_duq import SoftmaxModel as CNN\n",
        "\n",
        "from utils.resnet import ResNet\n",
        "from utils.resnet_duq import ResNet_DUQ\n",
        "from utils.evaluate_ood import get_cifar_svhn_ood, get_auroc_classification\n",
        "\n",
        "mod='CIFAR10' #['CIFAR10','FMnist']\n",
        "\n",
        "if mod=='FMnist':\n",
        "    ds1 = all_datasets[\"FashionMNIST\"]()\n",
        "    ds2 = all_datasets[\"MNIST\"]()\n",
        "    input_size = 28\n",
        "    num_classes = 10\n",
        "    embedding_size = 256\n",
        "    learnable_length_scale = False\n",
        "    gamma = 0.999\n",
        "    length_scale = 0.1\n",
        "    d=28\n",
        "    c=1\n",
        "\n",
        "    model = CNN_DUQ(\n",
        "    input_size,\n",
        "    num_classes,\n",
        "    embedding_size,\n",
        "    learnable_length_scale,\n",
        "    length_scale,\n",
        "    gamma,\n",
        "    ).cuda()\n",
        "    model.load_state_dict(torch.load('/content/gdrive/My Drive/Colab Notebooks/DUQ_FM_30_FULL.pt'))\n",
        "\n",
        "    ensemble = [CNN(input_size, num_classes).cuda() for _ in range(5)]\n",
        "    ensemble = torch.nn.ModuleList(ensemble);\n",
        "    ensemble.load_state_dict(torch.load('/content/gdrive/My Drive/Colab Notebooks/FM_5_ensemble_30.pt'))\n",
        "\n",
        "else:\n",
        "    ds1 = all_datasets[\"CIFAR10\"]()\n",
        "    ds2 = all_datasets[\"SVHN\"]()\n",
        "    length_scale = 0.1\n",
        "    input_size, num_classes, dataset, test_dataset = ds1\n",
        "    centroid_size=512\n",
        "    model_output_size=512 \n",
        "    gamma = 0.999\n",
        "    d=32\n",
        "    c=3\n",
        "\n",
        "    model = ResNet_DUQ(\n",
        "            input_size, num_classes, centroid_size, model_output_size, length_scale, gamma\n",
        "        ).cuda()\n",
        "    model.load_state_dict(torch.load('/content/gdrive/My Drive/Colab Notebooks/DUQ_CIFAR_75.pt'))\n",
        "    ensemble = [\n",
        "            ResNet(input_size, num_classes).cuda() for _ in range(5)\n",
        "        ]\n",
        "    ensemble = torch.nn.ModuleList(ensemble);\n",
        "    ensemble.load_state_dict(torch.load('/content/gdrive/My Drive/Colab Notebooks/CIFAR10_5_ensemble.pt'))\n",
        "    \n",
        "model.eval()\n",
        "ensemble.eval()"
      ],
      "execution_count": 12,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "Files already downloaded and verified\n",
            "Files already downloaded and verified\n",
            "Using downloaded and verified file: ./data/SVHN/train_32x32.mat\n",
            "Using downloaded and verified file: ./data/SVHN/test_32x32.mat\n"
          ],
          "name": "stdout"
        },
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "ModuleList(\n",
              "  (0): ResNet(\n",
              "    (resnet): ResNet(\n",
              "      (conv1): Conv2d(3, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (maxpool): Identity()\n",
              "      (layer1): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer2): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 128, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(64, 128, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer3): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(128, 256, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(128, 256, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer4): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(256, 512, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(256, 512, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (avgpool): AdaptiveAvgPool2d(output_size=(1, 1))\n",
              "      (fc): Linear(in_features=512, out_features=10, bias=True)\n",
              "    )\n",
              "  )\n",
              "  (1): ResNet(\n",
              "    (resnet): ResNet(\n",
              "      (conv1): Conv2d(3, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (maxpool): Identity()\n",
              "      (layer1): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer2): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 128, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(64, 128, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer3): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(128, 256, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(128, 256, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer4): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(256, 512, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(256, 512, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (avgpool): AdaptiveAvgPool2d(output_size=(1, 1))\n",
              "      (fc): Linear(in_features=512, out_features=10, bias=True)\n",
              "    )\n",
              "  )\n",
              "  (2): ResNet(\n",
              "    (resnet): ResNet(\n",
              "      (conv1): Conv2d(3, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (maxpool): Identity()\n",
              "      (layer1): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer2): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 128, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(64, 128, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer3): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(128, 256, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(128, 256, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer4): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(256, 512, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(256, 512, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (avgpool): AdaptiveAvgPool2d(output_size=(1, 1))\n",
              "      (fc): Linear(in_features=512, out_features=10, bias=True)\n",
              "    )\n",
              "  )\n",
              "  (3): ResNet(\n",
              "    (resnet): ResNet(\n",
              "      (conv1): Conv2d(3, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (maxpool): Identity()\n",
              "      (layer1): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer2): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 128, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(64, 128, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer3): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(128, 256, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(128, 256, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer4): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(256, 512, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(256, 512, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (avgpool): AdaptiveAvgPool2d(output_size=(1, 1))\n",
              "      (fc): Linear(in_features=512, out_features=10, bias=True)\n",
              "    )\n",
              "  )\n",
              "  (4): ResNet(\n",
              "    (resnet): ResNet(\n",
              "      (conv1): Conv2d(3, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (maxpool): Identity()\n",
              "      (layer1): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer2): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(64, 128, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(64, 128, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer3): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(128, 256, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(128, 256, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (layer4): Sequential(\n",
              "        (0): BasicBlock(\n",
              "          (conv1): Conv2d(256, 512, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (downsample): Sequential(\n",
              "            (0): Conv2d(256, 512, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "            (1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          )\n",
              "        )\n",
              "        (1): BasicBlock(\n",
              "          (conv1): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "          (relu): ReLU(inplace=True)\n",
              "          (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "          (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "        )\n",
              "      )\n",
              "      (avgpool): AdaptiveAvgPool2d(output_size=(1, 1))\n",
              "      (fc): Linear(in_features=512, out_features=10, bias=True)\n",
              "    )\n",
              "  )\n",
              ")"
            ]
          },
          "metadata": {
            "tags": []
          },
          "execution_count": 12
        }
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "NT-7TDRzBwAa",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 974
        },
        "outputId": "258b8e52-51f1-4a9d-a712-ae4d63f12b18"
      },
      "source": [
        "args={'ensemble':5}\n",
        "input_size, num_classes, _ , test_dataset_n = ds1\n",
        "_ , _ , _ , test_dataset_o = ds2\n",
        "er=[0]\n",
        "fig = plt.figure(figsize=(12,5))\n",
        "\n",
        "\n",
        "m=test_dataset_n[0][0].max()\n",
        "rejection_list = [0.1 , 0.2 , 0.3 ,0.4 , 0.5 ,0.6 , 0.7 , 0.8 , 0.9]\n",
        "\n",
        "\n",
        "for p,e in enumerate(er):\n",
        "    sample=(test_dataset_n[10][0]).numpy()\n",
        "\n",
        "    Data = test_dataset_n+test_dataset_o\n",
        "    num=len(Data)\n",
        "    b=100\n",
        "    r=len(Data)\n",
        "    ls=[]\n",
        "    for i in range(int(r/b)+1):\n",
        "        data=[]\n",
        "        target=[]\n",
        "        for j in range(0,b):\n",
        "            cnt=i*b+j\n",
        "            if(cnt>=r):\n",
        "              break\n",
        "            data.append(Data[cnt][0])\n",
        "            target.append(Data[cnt][1])\n",
        "        if(len(data)==0):\n",
        "          break\n",
        "        data=torch.stack(data)\n",
        "        target=torch.tensor(target)\n",
        "        ls.append((data,target))\n",
        "\n",
        "\n",
        "    target = np.zeros((num,))\n",
        "    confidence_DUQ = np.zeros((num,))\n",
        "    pred_DUQ = np.zeros((num,))\n",
        "    cn=0\n",
        "    i=0\n",
        "    for Data,t1 in ls:\n",
        "      with torch.no_grad():\n",
        "        _ , output = model(Data.cuda())\n",
        "        c1,p1= output.max(1)\n",
        "\n",
        "        for j in range(0,len(Data)):\n",
        "          confidence_DUQ[cn]=c1[j]\n",
        "          target[cn]=t1[j]\n",
        "          pred_DUQ[cn]=p1[j]\n",
        "          cn+=1\n",
        "        i+=1\n",
        "        if(i%20==0):\n",
        "          print(cn)\n",
        "\n",
        "    a  = np.concatenate((target.reshape(-1,1),pred_DUQ.reshape(-1,1),confidence_DUQ.reshape(-1,1)) , axis=1)\n",
        "    x  = a[a[:,-1].argsort()]\n",
        "    accuracy_DUQ = np.zeros((len(rejection_list),1))\n",
        "    rejected_DUQ = np.zeros((len(rejection_list),1))\n",
        "    i=0\n",
        "    for reject in rejection_list :\n",
        "      y = x[:][int(reject*num):]\n",
        "      accuracy_DUQ[i] = ((y[:,0]==y[:,1]).sum())/((1-reject)*num)\n",
        "      rejected_DUQ[i] = reject*100\n",
        "      i+=1\n",
        "\n",
        "\n",
        "\n",
        "    target = np.zeros((num))\n",
        "    confidence_DE = np.zeros((num,))\n",
        "    pred_DE = np.zeros((num,))\n",
        "    i=0\n",
        "    cn=0\n",
        "    for Data,t1 in ls:\n",
        "      with torch.no_grad():\n",
        "          predictions = torch.stack([model(Data.cuda()) for model in ensemble])\n",
        "          mean_prediction = torch.mean(predictions.exp(), dim=0)\n",
        "          p1 = mean_prediction.max(1)[1]\n",
        "          c1= torch.sum(mean_prediction * torch.log(mean_prediction), dim=1)\n",
        "\n",
        "          for j in range(0,len(Data)):\n",
        "              confidence_DE[cn]=c1[j]\n",
        "              target[cn]=t1[j]\n",
        "              pred_DE[cn]=p1[j]\n",
        "              cn+=1\n",
        "          i+=1\n",
        "          if(i%20==0):\n",
        "              print(cn)\n",
        "\n",
        "    a  = np.concatenate((target.reshape(-1,1),pred_DE.reshape(-1,1),confidence_DE.reshape(-1,1)) , axis=1)\n",
        "    x  = a[a[:,-1].argsort()]\n",
        "    accuracy_DE = np.zeros((len(rejection_list),1))\n",
        "    rejected_DE = np.zeros((len(rejection_list),1))\n",
        "    i=0\n",
        "    for reject in rejection_list :\n",
        "      y = x[:][int(reject*num):]\n",
        "      accuracy_DE[i] = ((y[:,0]==y[:,1]).sum())/((1-reject)*num) \n",
        "      rejected_DE[i] = reject*100\n",
        "      i+=1\n",
        "\n",
        "\n",
        "\n",
        "    ax=fig.add_subplot(len(er),2,2*p+1,xlabel='%',ylabel='Acc',title='Rejection plot',xlim=(0,100),ylim=(0.4,1.01))\n",
        "\n",
        "    ax.plot(rejected_DUQ, accuracy_DUQ, color='blue', linewidth = 2, \n",
        "            marker='o', markerfacecolor='blue', markersize=5)\n",
        "    ax.plot(rejected_DE , accuracy_DE , color='orange', linewidth = 2, \n",
        "            marker='o', markerfacecolor='orange', markersize=5 )\n",
        "    ax.legend(['DUQ','5-Deep Ensemble'])\n",
        "\n",
        "    ax=fig.add_subplot(len(er),2,2*p+2)\n",
        "    sample-=sample.min()   \n",
        "    sample/=sample.max()\n",
        "    if(sample.shape[0]>1):\n",
        "      sample=sample.transpose((1,2,0))\n",
        "    else:\n",
        "      sample=sample.reshape(d,d)\n",
        "    ax.imshow(sample)\n",
        " \n",
        "fig.tight_layout()\n",
        "plt.show()\n"
      ],
      "execution_count": 13,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "2000\n",
            "4000\n",
            "6000\n",
            "8000\n",
            "10000\n",
            "12000\n",
            "14000\n",
            "16000\n",
            "18000\n",
            "20000\n",
            "22000\n",
            "24000\n",
            "26000\n",
            "28000\n",
            "30000\n",
            "32000\n",
            "34000\n",
            "36000\n",
            "2000\n",
            "4000\n",
            "6000\n",
            "8000\n",
            "10000\n",
            "12000\n",
            "14000\n",
            "16000\n",
            "18000\n",
            "20000\n",
            "22000\n",
            "24000\n",
            "26000\n",
            "28000\n",
            "30000\n",
            "32000\n",
            "34000\n",
            "36000\n"
          ],
          "name": "stdout"
        },
        {
          "output_type": "display_data",
          "data": {
            "image/png": "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\n",
            "text/plain": [
              "<Figure size 864x360 with 2 Axes>"
            ]
          },
          "metadata": {
            "tags": [],
            "needs_background": "light"
          }
        }
      ]
    }
  ]
}