{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Topological regularization of Karate network embedding\n",
    "\n",
    "In this notebook, we show how a topological loss can be combined with a graph embedding procedure, as to regularize the embedding and better reflect the topological---in this case disconnected---prior. \n",
    "\n",
    "We start by setting the working directory and importing the necessary libraries."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Set working directory\n",
    "import os\n",
    "os.chdir(\"..\")\n",
    "\n",
    "# Tracking computation times\n",
    "import time\n",
    "\n",
    "# Handling arrays and data.frames\n",
    "import pandas as pd \n",
    "import numpy as np\n",
    "\n",
    "# Working with graphs in Python\n",
    "import networkx as nx \n",
    "\n",
    "# Pytorch compatible topology layer and losses\n",
    "import torch\n",
    "from topologylayer.nn import AlphaLayer\n",
    "from Code.losses import DiagramLoss, deepwalk_loss\n",
    "\n",
    "# Random sampling for topological loss\n",
    "import random\n",
    "\n",
    "# Ordinary and topologically regularized graph embedding\n",
    "from Code.topembed import DeepWalk\n",
    "\n",
    "# Plotting\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# Quantitative evaluation\n",
    "from sklearn.svm import SVC\n",
    "from Code.evaluation import evaluate_embeddings\n",
    "\n",
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Load and plot graph\n",
    "\n",
    "We now load and plot the Karate network."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of nodes: 34\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Import the graph from the networkx library\n",
    "G = nx.karate_club_graph()\n",
    "print(\"Number of nodes: \" + str(len(G.nodes())))\n",
    "\n",
    "# Obtain the node grouping labels\n",
    "node_labels = {0:\"Mr. Hi\", 33:\"John A.\"}\n",
    "node_color = [\"Orange\" if G.nodes()[n][\"club\"] == node_labels[0] else \"royalblue\" for n in G.nodes()]\n",
    "\n",
    "# The graph imported from networkx is unweighted, we use a local file (without group labels) to assign the weights\n",
    "adj = pd.read_csv(os.path.join(\"Data\", \"Karate.txt\"), header=None, sep=\" \")\n",
    "for index, row in adj.iterrows():\n",
    "    G[row[0]][row[1]][\"weight\"] = row[2]\n",
    "del adj\n",
    "\n",
    "# Plot the graph\n",
    "G_layout = G.copy()\n",
    "for u, v, d in G_layout.edges(data=True):\n",
    "    d[\"weight\"] = d[\"weight\"] ** -1 # make larger weight edges shorter in the plot\n",
    "layout = nx.spring_layout(G_layout, seed=17)\n",
    "del G_layout\n",
    "nx.draw(G, layout, node_color=node_color, labels=node_labels, font_size=16, node_size=100, edge_color=\"gray\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Regular graph embedding\n",
    "\n",
    "We first take a look at a regular embedding of this graph without topological regularization."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[epoch 1] [emb. loss: 2630.249512, top. loss: 0.000000, total loss: 2630.249512]\n",
      "[epoch 5] [emb. loss: 2398.659180, top. loss: 0.000000, total loss: 2398.659180]\n",
      "[epoch 10] [emb. loss: 2303.847168, top. loss: 0.000000, total loss: 2303.847168]\n",
      "[epoch 15] [emb. loss: 2117.511475, top. loss: 0.000000, total loss: 2117.511475]\n",
      "[epoch 20] [emb. loss: 2124.620850, top. loss: 0.000000, total loss: 2124.620850]\n",
      "[epoch 25] [emb. loss: 2072.329590, top. loss: 0.000000, total loss: 2072.329590]\n",
      "[epoch 30] [emb. loss: 2131.775879, top. loss: 0.000000, total loss: 2131.775879]\n",
      "[epoch 35] [emb. loss: 2057.221191, top. loss: 0.000000, total loss: 2057.221191]\n",
      "[epoch 40] [emb. loss: 2063.017090, top. loss: 0.000000, total loss: 2063.017090]\n",
      "[epoch 45] [emb. loss: 1962.598877, top. loss: 0.000000, total loss: 1962.598877]\n",
      "[epoch 50] [emb. loss: 2080.983398, top. loss: 0.000000, total loss: 2080.983398]\n",
      "Time for embedding: 00:00:29\n"
     ]
    },
    {
     "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": [
    "# Learning hyperparameters\n",
    "num_epochs = 50\n",
    "learning_rate = 1e-2\n",
    "\n",
    "# Conduct graph embedding\n",
    "model_emb = DeepWalk(G, num_epochs=num_epochs, learning_rate=learning_rate, random_state=42)\n",
    "Y_emb = model_emb.phi.detach().numpy()\n",
    "\n",
    "# View graph embedding\n",
    "fig, ax = plt.subplots()\n",
    "sns.scatterplot(x=Y_emb[:,0], y=Y_emb[:,1], s=50, c=node_color, ax=ax)\n",
    "plt.text(Y_emb[0, 0], Y_emb[0, 1], \"Mr. Hi\", horizontalalignment=\"center\", size=\"medium\", weight=\"semibold\")\n",
    "plt.text(Y_emb[33, 0], Y_emb[33, 1], \"John A.\", horizontalalignment=\"center\", size=\"medium\", weight=\"semibold\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Apply topological regularization to the embedding\n",
    "\n",
    "We now show how we can bias a graph embedding using a loss function that captures our topological prior. As a topological loss, we will use the persistence of the second most prominent component in our embedding (one will always be infinite). It is important to multiply this by a factor $\\lambda_{\\mathrm{top}} <0$, since we want this persistence to be high. To obtain this loss, we require an additional layer that constructs the alpha complex from the embedding, from which subsequently persistent homology is computed."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define topological loss\n",
    "def g(p): return p[1] - p[0] # function that returns the persistence d - b of a point (b, d)\n",
    "TopLayer = AlphaLayer(maxdim=1) # alpha complex layer\n",
    "Component2Persistence = DiagramLoss(dim=0, i=2, j=2, g=g) # compute persistence of second most prominent gap\n",
    "lambda_top = -5e1 # scalar factor that trades off embedding and topological loss\n",
    "top_frac = .25 # sample fraction for which the topological loss is computed\n",
    "sample_times = 10 # number of times we sample to approximate the expected topological loss value\n",
    "\n",
    "# Construct topological loss function\n",
    "def top_loss(output):\n",
    "    loss = torch.tensor(0).type(torch.float)\n",
    "    for _ in range(sample_times):\n",
    "        sample = random.sample(range(output.shape[0]), int(output.shape[0] * top_frac))\n",
    "        dgminfo = TopLayer(output[sample,:])\n",
    "        loss += Component2Persistence(dgminfo)\n",
    "    loss = lambda_top * loss / sample_times\n",
    "    \n",
    "    return loss"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The topological loss is computed from a small sample in the embedding each time. This prevents the same point to be selected and moved away from the rest during each iteration, encouraging more natural communities, while simultaneously increasing computational efficiency.\n",
    "\n",
    "We can now conduct the topologically regularized graph embedding as follows."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[epoch 1] [emb. loss: 2630.249512, top. loss: -22.328678, total loss: 2607.920898]\n",
      "[epoch 5] [emb. loss: 2395.988281, top. loss: -31.885687, total loss: 2364.102539]\n",
      "[epoch 10] [emb. loss: 2308.558594, top. loss: -43.800529, total loss: 2264.758057]\n",
      "[epoch 15] [emb. loss: 2110.895020, top. loss: -59.299263, total loss: 2051.595703]\n",
      "[epoch 20] [emb. loss: 2122.801270, top. loss: -76.591385, total loss: 2046.209839]\n",
      "[epoch 25] [emb. loss: 2069.678711, top. loss: -89.287186, total loss: 1980.391479]\n",
      "[epoch 30] [emb. loss: 2156.198730, top. loss: -102.294189, total loss: 2053.904541]\n",
      "[epoch 35] [emb. loss: 2069.015137, top. loss: -106.906494, total loss: 1962.108643]\n",
      "[epoch 40] [emb. loss: 2158.476074, top. loss: -108.399857, total loss: 2050.076172]\n",
      "[epoch 45] [emb. loss: 2010.940430, top. loss: -109.103287, total loss: 1901.837158]\n",
      "[epoch 50] [emb. loss: 2159.264893, top. loss: -112.310242, total loss: 2046.954590]\n",
      "Time for embedding: 00:00:29\n"
     ]
    },
    {
     "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": [
    "# Learning hyperparameters\n",
    "num_epochs = 50\n",
    "learning_rate = 1e-2\n",
    "\n",
    "# Conduct topologically regularized graph embedding\n",
    "model_top = DeepWalk(G, top_loss=top_loss, num_epochs=num_epochs, learning_rate=learning_rate, random_state=42)\n",
    "Y_top = model_top.phi.detach().numpy()\n",
    "\n",
    "# View topologically regularized graph embedding\n",
    "fig, ax = plt.subplots()\n",
    "sns.scatterplot(x=Y_top[:,0], y=Y_top[:,1], s=50, c=node_color, ax=ax)\n",
    "plt.text(Y_top[0, 0], Y_top[0, 1], \"Mr. Hi\", horizontalalignment=\"center\", size=\"medium\", weight=\"semibold\")\n",
    "plt.text(Y_top[33, 0], Y_top[33, 1], \"John A.\", horizontalalignment=\"center\", size=\"medium\", weight=\"semibold\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Compare with ordinary topological optimization\n",
    "\n",
    "For comparison, we also conduct the same topological optimization procedure directly on the ordinary graph embedding."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[epoch 1] [emb. loss: 0.000000, top. loss: -59.893780, total loss: -59.893780]\n",
      "[epoch 5] [emb. loss: 0.000000, top. loss: -137.522888, total loss: -137.522888]\n",
      "[epoch 10] [emb. loss: 0.000000, top. loss: -238.692505, total loss: -238.692505]\n",
      "[epoch 15] [emb. loss: 0.000000, top. loss: -345.417175, total loss: -345.417175]\n",
      "[epoch 20] [emb. loss: 0.000000, top. loss: -513.678772, total loss: -513.678772]\n",
      "[epoch 25] [emb. loss: 0.000000, top. loss: -642.297363, total loss: -642.297363]\n",
      "[epoch 30] [emb. loss: 0.000000, top. loss: -800.632690, total loss: -800.632690]\n",
      "[epoch 35] [emb. loss: 0.000000, top. loss: -861.788879, total loss: -861.788879]\n",
      "[epoch 40] [emb. loss: 0.000000, top. loss: -865.923523, total loss: -865.923523]\n",
      "[epoch 45] [emb. loss: 0.000000, top. loss: -1295.541382, total loss: -1295.541382]\n",
      "[epoch 50] [emb. loss: 0.000000, top. loss: -1572.111328, total loss: -1572.111328]\n",
      "Time for embedding: 00:00:01\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXIAAAD4CAYAAADxeG0DAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8QVMy6AAAACXBIWXMAAAsTAAALEwEAmpwYAAAYWklEQVR4nO3deZRV5Znv8e9TA1VAAcVQQAlKCRLEOFtBvElo047XIVRik8DVBCQdkrR01OiNDL1ux27taF8T7Y7dtmRBJMshoobGG4cWMSjScSgQFETCIMhQQJXIUEAVNTz3j3OAAmo4dYY69R5/n7XOOvu8+z17P6+yfmufvd9d29wdEREJV1a6CxARkcQoyEVEAqcgFxEJnIJcRCRwCnIRkcDldOTO+vXr5yUlJR25SxGR4C1btqzK3YtaWt+hQV5SUkJ5eXlH7lJEJHhmtrm19Tq1IiISOAW5iEjgFOQiIoFTkIuIBK5DL3aKiKRDfYOze28DGPTtmU12tqW7pKRSkItIRtu5u57/XLyf55dUYwZjxxTw9TE9GNAnc+Ivc0YiInKCA4caeXjeZyx9/9DRtqde2U9FVT133tSXbvmZcXY5M0YhItKMHZ/WHxfiRyxefogdn9anoaLUUJCLSMaqOdzy8xZqW1kXGgW5iGSsvr2y6Nn95Jjr3SOLvr2y01BRaijIRSRjDeiTw4xJfclpktm5OTB9Ul/662KniEjnZ2ZceGY+v55RzOYddQAMGZjL4P6ZFX2ZNRoRkRPkZBtDinMZUpyb7lJSRqdWREQCpyAXEQlcm0FuZvlm9o6ZrTSz1WZ2d7S9j5ktNLN10ffeqS9XREROFMsReS3wl+5+HnA+cLWZjQamAYvcfTiwKPpZREQ6WJtB7hHV0Y+50ZcDY4G50fa5QFkqChQRkdbFdI7czLLNbAWwC1jo7m8DA9y9AiD63r+F704xs3IzK6+srExS2SIickRMQe7uDe5+PjAYGGVmZ8e6A3ef5e6l7l5aVNTis0NFRCRO7Zq14u57gMXA1cBOMysGiL7vSnZxIiLStlhmrRSZWWF0uStwOfAR8DwwMdptIrAgRTWKiEgrYrmzsxiYa2bZRIJ/nrv/wcz+BMwzs+8BnwDjUliniIi0oM0gd/f3gQuaaf8UuCwVRYmISOx0Z6eISOAU5CIigVOQi4gETkEuIhI4BbmISOAU5CIigVOQi4gETkEuIhI4BbmISOAU5CIigVOQi4gETkEuIhK4WP76Yfod2gGN9ZA/ALJz012NiEin0rmPyGsqYcNv4JVL4MVzYPlPYP+GdFclItKpdN4gd4dNT8Dbk+HAJqjbA+sehjfK4GBFmosTEek8Om+QH/wEPvjZye17V0H1+g4vR0Sks+q8Qd5QA3V7m1/XUruIyOdQ5w3yvCLoU9rMCoPuQzq8HBGRzqoTB3kf+NIjkNPj+PYLfgEFQ9NTk4hIJ9S5px/2LYWry+Gz9+DwnsgRes8vQE73dFcmItJptBnkZnYq8FtgINAIzHL3fzGznwHfByqjXWe4+4tJr7DnFyIvERFpVixH5PXAHe6+3Mx6AMvMbGF03YPu/kDqyhMRkba0GeTuXgFURJf3m9kaYFCqCxMRkdi062KnmZUAFwBvR5ummtn7ZjbHzHq38J0pZlZuZuWVlZXNdRERkQTEHORmVgA8B9zm7vuAR4BhwPlEjth/0dz33H2Wu5e6e2lRUVHiFYuIyHFiCnIzyyUS4k+4++8B3H2nuze4eyPwa2BU6soUEZGWtBnkZmbAbGCNu/+ySXtxk27fAFYlvzwREWlLLLNWvgx8B/jAzFZE22YAE8zsfMCBTcAPUlCfiIi0IZZZK28C1syq5M8ZFxGRduu8t+iLiEhMFOQiIoFTkIuIBE5BLiISOAW5iEjgFOQiIoFTkIuIBE5BLiISOAW5iEjgFOQiIoFTkIuIBE5BLiISOAW5iEjgFOQiIoFTkIuIBE5BLiISOAW5iEjgFOQiIoFTkIuIBK7NIDezU83sj2a2xsxWm9mt0fY+ZrbQzNZF33unvlwRETlRLEfk9cAd7j4SGA3cYmZnAdOARe4+HFgU/SwiIh2szSB39wp3Xx5d3g+sAQYBY4G50W5zgbIU1SgiIq1o1zlyMysBLgDeBga4ewVEwh7o38J3pphZuZmVV1ZWJliuiIicKOYgN7MC4DngNnffF+v33H2Wu5e6e2lRUVE8NYqISCtiCnIzyyUS4k+4+++jzTvNrDi6vhjYlZoSRUSkNbHMWjFgNrDG3X/ZZNXzwMTo8kRgQfLLExGRtuTE0OfLwHeAD8xsRbRtBnAfMM/Mvgd8AoxLSYUiItKqNoPc3d8ErIXVlyW3HBERaS/d2SkiEjgFuYhI4BTkIiKBU5CLiAROQS4iEjgFuYhI4BTkIiKBU5CLiAROQS4iEjgFuYhI4BTkIiKBU5CLiAROQS4iEjgFuYhI4BTkIiKBU5CLiAROQS4iEjgFuYhI4BTkIiKB+/wEeV01NNSkuwoRkaRrM8jNbI6Z7TKzVU3afmZm28xsRfR1TWrLTMDBrbBhNrx2GbzxDah4FQ7vTXdVIiJJkxNDn8eAh4HfntD+oLs/kPSKkqn2Uyj/MWydf6yt4mW45Ldw+nfSV5eISBK1eUTu7m8AuzugluSr3nh8iB+x/A44sKXj6xERSYFEzpFPNbP3o6deerfUycymmFm5mZVXVlYmsLs41Fa10F4J9fs7thYRkRSJN8gfAYYB5wMVwC9a6ujus9y91N1Li4qK4txdnLoWN9/efQjkFnZoKSIiqRJXkLv7TndvcPdG4NfAqOSWlSTdh8KZtx/fZlnwpUeg2ynpqUlEJMliudh5EjMrdveK6MdvAKta6582XXrCWdOh+GrYPA/y+sKQb0Gvs9NdmYhI0rQZ5Gb2FHAp0M/MtgJ/D1xqZucDDmwCfpC6EhOUXwTFV0ZeIiIZqM0gd/cJzTTPTkEtIiISh8/PnZ0iIhlKQS4iEjgFuYhI4BTkIiKBU5CLiAROQS4iEjgFuYhI4BTkIiKBU5CLiAROQS4iEjgFuYhI4BTkIiKBU5CLiAROQS4iEjgFuYhI4BTkIiKBU5CLiAROQS4iEjgFuYhI4NoMcjObY2a7zGxVk7Y+ZrbQzNZF33untkwREWlJLEfkjwFXn9A2DVjk7sOBRdHPIiKSBm0Gubu/Aew+oXksMDe6PBcoS25ZIiISq3jPkQ9w9wqA6Hv/5JUkIiLtkfKLnWY2xczKzay8srIy1bsTEfnciTfId5pZMUD0fVdLHd19lruXuntpUVFRnLsTEZGWxBvkzwMTo8sTgQXJKUdERNorlumHTwF/AkaY2VYz+x5wH3CFma0Droh+FhGRNMhpq4O7T2hh1WVJrkVEROKgOztFRAKnIBcRCZyCXEQkcApyEZHAKchFRAKnIBcRCZyCXEQkcApyEZHAKchFRAKnIBcRCZyCXEQkcApyEZHAKchFRAKnIBcRCZyCXEQkcApyEZHAKchFRAKnIBcRCZyCXEQkcApyEZHAtfnw5daY2SZgP9AA1Lt7aTKKEhGR2CUU5FFfc/eqJGxHRETioFMrIiKBSzTIHXjFzJaZ2ZTmOpjZFDMrN7PyysrKBHcnIiInSjTIv+zuFwL/E7jFzMac2MHdZ7l7qbuXFhUVJbg7ERE5UUJB7u7bo++7gPnAqGQUJSIisYs7yM2su5n1OLIMXAmsSlZhIiISm0RmrQwA5pvZke086e4vJ6UqERGJWdxB7u4bgfOSWIuIiMRB0w9FRAKnIBcRCZyCXEQkcApyEZHAKchFRAKnIBcRCZyCXEQkcApyEZHAKchFRAKnIBcRCZyCXEQkcApyEZHAKchFRAKnIBcRCZyCXEQkGeoPQO1nadl1Ig+WEBGRun1Q9Tas/jnUVsLQSXDqDVBQ0mElKMhFRBKx/SVYOv7Y5/fuhC3z4avPQdcBHVKCTq2IiMTr4HZYfsfJ7VVLYf+6DitDQS4iEq+Gg3BoW/PrDn/aYWUoyEVE4tWlEHqd0/y6boM7rIyEgtzMrjaztWa23symJasoEZEg5PWDUf8OWV2Obx/+N1AwtMPKiPtip5llA/8GXAFsBd41s+fd/cNkFSci0un1HQ1XvQsVL8Oh7TDoeig8F7r0ZntVHSv+XMuKP9dw9tA8Ljozn0H9c5NeQiJH5KOA9e6+0d0PA78DxianLBGR9DAzzj777Fb7TJo0CTOjvLwcsnLwwnOoLrmTunN/CQMvg/witlfW8dN/3cUDj+/m1XcO8tDvPuO2B3exZWfdcdtavXo1ZoaZMWvWrLhqTiTIBwFbmnzeGm07jplNMbNyMyuvrKxMYHciIp3Ptso6Hn95Hz95aCf/9NinrNlUS129s3HbYYYN7kLvnsdi9tO9DSxefvC47z/99NMAZGVlHV1ur0SC3Jpp85Ma3Ge5e6m7lxYVFSWwOxFJhU2bNh09IrznnnuOtk+ePPloe3stXrwYM2Pq1KlH26ZOnYqZsXjxYiC2I990qq2t5fbbb+eUU06hsLCQsWPHsmXLluP6PP7kc3zhjBK+P24k5e8s5fX3DnHzD2bSJTeLO+74Mf/6d6NZ+Mj5fLHwjxz5z/juhzUcPtx4dBvz5s1j0KBBjBs3jtdff52dO3e2u9ZEgnwrcGqTz4OB7QlsT0TS7De/+Q3uzoEDB3jmmWda7VtfX5/Qvp566ikeeOCBhLaRSvfeey8PPfQQV155JXfddRd/+MMfuPHGG4/rs2jRHxlw5gRqD1TwcfmDFBZkcdbpkQuf29cv4buTfsihg/v4/W/v5pa/6g3AyJIu5OZGUn3lypWsXbuWb37zm4wbN46Ghgaee+65dteaSJC/Cww3s9PNrAswHng+ge2JSBoNHTqUjRs3snjxYp5++mnq6uoYNOjY2dIjR9nXXHMNo0aNYvTo0SdvpO4gHN5/9GNNTQ1VVVVUVVVRU1NzXNcJEyZw5513pmw8iXrxxRfJysri0UcfZfr06YwePZolS5ZQXV19tM/14++i5KIfk5WdR231VqZP6sP2qgYAup8+mQ+qv023Xqexo2Iz1Yca+IsLunL1Jd2P/so5cirl4osvZuTIkXTp0oV58+a1u9a4Z624e72ZTQX+C8gG5rj76ni3JyLpNXLkSIqKipgzZw4ff/wxZWVlrFq1im3bjr/h5dVXX+Xuu+/mtMEDYf96wMHy4dP/hvXRi3W7IiE/e/ZsZs+e3cEj6TgjzyjirS2QlZVNz67Q2AhrNx8GICevFwANjdk0Njby7oc1TCkrpKT42KyVI796brrppqNtS5YsoaKiguLiYmpra2OqI6F55O7+ort/wd2Hufu9iWxLRNJv8uTJPPPMMyxdupSbb7652T7XXXcd0/92AjcOfxNeHwv7N8D6f4v8vZGdr0Veq/8JgLHXXcnChQtZuHAhY8eGNant2muvpbGxkR/96Efcf//9vPXWW4wZM4aCgoKjfYqLcrhwRB45OUb1oUY+3HS4xe2dNzyfTRV1R4/Gly1bxvr167n++uuZP38+8+fPZ9q0aTQ2NvLss88CMGLECPr27dtmrbqzU0SOGj9+PNnZ2QwePJgrrrii2T6nDOwHy39Cfe5gas78Rxrra2DNL5rtO7ioG5d/9QIuv/xyBg/uuDsd47V7924ACgsLmTFjBrfeeisvvfQSP//5z7nuuut4/PHHj+tfWJDDtIl9waHRIS+35QvDBw410r3rscg9cgpl4sSJlJWVUVZWxm233YaZtXv2ioJcRI7q2bMnc+bM4dFHHyUrq4V4qK+GYX/NPU9uoOvwG/j9C2+Ct3Dhs+EQ7H4vdQUn0WuvvUZZWRkAX/nKV8jLy+Ohhx6ioqKCPXv2sGDBAk49NTK/47HHHsPdKS0tpV9hDv/x7FZG37iUgm5ZfPGrP+FrP9pM/2HXAjBq/EKeXbSXdz88xNBBx06r3H///bg7N9xww9G2AQMG0NjYyJtvvglEZhQ1PSffEgW5iBzn29/+Ntdcc03LHQ58ErmD8fDuY21Zec33ze4Gax+EhtjO9abTkiVL+Oijj5g0aRIzZ85s13fPPSMfgCdf3set4/tw9rDIzJW+vbL5flkhn+1v4H/f1IeSXlVJrxvA3E+a+p0ypaWlXl5e3mH7E5Ek27MKXjwXvvIM1OyA8qnQ76sw6FpYecKfWzr3Huh/Gbw/HS59EXK6pqfmDvDxtlr+841q/t+SA+R3Mb5W2o2LzsxjUFEu2yvr6FWQTe2BPVwyZAv0G9Xu7ZvZMncvbWm9HiwhIrE7sBlwyO0ZCfKcHlC1BPqcD2MWwJZnwQ1OuyHy5JyarXDWtIwOcYDtVQ1UH3TuvKkPBw410rWLUV8Pu/c1kJeXxdwX9lK5p5ERtwykTwr2r1MrIhK73J6R9wOfQP1BKP1V5M+1/vlXsOQG6HUujLgV3vkBvPs3cHArdB2Y3po7wPK1NRysbWT5RzW8/FY1K9bVsnxtLc++tp/eBVl0yzcO1jgN2b1Ssn8FuYjEruCMyN/fXnMf9BwJ+QPhnH+ASx6PnD7JzoNXvhS5IHrJXPjzw5HQz2A1hxspHZmPARVV9Vx1cQEXnZnPlp11jDorny076+lXmMO07/ahT5+eKalBp1ZEJHbdimHMfPj4CajfFzl9snU+ZHWFfhdDbi+46F/AGyJPla/eEAn3DLZibQ0zHzl2EXPNpsN8cWgXzj0jj0fn72XMBV3JyTZeWHqAc87Ip3vX9v/tmrYoyEWkfXoMgyHfgpcvhH6jI0fpG34N2xYc69P/Uug5IvLghR5npK3UVNu9r4GHn9lzUvvqjYe5anTkxqE33jvE7RN68+BTn7Fjdz3DBnU5qX+idGpFRNrv4ObIHPGdf4S8vnD230G30yB/AIz8KZxaBp+thL9c2KFPyuloNYcb2V7V/Bz6mtpGsqIH34frj7SlZpagjshFpP2sSXR8eB8UDIOSCZH55EP+F9AIJd+BvFTM0eg8uudnMXRQLhu31Z20rmteFo0OWVmQmwM9u2fRr1d2SurQEbmItF/BMMhr8nyB6g3w4f1waAcUlECvkRkf4gC9CrL522/1JvuEJP0f53blo82Rm6Bu+FoP3nzvIDNv7suAvqk5dtYNQSISn93vwZ++C3tXgWVFjsTPuwe6D0l3ZR2qvsHZtL2O15cfZOfuer58XmTO/Mp1tYz6Yj55XbLo0yObwf1zyM6O70JnWzcEKchFJH41lXCoIvIU+e6nQU63dFeUkXRnp4ikTn5R5CVppXPkIiKBU5CLiAROQS4iEjgFuYhI4BTkIiKB69Dph2ZWCWzusB0mrh+Qmkd6dB6ZPkaNL2yZPj6IbYxD3L3F6UEdGuShMbPy1uZuZoJMH6PGF7ZMHx8kZ4w6tSIiEjgFuYhI4BTkrZuV7gI6QKaPUeMLW6aPD5IwRp0jFxEJnI7IRUQCpyAXEQmcgrwVZnanmbmZ9WvSNt3M1pvZWjO7Kp31xcvM/tHM3jezFWb2ipmd0mRdJozv/5rZR9Exzjezwibrgh8fgJmNM7PVZtZoZqUnrMuUMV4dHcN6M5uW7noSZWZzzGyXma1q0tbHzBaa2broe++4Nu7uejXzAk4F/ovIDUz9om1nASuBPOB0YAOQne5a4xhbzybLPwb+I8PGdyWQE12+H7g/k8YXHctIYASwGCht0p4RYwSyo7UPBbpEx3RWuutKcExjgAuBVU3a/hmYFl2eduTfantfOiJv2YPAT4GmV4PHAr9z91p3/xhYD4xKR3GJcPd9TT5259gYM2V8r7j7kSfivgUMji5nxPgA3H2Nu69tZlWmjHEUsN7dN7r7YeB3RMYWLHd/A9h9QvNYYG50eS5QFs+2FeTNMLOvA9vcfeUJqwYBW5p83hptC46Z3WtmW4Abgf8Tbc6Y8TUxGXgpupyJ4ztRpowxU8bRlgHuXgEQfe8fz0Y+t08IMrNXgYHNrJoJzCDy8/ykrzXT1innb7Y2Pndf4O4zgZlmNh2YCvw9GTS+aJ+ZQD3wxJGvNdO/U44PYhtjc19rpq3TjrEVmTKODvG5DXJ3v7y5djM7h8i5xZVmBpGf5cvNbBSRo4JTm3QfDGxPcalxaWl8zXgSeIFIkGfM+MxsInAdcJlHT0AS0PigXf8PmwpqjK3IlHG0ZaeZFbt7hZkVA7vi2YhOrZzA3T9w9/7uXuLuJUT+QV3o7juA54HxZpZnZqcDw4F30lhuXMxseJOPXwc+ii5nyviuBu4Cvu7uB5usyojxtSFTxvguMNzMTjezLsB4ImPLNM8DE6PLE4GWfmm16nN7RB4Pd19tZvOAD4n8ZL/F3RvSXFY87jOzEUAjkVk5P4SMGt/DRGZtLIz+qnrL3X+YQePDzL4B/AooAl4wsxXuflWmjNHd681sKpGZY9nAHHdfneayEmJmTwGXAv3MbCuRX8H3AfPM7HvAJ8C4uLZ97FeniIiESKdWREQCpyAXEQmcglxEJHAKchGRwCnIRUQCpyAXEQmcglxEJHD/H9Mw6Ga8mzDeAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Learning hyperparameters\n",
    "num_epochs = 50\n",
    "learning_rate = 1e-1\n",
    "\n",
    "# Conduct topological optimization\n",
    "model_opt = DeepWalk(G, emb_loss=False, top_loss=top_loss, init=Y_emb, num_epochs=num_epochs, \n",
    "                     learning_rate=learning_rate, random_state=42)\n",
    "Y_opt = model_opt.phi.detach().numpy()\n",
    "\n",
    "# View topologically optimized graph embedding\n",
    "fig, ax = plt.subplots()\n",
    "sns.scatterplot(x=Y_opt[:,0], y=Y_opt[:,1], s=50, c=node_color, ax=ax)\n",
    "plt.text(Y_opt[0, 0], Y_opt[0, 1], \"Mr. Hi\", horizontalalignment=\"center\", size=\"medium\", weight=\"semibold\")\n",
    "plt.text(Y_opt[33, 0], Y_opt[33, 1], \"John A.\", horizontalalignment=\"center\", size=\"medium\", weight=\"semibold\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We observe that the graph embedding benefits most from combining both losses."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Quantitative evaluation\n",
    "\n",
    "First, we evaluate the different losses (embedding and topological) for all final embeddings."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[1mLosses for graph embedding: \u001b[0m\n",
      "Embedding: 2006.2999482421876\n",
      "Topological: -1.316281280517578\n",
      "\n",
      "\u001b[1mLosses for topologically optimized graph embedding: \u001b[0m\n",
      "Embedding: inf\n",
      "Topological: -28.54066162109375\n",
      "\n",
      "\u001b[1mLosses for topologically regularized graph embedding: \u001b[0m\n",
      "Embedding: 2111.991873046875\n",
      "Topological: -2.3883558654785157\n"
     ]
    }
   ],
   "source": [
    "emb_samples = 250 # number of samples for approximating (expected value of) the embedding loss\n",
    "top_samples = 250 # number of samples for approximating (expected value of) topological loss\n",
    "random.seed(42)\n",
    "np.random.seed(42)\n",
    "\n",
    "print(\"\\033[1mLosses for graph embedding: \\033[0m\")\n",
    "print(\"Embedding: \" + str(np.mean([deepwalk_loss(model_emb, G, w=3, t=6).item() for _ in range(emb_samples)])))\n",
    "print(\"Topological: \" + str(np.mean([top_loss(torch.tensor(Y_emb).type(torch.float)) for _ in range(top_samples)]) \n",
    "                            / np.abs(lambda_top)) + \"\\n\")\n",
    "\n",
    "print(\"\\033[1mLosses for topologically optimized graph embedding: \\033[0m\")\n",
    "print(\"Embedding: \" + str(np.mean([deepwalk_loss(model_opt, G, w=3, t=6).item() for _ in range(emb_samples)])))\n",
    "print(\"Topological: \" + str(np.mean([top_loss(torch.tensor(Y_opt).type(torch.float)) for _ in range(top_samples)]) \n",
    "                            / np.abs(lambda_top)) + \"\\n\")\n",
    "\n",
    "print(\"\\033[1mLosses for topologically regularized graph embedding: \\033[0m\")\n",
    "print(\"Embedding: \" + str(np.mean([deepwalk_loss(model_top, G, w=3, t=6).item() for _ in range(emb_samples)])))\n",
    "print(\"Topological: \" + str(np.mean([top_loss(torch.tensor(Y_top).type(torch.float)) for _ in range(top_samples)]) \n",
    "                            / np.abs(lambda_top)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Finally, we compare if the topologically regularized embedding improves on the ordinary graph embedding for predicting data point labels."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style type=\"text/css\">\n",
       "#T_b351c_row0_col0, #T_b351c_row2_col0 {\n",
       "  background-color: lightgreen;\n",
       "}\n",
       "</style>\n",
       "<table id=\"T_b351c_\">\n",
       "  <thead>\n",
       "    <tr>\n",
       "      <th class=\"blank level0\" >&nbsp;</th>\n",
       "      <th class=\"col_heading level0 col0\" >mean</th>\n",
       "      <th class=\"col_heading level0 col1\" >std</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th id=\"T_b351c_level0_row0\" class=\"row_heading level0 row0\" >emb</th>\n",
       "      <td id=\"T_b351c_row0_col0\" class=\"data row0 col0\" >0.972500</td>\n",
       "      <td id=\"T_b351c_row0_col1\" class=\"data row0 col1\" >0.078617</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_b351c_level0_row1\" class=\"row_heading level0 row1\" >top. opt.</th>\n",
       "      <td id=\"T_b351c_row1_col0\" class=\"data row1 col0\" >0.912500</td>\n",
       "      <td id=\"T_b351c_row1_col1\" class=\"data row1 col1\" >0.139330</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_b351c_level0_row2\" class=\"row_heading level0 row2\" >top. reg.</th>\n",
       "      <td id=\"T_b351c_row2_col0\" class=\"data row2 col0\" >0.972500</td>\n",
       "      <td id=\"T_b351c_row2_col1\" class=\"data row2 col1\" >0.078617</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n"
      ],
      "text/plain": [
       "<pandas.io.formats.style.Styler at 0x7fb4e2658100>"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Machine learning model to be used for label prediction\n",
    "Ys = {\"emb\": Y_emb, \"top. opt.\": Y_opt, \"top. reg.\": Y_top}\n",
    "model = SVC()\n",
    "scoring = \"accuracy\"\n",
    "\n",
    "# Hyperparameters for quantitative evaluation\n",
    "ntimes = 100\n",
    "test_frac = 0.1\n",
    "params = {\"C\":[0.01, 0.1, 1, 10, 100]}\n",
    "\n",
    "# Obtain performances over multiple train-test splits\n",
    "performances = evaluate_embeddings(Ys, node_color, model, scoring, params=params, stratify=node_color, \n",
    "                                   ntimes=ntimes, test_frac=test_frac, random_state=42)\n",
    "\n",
    "# View resulting performances\n",
    "pd.concat([pd.DataFrame({\"mean\":performances.mean(axis=0)}),\n",
    "           pd.DataFrame({\"std\":performances.std(axis=0)})], axis=1)\\\n",
    "            .style.highlight_max(subset=\"mean\", color=\"lightgreen\", axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
