{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Graph Dynamical Systems\n",
    "\n",
    "This notebooks contains the experiments to evaluate graph edit networks on simple graph dynamical systems, namely the edit cycles, degree rules, and game of life datasets."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Hyperparameter setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "import numpy as np\n",
    "import torch\n",
    "import pytorch_graph_edit_networks as gen\n",
    "import baseline_models\n",
    "import hep_th\n",
    "\n",
    "# model hyperparameters\n",
    "num_layers = 2\n",
    "dim_hid = 64\n",
    "\n",
    "# training hyperparameters\n",
    "learning_rate  = 1E-3\n",
    "weight_decay   = 1E-5\n",
    "loss_threshold = 1E-3\n",
    "max_epochs     = 50000\n",
    "print_step     = 1000\n",
    "\n",
    "# the number of repitions for each experiment\n",
    "R = 5\n",
    "# the number of test time series we use to evaluate learning afterwards\n",
    "N_test = 10"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Model setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# set up model names\n",
    "models = ['VGAE', 'GEN_crossent', 'GEN']\n",
    "\n",
    "# set up functions to initialize the models\n",
    "def setup_vgae(dim_in, nonlin):\n",
    "    return baseline_models.VGAE(num_layers = num_layers, dim_in = dim_in, dim_hid = dim_hid, beta = 1E-3, sigma_scaling = 1E-3, nonlin = nonlin)\n",
    "def setup_gen(dim_in, nonlin):\n",
    "    return gen.GEN(num_layers = num_layers, dim_in = dim_in, dim_hid = dim_hid, nonlin = nonlin)\n",
    "setup_funs = [setup_vgae, setup_gen, setup_gen]\n",
    "# set up functions to compute the loss\n",
    "loss_fun = gen.GEN_loss()\n",
    "crossent_loss_fun = gen.GEN_loss_crossent()\n",
    "def vgae_loss(model, A, X, delta, Epsilon):\n",
    "    B = A + Epsilon\n",
    "    # delete all outgoing and incoming edges of deleted nodes\n",
    "    B[delta < -0.5, :] = 0\n",
    "    B[:, delta < -0.5] = 0\n",
    "    return model.compute_loss(torch.tensor(A, dtype=torch.float), torch.tensor(B, dtype=torch.float), torch.tensor(X, dtype=torch.float))\n",
    "def gen_loss_crossent(model, A, X, delta, Epsilon):\n",
    "    delta_pred, Epsilon_pred = model(torch.tensor(A, dtype=torch.float), torch.tensor(X, dtype=torch.float))\n",
    "    return crossent_loss_fun(delta_pred, Epsilon_pred, torch.tensor(delta, dtype=torch.float), torch.tensor(Epsilon, dtype=torch.float), torch.tensor(A, dtype=torch.float))\n",
    "def gen_loss(model, A, X, delta, Epsilon):\n",
    "    delta_pred, Epsilon_pred = model(torch.tensor(A, dtype=torch.float), torch.tensor(X, dtype=torch.float))\n",
    "    return loss_fun(delta_pred, Epsilon_pred, torch.tensor(delta, dtype=torch.float), torch.tensor(Epsilon, dtype=torch.float), torch.tensor(A, dtype=torch.float))\n",
    "loss_funs = [vgae_loss, gen_loss_crossent, gen_loss]\n",
    "# set up prediction functions\n",
    "def vgae_pred(model, A, X):\n",
    "    B = model(torch.tensor(A, dtype=torch.float), torch.tensor(X, dtype=torch.float))\n",
    "    B = B.detach().numpy()\n",
    "    Epsilon = B - A\n",
    "    delta = np.zeros(A.shape[0])\n",
    "    delta[np.sum(B, 1) < 0.5] = -1.\n",
    "    Epsilon[delta < -0.5, :] = 0.\n",
    "    Epsilon[:, delta < -0.5] = 0.\n",
    "    return delta, Epsilon\n",
    "def gen_pred(model, A, X):\n",
    "    delta_pred, Epsilon_pred = model(torch.tensor(A, dtype=torch.float), torch.tensor(X, dtype=torch.float))\n",
    "    delta_pred = delta_pred.detach().numpy()\n",
    "    Epsilon_pred = Epsilon_pred.detach().numpy()\n",
    "    delta = np.zeros(A.shape[0])\n",
    "    delta[delta_pred > 0.5] = 1.\n",
    "    delta[delta_pred < -0.5] = -1.\n",
    "    Epsilon = np.zeros(A.shape)\n",
    "    Epsilon[np.logical_and(A > 0.5, Epsilon_pred < -0.5)] = -1.\n",
    "    Epsilon[np.logical_and(A < 0.5, Epsilon_pred > +0.5)] = +1.\n",
    "    return delta, Epsilon\n",
    "pred_funs = [vgae_pred, gen_pred, gen_pred]\n",
    "\n",
    "eval_criteria = ['node_ins_recall',\n",
    "                 'node_ins_precision',\n",
    "                 'node_del_recall',\n",
    "                 'node_del_precision',\n",
    "                 'edge_ins_recall',\n",
    "                 'edge_ins_precision',\n",
    "                 'edge_del_recall',\n",
    "                 'edge_del_precision']\n",
    "# set up a function to compute precision and recall\n",
    "def prec_rec(X, Y):\n",
    "    # X is the prediction, Y is the target\n",
    "    target_insertions = Y > 0.5\n",
    "    predicted_insertions = X > 0.5\n",
    "    target_deletions = Y < -0.5\n",
    "    predicted_deletions = X < -0.5\n",
    "    # first, check the insertion recall\n",
    "    if np.sum(target_insertions) < 0.5:\n",
    "        ins_rec = 1.\n",
    "    else:\n",
    "        ins_rec  = np.mean(X[target_insertions] > 0.5)\n",
    "    # then the insertion precision\n",
    "    if np.sum(predicted_insertions) < 0.5:\n",
    "        ins_prec = 1.\n",
    "    else:\n",
    "        ins_prec = np.mean(Y[predicted_insertions] > 0.5)\n",
    "    # then the deletion recall\n",
    "    if np.sum(target_deletions) < 0.5:\n",
    "        del_rec = 1.\n",
    "    else:\n",
    "        del_rec  = np.mean(X[target_deletions] < -0.5)\n",
    "    # and finally the deletion precision\n",
    "    if np.sum(predicted_deletions) < 0.5:\n",
    "        del_prec = 1.\n",
    "    else:\n",
    "        del_prec = np.mean(Y[predicted_deletions] < -0.5)\n",
    "    return ins_rec, ins_prec, del_rec, del_prec"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Dataset setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import graph_edit_cycles\n",
    "import degree_rules\n",
    "import game_of_life\n",
    "import random\n",
    "\n",
    "datasets = ['edit_cycles', 'degree_rules', 'game_of_life']\n",
    "dim_ins  = [4, 32, 1]\n",
    "\n",
    "# set up a generative function for each data set\n",
    "def generate_edit_cycle():\n",
    "    As, Xs, tuples = graph_edit_cycles.generate_time_series(random.randrange(3), random.randrange(12), random.randrange(4, 12))\n",
    "    deltas = []\n",
    "    Epsilons = []\n",
    "    for tpl in tuples:\n",
    "        deltas.append(tpl[0])\n",
    "        Epsilons.append(tpl[1])\n",
    "    return As, Xs, deltas, Epsilons\n",
    "def generate_degree_rules():\n",
    "    # the initial number of nodes in each graph\n",
    "    n_init = 8\n",
    "    # the maximum number of nodes that can occur in each graph during evolution\n",
    "    n_max  = n_init * 4\n",
    "    return degree_rules.generate_time_series_from_random_matrix(n_init, n_max = n_max)\n",
    "def generate_game_of_life():\n",
    "    # set hyper-parameters for the game of life random grid generation\n",
    "    grid_size = 10\n",
    "    num_shapes = 1\n",
    "    p = 0.1\n",
    "    T_max = 10\n",
    "    A, Xs, deltas = game_of_life.generate_random_time_series(grid_size, num_shapes, p, T_max)\n",
    "    As = [A] * len(Xs)\n",
    "    Epsilons = [np.zeros_like(A)] * len(Xs)\n",
    "    return As, Xs, deltas, Epsilons\n",
    "generator_funs = [generate_edit_cycle, generate_degree_rules, generate_game_of_life]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Actual Experiment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--- data set edit_cycles ---\n",
      "\n",
      "--- model VGAE ---\n",
      "node_ins_recall: 0.616033 +- 0.0137438\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.690418 +- 0.0622833\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN_crossent ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 1 +- 0\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 1 +- 0\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "\n",
      "--- data set degree_rules ---\n",
      "\n",
      "--- model VGAE ---\n",
      "node_ins_recall: 0.146334 +- 0.033497\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.95537 +- 0.0244676\n",
      "edge_ins_recall: 0.881462 +- 0.0254464\n",
      "edge_ins_precision: 0.973631 +- 0.0516198\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 0.965385 +- 0.0692308\n",
      "--- model GEN_crossent ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 1 +- 0\n",
      "edge_ins_recall: 0.967017 +- 0.0453511\n",
      "edge_ins_precision: 0.9869 +- 0.0167524\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 1 +- 0\n",
      "edge_ins_recall: 0.970246 +- 0.0569909\n",
      "edge_ins_precision: 0.985629 +- 0.0287424\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "\n",
      "--- data set game_of_life ---\n",
      "\n",
      "--- model VGAE ---\n",
      "node_ins_recall: 0.268 +- 0.0808455\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.0341 +- 0.00352307\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN_crossent ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 1 +- 0\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.979732 +- 0.0405352\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n",
      "--- model GEN ---\n",
      "node_ins_recall: 1 +- 0\n",
      "node_ins_precision: 0.997893 +- 0.0042144\n",
      "node_del_recall: 1 +- 0\n",
      "node_del_precision: 0.9991 +- 0.00111355\n",
      "edge_ins_recall: 1 +- 0\n",
      "edge_ins_precision: 1 +- 0\n",
      "edge_del_recall: 1 +- 0\n",
      "edge_del_precision: 1 +- 0\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import time\n",
    "# iterate over all datasets\n",
    "for d in range(len(datasets)):\n",
    "    print('\\n--- data set %s ---\\n' % datasets[d])\n",
    "    # load partial runtime results if possible\n",
    "    runtimes_file = 'results/%s_runtimes.csv' % datasets[d]\n",
    "    if os.path.exists(runtimes_file):\n",
    "        runtimes = np.loadtxt(runtimes_file, skiprows = 1, delimiter = '\\t')\n",
    "    else:\n",
    "        runtimes = np.full((R, len(models)), np.nan)\n",
    "    # iterate over all models\n",
    "    for k in range(len(models)):\n",
    "        print('--- model %s ---' % models[k])\n",
    "        # load partial results if possible\n",
    "        results_file = 'results/%s_%s_results.csv' % (datasets[d], models[k])\n",
    "        curves_file  = 'results/%s_%s_learning_curves.csv' % (datasets[d], models[k])\n",
    "        if os.path.exists(results_file):\n",
    "            results = np.loadtxt(results_file, skiprows = 1, delimiter = '\\t')\n",
    "            learning_curves = np.loadtxt(curves_file, delimiter = '\\t')\n",
    "        else:\n",
    "            results = np.full((R, len(eval_criteria)), np.nan)\n",
    "            learning_curves = np.full((max_epochs, R), np.nan)\n",
    "        # iterate over experimental repeats\n",
    "        for r in range(R):\n",
    "            # check if this repeat is already evaluated; if so, skip it\n",
    "            if not np.isnan(learning_curves[0, r]):\n",
    "                continue\n",
    "            print('-- repeat %d of %d --' % (r+1, R))\n",
    "            start_time = time.time()\n",
    "            # set up model\n",
    "            if datasets[d] == 'game_of_life':\n",
    "                nonlin = torch.nn.Sigmoid()\n",
    "            else:\n",
    "                nonlin = torch.nn.ReLU()\n",
    "            model = setup_funs[k](dim_ins[d], nonlin)\n",
    "            # set up optimizer\n",
    "            optimizer = torch.optim.Adam(model.parameters(), lr=learning_rate, weight_decay=weight_decay)\n",
    "            # initialize moving loss average for printing\n",
    "            loss_avg = None\n",
    "            # start training\n",
    "            for epoch in range(max_epochs):\n",
    "                optimizer.zero_grad()\n",
    "                # sample a time series from the data set\n",
    "                As, Xs, deltas, Epsilons = generator_funs[d]()\n",
    "                # compute the loss over all time steps\n",
    "                loss = 0.\n",
    "                for t in range(len(As)):\n",
    "                    # compute loss\n",
    "                    loss_obj = loss_funs[k](model, As[t], Xs[t], deltas[t], Epsilons[t])\n",
    "                    # compute gradient\n",
    "                    loss_obj.backward()\n",
    "                    # accumulate loss\n",
    "                    loss += loss_obj.item()\n",
    "                # perform an optimizer step\n",
    "                optimizer.step()\n",
    "                # store the current loss value in the learning curve\n",
    "                learning_curves[epoch, r] = loss\n",
    "                # compute a new moving average over the loss\n",
    "                if loss_avg is None:\n",
    "                    loss_avg = loss\n",
    "                else:\n",
    "                    loss_avg = loss_avg * 0.9 + 0.1 * loss\n",
    "                # print every print_step steps\n",
    "                if(epoch+1) % print_step == 0:\n",
    "                    print('loss avg after %d epochs: %g' % (epoch+1, loss_avg))\n",
    "                # stop early if the moving average is small\n",
    "                if loss_avg < loss_threshold:\n",
    "                    break\n",
    "            # perform evaluation on new time series\n",
    "            results[r, :] = 0.\n",
    "            T = 0\n",
    "            for j in range(N_test):\n",
    "                # get a random time series from the dataset\n",
    "                As, Xs, deltas, Epsilons = generator_funs[d]()\n",
    "                for t in range(len(As)):\n",
    "                    # predict the current time step with the network\n",
    "                    delta, Epsilon = pred_funs[k](model, As[t], Xs[t])\n",
    "                    # assess node edit precision and recall\n",
    "                    results[r, :4] += prec_rec(delta, deltas[t])\n",
    "                    # assess edge edit precision and recall\n",
    "                    results[r, 4:] += prec_rec(Epsilon, Epsilons[t])\n",
    "                    # TODO this is only for debugging purposes\n",
    "                    if (d != 1 and k == 2 and np.any(results[r, :] < 0.99)) or np.any(np.isnan(results[r, :])):\n",
    "                        print('delta (predicted) = %s' % str(delta))\n",
    "                        print('delta (target) = %s' % str(deltas[t]))\n",
    "                        print('Epsilon (predicted) = %s' % str(Epsilon))\n",
    "                        print('Epsilon (target) = %s' % str(Epsilons[t]))\n",
    "                        raise ValueError('stop')\n",
    "                        \n",
    "                T += len(As)\n",
    "            results[r, :] /= T\n",
    "            # store runtime\n",
    "            runtimes[r, k] = time.time() - start_time\n",
    "            np.savetxt(runtimes_file, runtimes, delimiter = '\\t', fmt = '%g', header = '\\t'.join(models), comments = '')\n",
    "            # store results\n",
    "            np.savetxt(results_file, results, delimiter = '\\t', fmt = '%g', header = '\\t'.join(eval_criteria), comments = '')\n",
    "            # store learning curves\n",
    "            np.savetxt(curves_file, learning_curves, delimiter = '\\t', fmt = '%g')\n",
    "        # print results\n",
    "        for crit in range(len(eval_criteria)):\n",
    "            print('%s: %g +- %g' % (eval_criteria[crit], np.mean(results[:, crit]), np.std(results[:, crit])))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/usr/lib/python3.7/site-packages/ipykernel_launcher.py:9: RuntimeWarning: Mean of empty slice\n",
      "  if __name__ == '__main__':\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# visualize learning curves\n",
    "import matplotlib.pyplot as plt\n",
    "smoothing_steps = 10\n",
    "fig, axes = plt.subplots(ncols=1, nrows=len(datasets))\n",
    "for d in range(len(datasets)):\n",
    "    for k in range(len(models)):\n",
    "        curves_file  = 'results/%s_%s_learning_curves.csv' % (datasets[d], models[k])\n",
    "        learning_curves = np.loadtxt(curves_file, delimiter = '\\t')\n",
    "        acum = np.cumsum(np.nanmean(learning_curves, 1))\n",
    "        axes[d].semilogy((acum[smoothing_steps:] - acum[:-smoothing_steps])/smoothing_steps)\n",
    "    axes[d].set_xlabel('epoch')\n",
    "    axes[d].set_ylabel('loss')\n",
    "    axes[d].set_title(datasets[d])\n",
    "    axes[d].legend(models)\n",
    "plt.show()"
   ]
  }
 ],
 "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.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
