{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cpu\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.colors as colors\n",
    "from matplotlib import cm\n",
    "\n",
    "from scipy.ndimage import gaussian_filter\n",
    "from scipy.spatial import KDTree\n",
    "from sklearn.metrics import roc_curve, roc_auc_score\n",
    "from sklearn.manifold import MDS\n",
    "\n",
    "import pickle\n",
    "import time\n",
    "import os.path\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from torchvision.datasets import MNIST\n",
    "from torchvision.transforms import *\n",
    "\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "print(device)\n",
    "\n",
    "DIM = 28*28\n",
    "PDIM = 3\n",
    "\n",
    "# np.random.seed(1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "mnist_train = MNIST(root='./mnist', train=True, download=True)\n",
    "mnist_test = MNIST(root='./mnist', train=False, download=True)\n",
    "\n",
    "DIGIT = 3\n",
    "mnist_train_pos = mnist_train.data[mnist_train.targets == DIGIT].numpy()\n",
    "mnist_train_neg = mnist_train.data[mnist_train.targets != DIGIT].numpy()\n",
    "mnist_test_pos = mnist_test.data[mnist_test.targets == DIGIT].numpy()\n",
    "mnist_test_neg = mnist_test.data[mnist_test.targets != DIGIT].numpy()\n",
    "del mnist_train, mnist_test\n",
    "\n",
    "transform = Compose([ToPILImage(), \n",
    "                     RandomAffine(degrees=30, translate=(0.1, 0.1),\n",
    "                                  interpolation=InterpolationMode.BILINEAR),\n",
    "                     ToTensor()])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "6131"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(mnist_train_pos)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "code_folding": [
     0,
     5,
     11,
     22,
     34,
     42,
     64
    ]
   },
   "outputs": [],
   "source": [
    "def zero_normalize(X):\n",
    "    X = X - X.mean(axis=-1, keepdims=True)\n",
    "    X = X / (np.linalg.norm(X, axis=-1, keepdims=True) + 1e-100)\n",
    "    return X\n",
    "\n",
    "def get_neighbors(locs_all, locs_query, radius, lam):\n",
    "    dist2 = ((locs_all - locs_query.reshape(1, PDIM)) ** 2).sum(axis=1)\n",
    "    neighbors = np.where((dist2 > 0) * (dist2 < radius**2))[0]\n",
    "    weights = np.exp(- lam * dist2[neighbors])\n",
    "    return neighbors, weights\n",
    "    \n",
    "def get_gradients(radius, lam):\n",
    "    N_base = len(X_base)\n",
    "    gradients = np.zeros((N_base, PDIM, DIM))\n",
    "    \n",
    "    for i in range(N_base):\n",
    "        neighbors, weights = get_neighbors(param_train_pos, param_base[i], radius, lam)\n",
    "        A = (param_train_pos[neighbors] - param_base[i].reshape(1, PDIM)) * np.sqrt(weights).reshape(-1, 1)\n",
    "        B = (X_train_pos[neighbors] - X_base[i].reshape(1, DIM)) * np.sqrt(weights).reshape(-1, 1)\n",
    "        gradients[i] = np.linalg.inv(A.T @ A) @ (A.T @ B)\n",
    "    return gradients\n",
    "\n",
    "def evaluate_tpopt(model, X, n_nbr=1, test=True):\n",
    "    param = model(X, n_nbr).detach()\n",
    "    _, idx = model.kdtree.query(param, k=1)\n",
    "    y = (X_base[idx] * X).sum(axis=1)\n",
    "    \n",
    "    if test:\n",
    "        fpr, tpr, _ = roc_curve(y_test_true, y)\n",
    "        auc = roc_auc_score(y_test_true, y)\n",
    "        return y, fpr, tpr, auc\n",
    "    else:\n",
    "        return y\n",
    "\n",
    "def get_random_templates(n):\n",
    "    X_tmplts = np.zeros((n, DIM))\n",
    "    for i in range(n):\n",
    "        rand_idx = np.random.randint(len(mnist_train_pos))\n",
    "        X_tmplts[i] = transform(mnist_train_pos[rand_idx]).float().numpy().reshape(-1)\n",
    "    X_tmplts = zero_normalize(X_tmplts)\n",
    "    return X_tmplts\n",
    "\n",
    "def get_mf_best_templates(cplx):\n",
    "    filename = data_path + f'best_mf_cplx_{cplx}.npy'\n",
    "    \n",
    "    if os.path.exists(filename):\n",
    "        best_tmplts = np.load(filename)\n",
    "\n",
    "    else:\n",
    "        best_auc = 0\n",
    "        \n",
    "        for i_try in range(1000):\n",
    "            tmplts_mf = get_random_templates(cplx)\n",
    "\n",
    "            y_mf = (X_train @ tmplts_mf.T).max(axis=1)\n",
    "            auc = roc_auc_score(y_train_true, y_mf)\n",
    "            if auc > best_auc:\n",
    "                best_auc = auc\n",
    "                best_tmplts = tmplts_mf\n",
    "                \n",
    "        np.save(filename, best_tmplts)\n",
    "        \n",
    "    return best_tmplts\n",
    "\n",
    "def evaluate_mf(cplx, X, test=True):\n",
    "    tmplts = get_mf_best_templates(cplx)\n",
    "    y = (X @ tmplts.T).max(axis=1)\n",
    "    \n",
    "    if test:\n",
    "        fpr, tpr, _ = roc_curve(y_test_true, y)\n",
    "        auc = roc_auc_score(y_test_true, y)\n",
    "        return y, fpr, tpr, auc\n",
    "    else:\n",
    "        return y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "10000 10000\n"
     ]
    }
   ],
   "source": [
    "data_path = 'data_MNIST/'\n",
    "\n",
    "with open(data_path + 'data_dim3.pkl', 'rb') as f:\n",
    "    [X_train_pos, X_train_neg, X_test_pos, X_test_neg, param_train_pos] = pickle.load(f)\n",
    "    \n",
    "N_train = len(X_train_pos)\n",
    "N_test = len(X_test_pos)\n",
    "print(N_train, N_test)\n",
    "\n",
    "X_train = np.vstack((X_train_pos, X_train_neg))\n",
    "y_train_true = np.concatenate((np.ones(N_train), np.zeros(N_train)))\n",
    "X_test = np.vstack((X_test_pos, X_test_neg))\n",
    "y_test_true = np.concatenate((np.ones(N_test), np.zeros(N_test)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Create a random base set for later interpolation\n",
    "N_base = 1000\n",
    "idx_base = np.random.permutation(N_train)[:N_base]\n",
    "\n",
    "param_base = param_train_pos[idx_base]\n",
    "X_base = X_train_pos[idx_base]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Create a KDTree at the base set for interpolation\n",
    "kdtree = KDTree(param_base)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "N_layers = 1  # desired number of layers to use in TpopT\n",
    "\n",
    "# The following are initializations from the optimization heuristic\n",
    "# Only take the first N_layers layers\n",
    "\n",
    "lams_all = [20, 100, 60, 60, 100]\n",
    "\n",
    "weights_init_all = np.zeros((5, N_base, PDIM, DIM))\n",
    "weights_init_all[0] = get_gradients(radius=0.5, lam=lams_all[0]) * 0.9\n",
    "weights_init_all[1] = get_gradients(radius=0.5, lam=lams_all[1]) * 0.1\n",
    "weights_init_all[2] = get_gradients(radius=0.5, lam=lams_all[2]) * 0.2\n",
    "weights_init_all[3] = get_gradients(radius=0.5, lam=lams_all[3]) * 0.1\n",
    "weights_init_all[4] = get_gradients(radius=0.5, lam=lams_all[4]) * 0.1\n",
    "\n",
    "lams = lams_all[:N_layers]\n",
    "weights_init = weights_init_all[:N_layers]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "code_folding": []
   },
   "outputs": [],
   "source": [
    "class TrainableTpopt(nn.Module):\n",
    "    def __init__(self, kdtree, param_base, weights_init, lams):\n",
    "        super(TrainableTpopt, self).__init__()\n",
    "        \n",
    "        self.kdtree = kdtree\n",
    "        self.param_base = torch.tensor(param_base).float()\n",
    "\n",
    "        self.lams = nn.Parameter(torch.tensor(lams).float())\n",
    "        self.lams.requires_grad = True\n",
    "        \n",
    "        self.n_layers = len(weights_init)\n",
    "        self.weights = nn.Parameter(torch.tensor(weights_init).float())\n",
    "        self.weights.requires_grad = True\n",
    "        \n",
    "        self.param_init = torch.zeros(1, PDIM)\n",
    "        \n",
    "        self.cur_iter = 0\n",
    "        self.loss_list = np.zeros(0)\n",
    "        \n",
    "        \n",
    "    def forward(self, X, n_nbr):\n",
    "        if type(X) is np.ndarray:\n",
    "            X = torch.tensor(X).float()\n",
    "        n_input = len(X)\n",
    "        \n",
    "        param = torch.tile(self.param_init, (n_input, 1))\n",
    "            \n",
    "        for i_layer in range(self.n_layers):\n",
    "            _, idx_nbrs = self.kdtree.query(param.detach(), k=n_nbr)\n",
    "            dists = ((torch.tile(param.unsqueeze(1), (1, n_nbr, 1)) - \n",
    "                      self.param_base[idx_nbrs.flatten()].reshape(n_input, n_nbr, PDIM)) ** 2).sum(dim=-1)\n",
    "            \n",
    "            coefs = torch.exp(- self.lams[i_layer] * dists**2)\n",
    "            coefs = coefs / coefs.sum(dim=1, keepdim=True)\n",
    "            \n",
    "            param_nbrs = self.param_base[idx_nbrs.flatten()].reshape(n_input, n_nbr, PDIM)\n",
    "            param_nbrs = torch.einsum('ijk,ij->ik', param_nbrs, coefs)\n",
    "            weight_nbrs = self.weights[i_layer][idx_nbrs.flatten()].reshape(n_input, n_nbr, PDIM, DIM)\n",
    "            weight_nbrs = torch.einsum('ijkl,ij->ikl', weight_nbrs, coefs)\n",
    "            param = param_nbrs + torch.einsum('ijk,ik->ij', weight_nbrs, X)\n",
    "                \n",
    "        return param\n",
    "            \n",
    "    \n",
    "    def train(self, X_train, param_train, n_nbr, n_iter, n_train_pos=None, lr=1e-2, disp_gap=10):\n",
    "        if n_train_pos is None:\n",
    "            n_train_pos = len(X_train) // 2\n",
    "\n",
    "        if type(X_train) is np.ndarray:\n",
    "            X_train = torch.tensor(X_train).float()\n",
    "        if type(param_train) is np.ndarray:\n",
    "            param_train = torch.tensor(param_train).float()\n",
    "\n",
    "        optimizer = torch.optim.Adam(self.parameters(), lr=lr)\n",
    "        criterion = nn.MSELoss()\n",
    "\n",
    "        self.loss_list = np.concatenate((self.loss_list, np.zeros(n_iter)))\n",
    "\n",
    "        for i_iter in range(n_iter):\n",
    "\n",
    "            # Forward\n",
    "            param = self(X_train, n_nbr)\n",
    "            loss = criterion(param, param_train)\n",
    "\n",
    "            # Backward\n",
    "            optimizer.zero_grad()\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "\n",
    "            # Record loss\n",
    "            self.loss_list[self.cur_iter] = loss.detach()\n",
    "            self.cur_iter += 1\n",
    "\n",
    "            if self.cur_iter % disp_gap == 0:\n",
    "                print(\"Iter # {}, loss {:.4f}\".format(self.cur_iter, loss.item()))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "model = TrainableTpopt(kdtree, param_base, weights_init, lams)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "y_mf, fpr_mf, tpr_mf, auc_mf = evaluate_mf(N_layers*3+1, X_test)\n",
    "y_init, fpr_init, tpr_init, auc_init = evaluate_tpopt(model, X_test, n_nbr=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.79341586\n",
      "0.70993414\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "print(auc_mf)\n",
    "print(auc_init)\n",
    "\n",
    "plt.figure(figsize=(6,6))\n",
    "plt.plot(fpr_mf, tpr_mf)\n",
    "plt.plot(fpr_init, tpr_init)\n",
    "\n",
    "plt.xlabel('FPR')\n",
    "plt.ylabel('TPR')\n",
    "# plt.xlim(0, 0.2)\n",
    "# plt.ylim(0.8, 1)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Iter # 10, loss 0.0981\n",
      "Iter # 20, loss 0.0800\n",
      "Iter # 30, loss 0.0624\n",
      "Iter # 40, loss 0.0580\n",
      "Iter # 50, loss 0.0338\n",
      "Iter # 60, loss 0.0212\n",
      "Iter # 70, loss 0.0329\n",
      "Iter # 80, loss 0.0264\n",
      "Iter # 90, loss 0.0193\n",
      "Iter # 100, loss 0.0245\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7f8224f14c18>]"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "n_epoch = 1\n",
    "\n",
    "batch_size = 100\n",
    "n_batch = len(X_train_pos) // batch_size\n",
    "\n",
    "for i_epoch in range(n_epoch):\n",
    "    randidx = np.random.permutation(N_train)\n",
    "    \n",
    "    for i_batch in range(n_batch):\n",
    "        X_batch = X_train_pos[randidx[batch_size*i_batch : batch_size*(i_batch+1)]]\n",
    "        param_batch = param_train_pos[randidx[batch_size*i_batch : batch_size*(i_batch+1)]]\n",
    "\n",
    "        model.train(X_batch, param_batch, n_nbr=1, n_iter=1, lr=1e-3, disp_gap=10)\n",
    "\n",
    "plt.plot(model.loss_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.79341586\n",
      "0.81667386\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "y_trained, fpr_trained, tpr_trained, auc_trained = evaluate_tpopt(model, X_test, n_nbr=1)\n",
    "\n",
    "print(auc_mf)\n",
    "print(auc_trained)\n",
    "\n",
    "plt.figure(figsize=(6,6))\n",
    "plt.plot(fpr_mf, tpr_mf)\n",
    "plt.plot(fpr_trained, tpr_trained)\n",
    "\n",
    "plt.xlabel('FPR')\n",
    "plt.ylabel('TPR')\n",
    "plt.legend(['MF', 'TpopT'])\n",
    "# plt.xlim(0, 0.2)\n",
    "# plt.ylim(0.8, 1)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "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.7.1"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
