{
  "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": "two_moons.ipynb",
      "provenance": []
    }
  },
  "cells": [
    {
      "cell_type": "code",
      "metadata": {
        "id": "GMpIxiqqQCBs",
        "outputId": "a8d87392-e161-4af5-ffb9-9ec541d9cbe0",
        "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==0.4.0.dev20200404\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": 338,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "Requirement already satisfied: pytorch-ignite==0.4.0.dev20200404 in /usr/local/lib/python3.6/dist-packages (0.4.0.dev20200404)\n",
            "Requirement already satisfied: torch in /usr/local/lib/python3.6/dist-packages (from pytorch-ignite==0.4.0.dev20200404) (1.6.0+cu101)\n",
            "Requirement already satisfied: numpy in /usr/local/lib/python3.6/dist-packages (from torch->pytorch-ignite==0.4.0.dev20200404) (1.18.5)\n",
            "Requirement already satisfied: future in /usr/local/lib/python3.6/dist-packages (from torch->pytorch-ignite==0.4.0.dev20200404) (0.16.0)\n"
          ],
          "name": "stdout"
        }
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "0byQcEp_QCB1"
      },
      "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",
        "        \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 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",
        "        z = self.embed(x)\n",
        "        y_pred = self.bilinear(z)\n",
        "        \n",
        "        return z, 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": 339,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "Om3akE02QCB5"
      },
      "source": [
        "np.random.seed(0)\n",
        "torch.manual_seed(0)\n",
        "\n",
        "l_gradient_penalty = 0.8\n",
        "\n",
        "# Moons\n",
        "noise = 0.1\n",
        "X_train, y_train = sklearn.datasets.make_blobs(n_samples=[200,200],centers=[[-1.2,0],[1.2,0]])\n",
        "X_test, y_test = sklearn.datasets.make_blobs(n_samples=[200,200],centers=[[-1.2,0],[1.2,0]])\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)\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": 340,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "scrolled": false,
        "id": "ixSs2rsQQCB9",
        "outputId": "ed5f7b44-f71e-4203-c7e4-62da205a2132",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 689
        }
      },
      "source": [
        "trainer.run(dl_train, max_epochs=30)"
      ],
      "execution_count": 341,
      "outputs": [
        {
          "output_type": "stream",
          "text": [
            "Test Results - Epoch: 1 Acc: 0.8450 BCE: 1.03 GP 0.86\n",
            "Test Results - Epoch: 2 Acc: 0.8650 BCE: 0.63 GP 0.70\n",
            "Test Results - Epoch: 3 Acc: 0.8675 BCE: 0.60 GP 0.49\n",
            "Test Results - Epoch: 4 Acc: 0.8650 BCE: 0.52 GP 0.50\n",
            "Test Results - Epoch: 5 Acc: 0.8900 BCE: 0.52 GP 0.48\n",
            "Test Results - Epoch: 6 Acc: 0.8850 BCE: 0.44 GP 0.48\n",
            "Test Results - Epoch: 7 Acc: 0.8950 BCE: 0.50 GP 0.45\n",
            "Test Results - Epoch: 8 Acc: 0.8800 BCE: 0.36 GP 0.46\n",
            "Test Results - Epoch: 9 Acc: 0.8950 BCE: 0.38 GP 0.43\n",
            "Test Results - Epoch: 10 Acc: 0.8775 BCE: 0.45 GP 0.36\n",
            "Test Results - Epoch: 11 Acc: 0.8850 BCE: 0.41 GP 0.32\n",
            "Test Results - Epoch: 12 Acc: 0.8950 BCE: 0.44 GP 0.31\n",
            "Test Results - Epoch: 13 Acc: 0.8950 BCE: 0.42 GP 0.30\n",
            "Test Results - Epoch: 14 Acc: 0.8925 BCE: 0.43 GP 0.31\n",
            "Test Results - Epoch: 15 Acc: 0.8925 BCE: 0.41 GP 0.37\n",
            "Test Results - Epoch: 16 Acc: 0.8875 BCE: 0.38 GP 0.40\n",
            "Test Results - Epoch: 17 Acc: 0.9000 BCE: 0.41 GP 0.35\n",
            "Test Results - Epoch: 18 Acc: 0.8925 BCE: 0.37 GP 0.36\n",
            "Test Results - Epoch: 19 Acc: 0.8950 BCE: 0.41 GP 0.39\n",
            "Test Results - Epoch: 20 Acc: 0.9025 BCE: 0.41 GP 0.34\n",
            "Test Results - Epoch: 21 Acc: 0.8950 BCE: 0.40 GP 0.36\n",
            "Test Results - Epoch: 22 Acc: 0.8975 BCE: 0.39 GP 0.34\n",
            "Test Results - Epoch: 23 Acc: 0.8950 BCE: 0.40 GP 0.35\n",
            "Test Results - Epoch: 24 Acc: 0.8975 BCE: 0.44 GP 0.29\n",
            "Test Results - Epoch: 25 Acc: 0.8950 BCE: 0.44 GP 0.32\n",
            "Test Results - Epoch: 26 Acc: 0.9025 BCE: 0.42 GP 0.31\n",
            "Test Results - Epoch: 27 Acc: 0.9025 BCE: 0.53 GP 0.26\n",
            "Test Results - Epoch: 28 Acc: 0.9000 BCE: 0.43 GP 0.26\n",
            "Test Results - Epoch: 29 Acc: 0.9000 BCE: 0.46 GP 0.25\n",
            "Test Results - Epoch: 30 Acc: 0.9050 BCE: 0.41 GP 0.23\n"
          ],
          "name": "stdout"
        },
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "State:\n",
              "\titeration: 180\n",
              "\tepoch: 30\n",
              "\tepoch_length: 6\n",
              "\tmax_epochs: 30\n",
              "\toutput: 0.656029462814331\n",
              "\tbatch: <class 'list'>\n",
              "\tmetrics: <class 'dict'>\n",
              "\tdataloader: <class 'torch.utils.data.dataloader.DataLoader'>\n",
              "\tseed: 65605364"
            ]
          },
          "metadata": {
            "tags": []
          },
          "execution_count": 341
        }
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "HHnGHxdTQCCA",
        "outputId": "dde63cf3-ba2b-4677-d306-ffea48e0aca0",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 285
        }
      },
      "source": [
        "domain = 5\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_blobs(n_samples=[100,100],centers=[[-1.2,0],[1.2,0]])\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": 342,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "<matplotlib.collections.PathCollection at 0x7eff8a48dc50>"
            ]
          },
          "metadata": {
            "tags": []
          },
          "execution_count": 342
        },
        {
          "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": "5YY0W8u9QCCD"
      },
      "source": [
        ""
      ],
      "execution_count": 342,
      "outputs": []
    }
  ]
}