{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "kernelspec": {
      "display_name": "Python 3",
      "language": "python",
      "name": "python3"
    },
    "language_info": {
      "codemirror_mode": {
        "name": "ipython",
        "version": 3
      },
      "file_extension": ".py",
      "mimetype": "text/x-python",
      "name": "python",
      "nbconvert_exporter": "python",
      "pygments_lexer": "ipython3",
      "version": "3.7.7"
    },
    "colab": {
      "name": "DUQ_two_moons_allexps.ipynb",
      "provenance": []
    }
  },
  "cells": [
    {
      "cell_type": "code",
      "metadata": {
        "id": "4AbZggAMrlAi",
        "outputId": "d799566f-7bc8-4ddd-a023-f5d122a343c8",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 84
        }
      },
      "source": [
        "import torch\n",
        "import torch.utils.data\n",
        "from torch import nn\n",
        "from torch.nn import functional as F\n",
        "!pip install pytorch-ignite\n",
        "from ignite.engine import Events, Engine\n",
        "from ignite.metrics import Accuracy, Loss\n",
        "\n",
        "import numpy as np\n",
        "import sklearn.datasets\n",
        "\n",
        "import matplotlib.pyplot as plt\n",
        "import seaborn as sns\n",
        "\n",
        "sns.set()"
      ],
      "execution_count": 83,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "Requirement already satisfied: pytorch-ignite in /usr/local/lib/python3.6/dist-packages (0.4.2)\n",
            "Requirement already satisfied: torch<2,>=1.3 in /usr/local/lib/python3.6/dist-packages (from pytorch-ignite) (1.6.0+cu101)\n",
            "Requirement already satisfied: numpy in /usr/local/lib/python3.6/dist-packages (from torch<2,>=1.3->pytorch-ignite) (1.18.5)\n",
            "Requirement already satisfied: future in /usr/local/lib/python3.6/dist-packages (from torch<2,>=1.3->pytorch-ignite) (0.16.0)\n"
          ],
          "name": "stdout"
        }
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "Dy5orSk3rlAv"
      },
      "source": [
        "class Model_bilinear(nn.Module):\n",
        "    def __init__(self, features, num_embeddings):\n",
        "        super().__init__()\n",
        "        \n",
        "        self.gamma = 0.99\n",
        "        self.sigma = 0.3\n",
        "        \n",
        "        embedding_size = 10\n",
        "        \n",
        "        self.fc1 = nn.Linear(2, features)\n",
        "        self.fc2 = nn.Linear(features, features)\n",
        "        self.fc3 = nn.Linear(features, features)\n",
        "        \n",
        "        self.W = nn.Parameter(torch.normal(torch.zeros(embedding_size, num_embeddings, features), 1))\n",
        "        \n",
        "        self.register_buffer('N', torch.ones(num_embeddings) * 20)\n",
        "        self.register_buffer('m', torch.normal(torch.zeros(embedding_size, num_embeddings), 1))\n",
        "        \n",
        "        self.m = self.m * self.N.unsqueeze(0)\n",
        "\n",
        "    def embed(self, x):\n",
        "        x = F.relu(self.fc1(x))\n",
        "        x = F.relu(self.fc2(x))\n",
        "        x = self.fc3(x)\n",
        "        f=x\n",
        "        # i is batch, m is embedding_size, n is num_embeddings (classes)\n",
        "        x = torch.einsum('ij,mnj->imn', x, self.W)\n",
        "        \n",
        "        return f,x\n",
        "\n",
        "    def bilinear(self, z):\n",
        "        embeddings = self.m / self.N.unsqueeze(0)\n",
        "        \n",
        "        diff = z - embeddings.unsqueeze(0)            \n",
        "        y_pred = (- diff**2).mean(1).div(2 * self.sigma**2).exp()\n",
        "\n",
        "        return y_pred\n",
        "\n",
        "    def forward(self, x):\n",
        "        f,z = self.embed(x)\n",
        "        y_pred = self.bilinear(z)\n",
        "        \n",
        "        return f, y_pred\n",
        "\n",
        "    def update_embeddings(self, x, y):\n",
        "        _,z = self.embed(x)\n",
        "        \n",
        "        # normalizing value per class, assumes y is one_hot encoded\n",
        "        self.N = torch.max(self.gamma * self.N + (1 - self.gamma) * y.sum(0), torch.ones_like(self.N))\n",
        "        \n",
        "        # compute sum of embeddings on class by class basis\n",
        "        features_sum = torch.einsum('ijk,ik->jk', z, y)\n",
        "        \n",
        "        self.m = self.gamma * self.m + (1 - self.gamma) * features_sum"
      ],
      "execution_count": 84,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "_dX5_QLCrlA7"
      },
      "source": [
        "np.random.seed(0)\n",
        "torch.manual_seed(0)\n",
        "\n",
        "l_gradient_penalty = 1.0\n",
        "\n",
        "# Moons\n",
        "noise = 0.1\n",
        "X_train, y_train = sklearn.datasets.make_moons(n_samples=1500, noise=noise)\n",
        "X_test, y_test = sklearn.datasets.make_moons(n_samples=200, noise=noise)\n",
        "\n",
        "num_classes = 2\n",
        "batch_size = 64\n",
        "\n",
        "model = Model_bilinear(20, num_classes)\n",
        "\n",
        "optimizer = torch.optim.SGD(model.parameters(), lr=0.01, momentum=0.9, weight_decay=1e-4)\n",
        "\n",
        "\n",
        "def calc_gradient_penalty(x, y_pred):\n",
        "    gradients = torch.autograd.grad(\n",
        "            outputs=y_pred,\n",
        "            inputs=x,\n",
        "            grad_outputs=torch.ones_like(y_pred),\n",
        "            create_graph=True,\n",
        "        )[0]\n",
        "\n",
        "\n",
        "    gradients = gradients.flatten(start_dim=1)\n",
        "    \n",
        "    # L2 norm\n",
        "    grad_norm = gradients.norm(2, dim=1)\n",
        "\n",
        "    # Two sided penalty\n",
        "    gradient_penalty = ((grad_norm - 1) ** 2).mean()\n",
        "    \n",
        "    #One sided penalty - down\n",
        "    #gradient_penalty = F.relu(grad_norm - 1).mean()\n",
        "\n",
        "    return gradient_penalty\n",
        "\n",
        "\n",
        "def output_transform_acc(output):\n",
        "    y_pred, y, x, z = output\n",
        "    \n",
        "    y = torch.argmax(y, dim=1)\n",
        "        \n",
        "    return y_pred, y\n",
        "\n",
        "\n",
        "def output_transform_bce(output):\n",
        "    y_pred, y, x, z = output\n",
        "\n",
        "    return y_pred, y\n",
        "\n",
        "\n",
        "def output_transform_gp(output):\n",
        "    y_pred, y, x, z = output\n",
        "\n",
        "    return x, y_pred\n",
        "\n",
        "\n",
        "def step(engine, batch):\n",
        "    model.train()\n",
        "    optimizer.zero_grad()\n",
        "    \n",
        "    x, y = batch\n",
        "    x.requires_grad_(True)\n",
        "    \n",
        "    z, y_pred = model(x)\n",
        "    \n",
        "    loss1 =  F.binary_cross_entropy(y_pred, y)\n",
        "    loss2 = l_gradient_penalty * calc_gradient_penalty(x, y_pred.sum(1))\n",
        "    \n",
        "    #taking f(θ)\n",
        "    #loss2 = l_gradient_penalty * calc_gradient_penalty(x, z)\n",
        "    \n",
        "    \n",
        "    loss = loss1 + loss2\n",
        "    \n",
        "    loss.backward()\n",
        "    optimizer.step()\n",
        "    \n",
        "    with torch.no_grad():\n",
        "        model.update_embeddings(x, y)\n",
        "    \n",
        "    return loss.item()\n",
        "\n",
        "\n",
        "def eval_step(engine, batch):\n",
        "    model.eval()\n",
        "\n",
        "    x, y = batch\n",
        "\n",
        "    x.requires_grad_(True)\n",
        "\n",
        "    z, y_pred = model(x)\n",
        "\n",
        "    return y_pred, y, x, z\n",
        "    \n",
        "\n",
        "trainer = Engine(step)\n",
        "evaluator = Engine(eval_step)\n",
        "\n",
        "metric = Accuracy(output_transform=output_transform_acc)\n",
        "metric.attach(evaluator, \"accuracy\")\n",
        "\n",
        "metric = Loss(F.binary_cross_entropy, output_transform=output_transform_bce)\n",
        "metric.attach(evaluator, \"bce\")\n",
        "\n",
        "metric = Loss(calc_gradient_penalty, output_transform=output_transform_gp)\n",
        "metric.attach(evaluator, \"gp\")\n",
        "\n",
        "\n",
        "ds_train = torch.utils.data.TensorDataset(torch.from_numpy(X_train).float(), F.one_hot(torch.from_numpy(y_train)).float())\n",
        "dl_train = torch.utils.data.DataLoader(ds_train, batch_size=batch_size, shuffle=True, drop_last=True)\n",
        "\n",
        "ds_test = torch.utils.data.TensorDataset(torch.from_numpy(X_test).float(), F.one_hot(torch.from_numpy(y_test)).float())\n",
        "dl_test = torch.utils.data.DataLoader(ds_test, batch_size=200, shuffle=False)\n",
        "\n",
        "@trainer.on(Events.EPOCH_COMPLETED)\n",
        "def log_results(trainer):\n",
        "    evaluator.run(dl_test)\n",
        "    metrics = evaluator.state.metrics\n",
        "\n",
        "    print(\"Test Results - Epoch: {} Acc: {:.4f} BCE: {:.2f} GP {:.2f}\"\n",
        "          .format(trainer.state.epoch, metrics['accuracy'], metrics['bce'], metrics['gp']))"
      ],
      "execution_count": 85,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "scrolled": false,
        "id": "1owb7q2ArlBD",
        "outputId": "2de9fe00-8e66-4ab1-f91d-cc0014dcf17d",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 706
        }
      },
      "source": [
        "trainer.run(dl_train, max_epochs=30)"
      ],
      "execution_count": 86,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "Test Results - Epoch: 1 Acc: 0.8350 BCE: 0.46 GP 0.18\n",
            "Test Results - Epoch: 2 Acc: 0.8650 BCE: 0.40 GP 0.05\n",
            "Test Results - Epoch: 3 Acc: 0.9100 BCE: 0.30 GP 0.05\n",
            "Test Results - Epoch: 4 Acc: 0.9400 BCE: 0.23 GP 0.06\n",
            "Test Results - Epoch: 5 Acc: 0.9600 BCE: 0.17 GP 0.04\n",
            "Test Results - Epoch: 6 Acc: 0.9750 BCE: 0.14 GP 0.04\n",
            "Test Results - Epoch: 7 Acc: 0.9900 BCE: 0.13 GP 0.04\n",
            "Test Results - Epoch: 8 Acc: 1.0000 BCE: 0.11 GP 0.04\n",
            "Test Results - Epoch: 9 Acc: 1.0000 BCE: 0.11 GP 0.04\n",
            "Test Results - Epoch: 10 Acc: 0.9950 BCE: 0.09 GP 0.04\n",
            "Test Results - Epoch: 11 Acc: 0.9950 BCE: 0.10 GP 0.04\n",
            "Test Results - Epoch: 12 Acc: 1.0000 BCE: 0.10 GP 0.05\n",
            "Test Results - Epoch: 13 Acc: 0.9950 BCE: 0.11 GP 0.02\n",
            "Test Results - Epoch: 14 Acc: 1.0000 BCE: 0.10 GP 0.03\n",
            "Test Results - Epoch: 15 Acc: 1.0000 BCE: 0.09 GP 0.03\n",
            "Test Results - Epoch: 16 Acc: 1.0000 BCE: 0.08 GP 0.02\n",
            "Test Results - Epoch: 17 Acc: 1.0000 BCE: 0.08 GP 0.04\n",
            "Test Results - Epoch: 18 Acc: 0.9950 BCE: 0.09 GP 0.02\n",
            "Test Results - Epoch: 19 Acc: 0.9950 BCE: 0.08 GP 0.03\n",
            "Test Results - Epoch: 20 Acc: 0.9950 BCE: 0.08 GP 0.02\n",
            "Test Results - Epoch: 21 Acc: 0.9950 BCE: 0.08 GP 0.02\n",
            "Test Results - Epoch: 22 Acc: 1.0000 BCE: 0.08 GP 0.03\n",
            "Test Results - Epoch: 23 Acc: 1.0000 BCE: 0.08 GP 0.02\n",
            "Test Results - Epoch: 24 Acc: 0.9950 BCE: 0.08 GP 0.02\n",
            "Test Results - Epoch: 25 Acc: 1.0000 BCE: 0.08 GP 0.02\n",
            "Test Results - Epoch: 26 Acc: 1.0000 BCE: 0.08 GP 0.03\n",
            "Test Results - Epoch: 27 Acc: 0.9950 BCE: 0.08 GP 0.02\n",
            "Test Results - Epoch: 28 Acc: 0.9950 BCE: 0.08 GP 0.02\n",
            "Test Results - Epoch: 29 Acc: 1.0000 BCE: 0.07 GP 0.03\n",
            "Test Results - Epoch: 30 Acc: 0.9950 BCE: 0.08 GP 0.02\n"
          ],
          "name": "stdout"
        },
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "State:\n",
              "\titeration: 690\n",
              "\tepoch: 30\n",
              "\tepoch_length: 23\n",
              "\tmax_epochs: 30\n",
              "\toutput: 0.09503614157438278\n",
              "\tbatch: <class 'list'>\n",
              "\tmetrics: <class 'dict'>\n",
              "\tdataloader: <class 'torch.utils.data.dataloader.DataLoader'>\n",
              "\tseed: <class 'NoneType'>\n",
              "\ttimes: <class 'dict'>"
            ]
          },
          "metadata": {
            "tags": []
          },
          "execution_count": 86
        }
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "CVvsBbANrlBL",
        "outputId": "e8f1b465-5334-4b3a-d3fe-3a74b14da83a",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 289
        }
      },
      "source": [
        "domain = 3\n",
        "x_lin = np.linspace(-domain+0.5, domain+0.5, 100)\n",
        "y_lin = np.linspace(-domain, domain, 100)\n",
        "\n",
        "xx, yy = np.meshgrid(x_lin, y_lin)\n",
        "\n",
        "X_grid = np.column_stack([xx.flatten(), yy.flatten()])\n",
        "\n",
        "X_vis, y_vis = sklearn.datasets.make_moons(n_samples=1000, noise=noise)\n",
        "mask = y_vis.astype(np.bool)\n",
        "\n",
        "with torch.no_grad():\n",
        "    output = model(torch.from_numpy(X_grid).float())[1]\n",
        "    confidence = output.max(1)[0].numpy()\n",
        "\n",
        "\n",
        "z = confidence.reshape(xx.shape)\n",
        "\n",
        "plt.figure()\n",
        "plt.contourf(x_lin, y_lin, z, cmap='cividis')\n",
        "\n",
        "plt.scatter(X_vis[mask,0], X_vis[mask,1])\n",
        "plt.scatter(X_vis[~mask,0], X_vis[~mask,1])"
      ],
      "execution_count": 87,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "<matplotlib.collections.PathCollection at 0x7fb6867378d0>"
            ]
          },
          "metadata": {
            "tags": []
          },
          "execution_count": 87
        },
        {
          "output_type": "display_data",
          "data": {
            "image/png": "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\n",
            "text/plain": [
              "<Figure size 432x288 with 1 Axes>"
            ]
          },
          "metadata": {
            "tags": []
          }
        }
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "n-VLIgo7rlBT"
      },
      "source": [
        ""
      ],
      "execution_count": 87,
      "outputs": []
    }
  ]
}