{"cells":[{"cell_type":"code","execution_count":null,"metadata":{"id":"XlaajKHG2wZL"},"outputs":[],"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 sklearn.metrics import roc_curve, roc_auc_score\n","\n","import pickle\n","import time\n","import os\n","\n","DIM = 2048  # dimension D of input signal\n","PDIM = 2    # dimension d of parameters after embeddings\n","NOISE_LEVEL = 0.1\n","\n","import torch\n","import torch.nn as nn\n","# device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n","# print(device)"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"tDc4A7O92wZP"},"outputs":[],"source":["# Load previously generated GW and embeddings\n","\n","data_path = 'data_GW/'\n","\n","with open(data_path + 'waves_30000.pkl', 'rb') as f:\n","    waves, mass, spin = pickle.load(f)\n","\n","with open(data_path + 'waves_embeddings_30000.pkl', 'rb') as f:\n","    embeddings = pickle.load(f)\n","\n","with open(data_path + 'waves_10000_test.pkl', 'rb') as f:\n","    waves_test, _, _ = pickle.load(f)\n","\n","with open(data_path + 'waves_100k_train.pkl', 'rb') as f:\n","    waves_train, _, _ = pickle.load(f)"]},{"cell_type":"code","execution_count":null,"metadata":{"code_folding":[1,30,45,53,59,67,85,98,110],"id":"hturzVHr2wZV"},"outputs":[],"source":["# Discretize the training data embeddings, obtain\n","def preprocess_embeddings(waves, embeddings, M):\n","    X_MIN = embeddings[:,0].min()\n","    X_MAX = embeddings[:,0].max()\n","    Y_MIN = embeddings[:,1].min()\n","    Y_MAX = embeddings[:,1].max()\n","\n","    X_INC = (X_MAX - X_MIN) / M\n","    Y_INC = (Y_MAX - Y_MIN) / M\n","\n","    embeddings_pixs = np.zeros((len(embeddings), 2))\n","    for i in range(len(embeddings)):\n","        embeddings_pixs[i, 0] = np.clip(np.floor((embeddings[i, 0] - X_MIN) / X_INC), 0, M-1)\n","        embeddings_pixs[i, 1] = np.clip(np.floor((embeddings[i, 1] - Y_MIN) / Y_INC), 0, M-1)\n","\n","    counts = np.zeros((M, M))\n","    waves_rep = np.zeros((M, M, DIM))\n","\n","    for ix in range(M):\n","        for iy in range(M):\n","            idx = np.where((embeddings_pixs[:, 0] == ix) * (embeddings_pixs[:, 1] == iy))[0]\n","            counts[ix, iy] = len(idx)\n","            if len(idx) > 0:\n","                waves_rep[ix, iy] = waves[idx].mean(axis=0)\n","    mask_null = (counts == 0).astype(bool)\n","\n","    return waves_rep, mask_null\n","\n","# Gaussian smoothing with std=sigma\n","# Treats missing values as NULL instead of as 0\n","def masked_smoothing(data, mask_null, sigma):\n","\n","    V = data.copy()\n","    V[mask_null] = 0\n","    V = gaussian_filter(V, sigma, truncate=10)\n","\n","    W = np.ones_like(data)\n","    W[mask_null] = 0\n","    W = gaussian_filter(W, sigma, truncate=10)\n","\n","    return V / W\n","\n","# The vector-input version of the above function\n","def masked_smoothing_vecf(data, mask_null, sigma):\n","    # Vector form input: apply smoothing along the last dimension\n","\n","    data_smooth = np.zeros_like(data)\n","    for i in range(data.shape[-1]):\n","        data_smooth[:, :, i] = masked_smoothing(data[:, :, i], mask_null, sigma)\n","    return data_smooth\n","\n","def get_training_data():\n","    N = len(waves_train)\n","    X = np.random.randn(N, DIM) * NOISE_LEVEL\n","    X += waves_train\n","    return X\n","\n","def get_test_data():\n","    N = len(waves_test)\n","    X = np.random.randn(N * 2, DIM) * NOISE_LEVEL\n","    X[:N] += waves_test\n","    return X\n","\n","# Get neighbor indices in a square centered at (xp, yp) and their kernel weights\n","# Square side length = side*2+1\n","def get_neighbors(xp, yp, side=1):\n","    num_max = (side*2+1) ** 2\n","    neighbors = np.zeros((num_max, 2))\n","    weights = np.zeros(num_max)\n","    cnt = 0\n","\n","    for xd in range(-side, side+1):\n","        for yd in range(-side, side+1):\n","            if xd == 0 and yd == 0:\n","                continue\n","            if 0 <= xp+xd < M and 0 <= yp+yd < M:\n","                neighbors[cnt] = np.array([xp+xd, yp+yd])\n","                weights[cnt] = np.exp(- np.sqrt(xd**2 + yd**2))\n","                cnt += 1\n","\n","    return neighbors[:cnt].astype(int), weights[:cnt]\n","\n","# Estimate the gradient (Jacobian) using least squares\n","def get_gradients(waves, side=1):\n","    gradients = np.zeros((M, M, 2, DIM))\n","    for xp in range(M):\n","        for yp in range(M):\n","            neighbors, weights = get_neighbors(xp, yp, side)\n","            A = (neighbors - np.array([[xp, yp]])) * np.sqrt(weights).reshape(-1, 1)\n","            B = (waves[neighbors[:, 0], neighbors[:, 1]] - waves[xp, yp]) * np.sqrt(weights).reshape(-1, 1)\n","\n","            gradients[xp, yp] = np.linalg.inv(A.T @ A) @ (A.T @ B)\n","    return gradients\n","\n","# Get the \"ground truth\" parameters gamma for training\n","# Defined as the parameter cell that gives the highest correlation at the last layer\n","def get_optimal_gamma(X_inputs):\n","    n = len(X_inputs)\n","    gamma_opt = np.zeros((n, 2))\n","\n","    for i in range(n):\n","        corrs = waves_smooth[n_layers-1] @ X_inputs[i]\n","        gamma_opt[i] = np.unravel_index(np.argmax(corrs), (M, M))\n","\n","    return gamma_opt.astype(int)\n","\n","# Find a MF template bank with highest ROC\n","# Also stores it for future use\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"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"ZVKCxNaN2wZW"},"outputs":[],"source":["M = 30  # # discretize the embedding into an MxM grid\n","waves_rep, mask_null = preprocess_embeddings(waves, embeddings, M)"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"S-GQd8LJ2wZW"},"outputs":[],"source":["levels = [18, 3, 1, 1, 1]\n","lrs = [3900, 100, 25, 17, 22]\n","\n","n_layers = 5\n","waves_smooth = np.zeros((n_layers, M, M, DIM))\n","weights_init = np.zeros((n_layers, M, M, PDIM, DIM))\n","\n","for i in range(n_layers):\n","    side = 3 if levels[i] > 5 else 2\n","    waves_smooth[i] = masked_smoothing_vecf(waves_rep, mask_null, levels[i])\n","    weights_init[i] = get_gradients(waves_smooth[i], side=side) * lrs[i]"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"NawKzOJD2wZX"},"outputs":[],"source":["X_train = get_training_data()\n","gamma_train = get_optimal_gamma(X_train)\n","\n","X_test = get_test_data()\n","N_test = len(X_test) // 2\n","y_test = np.concatenate((np.ones(N_test), np.zeros(N_test)))"]},{"cell_type":"markdown","metadata":{"id":"Mc_Z6brp2wZX"},"source":["# Trainable network"]},{"cell_type":"code","execution_count":null,"metadata":{"code_folding":[],"id":"somHibhA2wZY"},"outputs":[],"source":["class TrainableTpopt(nn.Module):\n","    def __init__(self, weights_init):\n","        super(TrainableTpopt, self).__init__()\n","\n","        self.nlayers = len(weights_init)\n","        self.gamma_init = torch.tensor([[M//2, M//2]]).long()  # initliaze parameters (gamma) at the center\n","\n","        # Kernel width parameters for interpolation\n","        self.lams = nn.Parameter(torch.ones(self.nlayers))\n","        self.lams.requires_grad = True\n","\n","        # The W matrices\n","        self.weights = nn.Parameter(torch.tensor(weights_init).float())\n","        self.weights.requires_grad = True\n","\n","        self.cur_iter = 0\n","        self.loss_list = np.zeros(0)\n","\n","\n","    def forward(self, X, mode):\n","        if type(X) is np.ndarray:\n","            X = torch.tensor(X).float()\n","        N = len(X)\n","\n","        # Hard mode with no kernel interpolation, used during test time\n","        if mode == 'hard':\n","\n","            gamma = torch.tile(self.gamma_init, (N, 1))\n","\n","            for i_layer in range(self.nlayers):\n","                weight = self.weights[i_layer, gamma[:, 0], gamma[:, 1], :, :]\n","                gamma = gamma + torch.einsum('ijk,ik->ij', weight, X)\n","                gamma = torch.round(torch.clamp(gamma, 0, M-1)).long()\n","\n","            return gamma\n","\n","        # Soft mode with kernel interpolation, used during training time\n","        elif mode == 'soft':\n","\n","            gamma = torch.tile(self.gamma_init.unsqueeze(1), (N, 4, 1))\n","            probs = (torch.ones(N, 4) / 4).float()\n","\n","            for i_layer in range(self.nlayers):\n","                gamma_agg = torch.einsum('ijk,ij->ik', gamma.float(), probs)\n","\n","                weight_agg = torch.zeros(N, PDIM, DIM)\n","                for i_nbr in range(4):\n","                    weight = self.weights[i_layer, gamma[:,i_nbr,0], gamma[:,i_nbr,1], :, :]\n","                    weight_agg += torch.einsum('ijk,i->ijk', weight, probs[:, i_nbr])\n","\n","                gamma_agg = gamma_agg + torch.einsum('ijk,ik->ij', weight_agg, X)\n","                gamma_agg = torch.clamp(gamma_agg, 0, M-1)\n","\n","                gamma = torch.tile(torch.floor(gamma_agg).unsqueeze(1), (1, 4, 1))\n","                gamma[:, 1, :] += torch.tensor([[1, 0]])\n","                gamma[:, 2, :] += torch.tensor([[0, 1]])\n","                gamma[:, 3, :] += torch.tensor([[1, 1]])\n","                gamma = torch.clamp(gamma, 0, M-1)\n","\n","                dist = gamma - gamma_agg.unsqueeze(1)\n","                probs = torch.exp(-(dist ** 2).sum(dim=2) * self.lams[i_layer])\n","                probs = probs / probs.sum(dim=1, keepdim=True)\n","\n","                gamma = gamma.long()\n","\n","            return gamma, probs\n","\n","        else:\n","            raise ValueError(f'Wrong mode: {mode}')\n","\n","\n","    def train(self, X_train, gamma_train, n_iter, n_train_pos=None, lam=1e-4, 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(gamma_train) is np.ndarray:\n","            gamma_train = torch.tensor(gamma_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","            gamma, probs = self(X_train, 'soft')\n","            gamma = torch.einsum('ijk,ij->ik', gamma.float(), probs)\n","            loss = criterion(gamma, gamma_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":null,"metadata":{"id":"Yb-aqvlb2wZY"},"outputs":[],"source":["model = TrainableTpopt(weights_init)"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"B42MzreM2wZZ"},"outputs":[],"source":["# Evaluate MF performance\n","tmplts_mf = get_mf_best_templates(n_layers*2+1)\n","y_mf = (X_test @ tmplts_mf.T).max(axis=1)\n","fpr_mf, tpr_mf, _ = roc_curve(y_test, y_mf)\n","\n","# Evaluate TpopT performance at initialization\n","gamma = model(X_test, 'hard').detach().cpu().numpy()\n","y_init = (waves_smooth[n_layers-1, gamma[:, 0], gamma[:, 1]] * X_test).sum(axis=1)\n","fpr_init, tpr_init, _ = roc_curve(y_test, y_init)"]},{"cell_type":"code","execution_count":null,"metadata":{"scrolled":true,"id":"LkupX7RL2wZZ","outputId":"ca045a30-b921-4a66-db84-1b0438a5de4f"},"outputs":[{"name":"stdout","output_type":"stream","text":["Iter # 10, loss 6.6709\n","Iter # 20, loss 7.1846\n","Iter # 30, loss 6.8752\n","Iter # 40, loss 6.0613\n","Iter # 50, loss 5.5275\n","Iter # 60, loss 5.1745\n","Iter # 70, loss 4.3701\n","Iter # 80, loss 4.5263\n","Iter # 90, loss 5.1012\n","Iter # 100, loss 4.1970\n"]},{"data":{"text/plain":["[<matplotlib.lines.Line2D at 0x7f9793ccf8d0>]"]},"execution_count":10,"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":["# Train TpopT\n","# Can run this block repeatedly to train for more epochs\n","\n","n_epoch = 1\n","\n","batch_size = 1000\n","n_batch = len(X_train) // batch_size\n","\n","for i_epoch in range(n_epoch):\n","    for i_batch in range(n_batch):\n","\n","        X_batch = X_train[batch_size*i_batch : batch_size*(i_batch+1)]\n","        gamma_batch = gamma_train[batch_size*i_batch : batch_size*(i_batch+1)]\n","\n","        model.train(X_batch, gamma_batch, n_iter=1, lr=1e-2, disp_gap=10)\n","\n","plt.plot(model.loss_list)"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"jmDCGyu_2wZa"},"outputs":[],"source":["# Hard-mode test output\n","gamma = model(X_test, 'hard').detach().cpu().numpy()\n","\n","y_trained = (waves_smooth[n_layers-1, gamma[:, 0], gamma[:, 1]] * X_test).sum(axis=1)\n","fpr_trained, tpr_trained, _ = roc_curve(y_test, y_trained)\n","auc_trained = roc_auc_score(y_test, y_trained)\n","\n","# Soft-mode test output\n","gamma, probs = model(X_test, 'soft')\n","gamma = gamma.detach().float().cpu().numpy()\n","probs = probs.detach().cpu().numpy()\n","gamma = np.round(np.einsum('ijk,ij->ik', gamma, probs)).astype(int)\n","\n","y_trained_soft = (waves_smooth[n_layers-1, gamma[:, 0], gamma[:, 1]] * X_test).sum(axis=1)\n","fpr_trained_soft, tpr_trained_soft, _ = roc_curve(y_test, y_trained_soft)\n","auc_trained_soft = roc_auc_score(y_test, y_trained_soft)"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"1u0X5LvA2wZb","outputId":"818cde9c-234a-4d99-f01c-50882a319415"},"outputs":[{"name":"stdout","output_type":"stream","text":["0.9801293099999999\n","0.981165\n"]},{"data":{"image/png":"iVBORw0KGgoAAAANSUhEUgAAAfgAAAHkCAYAAADSPD2fAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/MnkTPAAAACXBIWXMAAAsTAAALEwEAmpwYAAA+FUlEQVR4nO3dd5xU9b3/8deHZWWpUkQUkKKCFBFQwAIqtohootGY2GOMchP1RpOYqyk/TTBR9Jp7o9HEitHY0asSQuwSUSwsAiK9iLIUpfey5fP7Y2bX2WF3tp45M2fez8eDB3PKzH4OC/vm+z3f8/2auyMiIiLR0iTsAkRERKTxKeBFREQiSAEvIiISQQp4ERGRCFLAi4iIRJACXkREJIICC3gzG29mX5nZp9UcNzO7x8yWmNknZnZkwrHvm9ni+K/vB1WjiIhIVAXZgv8bMCrF8TOAXvFfY4C/AphZe+AW4GhgGHCLmbULsE4REZHICSzg3f0dYEOKU84GHveYD4C2ZnYgcDrwurtvcPeNwOuk/o+CiIiIJAnzHnwXYEXCdlF8X3X7RUREpJaahl1AQ5jZGGLd+7Rs2fKoPn36hFyRiIhI+syYMWOdu3es6liYAb8SOChhu2t830pgZNL+KVV9gLs/CDwIMGTIEC8sLAyiThERkYxkZp9XdyzMLvqJwGXx0fTHAJvdfTXwKvANM2sXH1z3jfg+ERERqaXAWvBm9jSxlvh+ZlZEbGR8PoC73w9MBkYDS4AdwA/ixzaY2a3A9PhHjXX3VIP1REREJElgAe/uF9Zw3IFrqjk2HhgfRF0iIiK5IKsH2YmISPoUFxdTVFTErl27wi4l5xQUFNC1a1fy8/Nr/R4FvIiI1EpRURGtW7emR48emFnY5eQMd2f9+vUUFRXRs2fPWr9Pc9GLiEit7Nq1iw4dOijc08zM6NChQ517ThTwIiJSawr3cNTnz10BLyIikbJ7925OPfVUBg0axLPPPht2OaHRPXgREYmUmTNnAjBr1qxwCwmZWvAiIpI1li9fTp8+fbj88svp3bs3F198MW+88QbDhw+nV69efPTRR1xyySVMnz6dQYMGsXTp0rBLDo1a8CIiUme/+8dc5q3a0qif2a9zG275Zv8az1uyZAkTJkxg/PjxDB06lKeeeop3332XiRMnctttt/Hwww9z1113MWnSpEatL9so4EVEJKv07NmTAQMGANC/f39OOeUUzIwBAwawfPnycIvLIAp4ERGps9q0tIPSrFmzitdNmjSp2G7SpAklJSVhlZVxdA9eREQkghTwIiIiEWSxNV+yn9aDFxEJ1vz58+nbt2/YZeSsqv78zWyGuw+p6ny14EVERCJIAS8iIhJBCngREZEIUsCLiIhEkAJeREQkghTwIiIiEaSAFxGRrHHcccfVeM6VV17JvHnzALjtttuCLiljKeBFRCRrTJs2rcZzHn74Yfr16wco4EVERLJCq1atAJgyZQojR47kO9/5Dn369OHiiy+mfOK2kSNHUlhYyE033cTOnTsZNGgQF198cZhlh0KLzYiISN396yZYM6dxP/OAAXDGuFqfPnPmTObOnUvnzp0ZPnw47733HiNGjKg4Pm7cOO69915mzZrVuHVmCbXgRUQkKw0bNoyuXbvSpEkTBg0apKVik6gFLyIidVeHlnZQEpeNzcvL01KxSdSCFxGRyMrPz6e4uDjsMkKhgBcRkcgaM2YMRxxxRE4OstNysSIiUitaLjZcWi5WREREFPAiIiJRpIAXERGJIAW8iIhIBCngRUREIkgBLyIiEkEKeBERyRqbNm3iL3/5S53fN3r0aDZt2tQoNZQveJPpFPAiIpI1qgv4mqapnTx5Mm3btg2oqsykuehFRCRr3HTTTSxdupRBgwaRn59PQUEB7dq1Y8GCBSxatIhzzjmHFStWsGvXLq677jrGjBkDQI8ePSgsLGTbtm2cccYZjBgxgmnTptGlSxdefvllmjdvztKlS7nmmmtYu3YtLVq04KGHHqJPnz589tlnXHTRRWzbto2zzz475D+B2lPAi4hInd3x0R0s2LCgUT+zT/s+3DjsxpTnjBs3jk8//ZRZs2YxZcoUzjzzTD799FN69uwJwPjx42nfvj07d+5k6NChnHfeeXTo0KHSZyxevJinn36ahx56iO9+97u88MILXHLJJYwZM4b777+fXr168eGHH3L11Vfz1ltvcd111/HjH/+Yyy67jPvuu69RrzlICngREclaw4YNqwh3gHvuuYcXX3wRgBUrVrB48eK9Ar5nz54MGjQIgKOOOorly5ezbds2pk2bxvnnn19x3u7duwF47733eOGFFwC49NJLufHG1P8JyRQKeBERqbOaWtrp0rJly4rXU6ZM4Y033uD999+nRYsWjBw5kl27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\n","text/plain":["<Figure size 576x576 with 1 Axes>"]},"metadata":{"needs_background":"light"},"output_type":"display_data"}],"source":["print(auc_trained)\n","print(auc_trained_soft)\n","\n","plt.figure(figsize=(8,8))\n","\n","plt.plot(fpr_mf, tpr_mf)\n","plt.plot(fpr_init, tpr_init)\n","plt.plot(fpr_trained, tpr_trained)\n","plt.plot(fpr_trained_soft, tpr_trained_soft)\n","\n","plt.xlabel('FPR')\n","plt.ylabel('TPR')\n","plt.xlim(0, 0.15)\n","plt.ylim(0.85, 1)\n","\n","plt.legend(['mf', 'init', 'trained', 'trained_soft'])\n","plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"bSCbz5xC2wZb"},"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"},"colab":{"provenance":[]}},"nbformat":4,"nbformat_minor":0}