{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Tree Edit Networks\n",
    "\n",
    "This notebook contains applications of graph edit networks (without edge actions) to trees."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Task 1: Boolean Formula Simplification\n",
    "\n",
    "We generate a random Boolean formula over the variables $x$ and $y$ with at most 3 binary operators and then apply simplification rules until none apply anymore. The time series is the series of iteratively simpler formulae."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# set up the model\n",
    "import torch\n",
    "import numpy as np\n",
    "import pytorch_tree_edit_networks as ten\n",
    "\n",
    "# the number of experimental repititions\n",
    "R = 5\n",
    "# the number of test trees\n",
    "N_test = 10\n",
    "\n",
    "# training hyperparameters\n",
    "max_epochs     = 30000\n",
    "learning_rate  = 1E-3\n",
    "weight_decay   = 1E-5\n",
    "loss_threshold = 1E-3\n",
    "\n",
    "# model hyperparameters\n",
    "num_layers = 2\n",
    "dim_hid = 64\n",
    "skip_connections = False\n",
    "nonlin = torch.nn.ReLU()\n",
    "max_degree = 4\n",
    "\n",
    "accs = np.zeros(R)\n",
    "learning_curves = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--- repeat 1 of 5 ---\n",
      "loss avg after 100 epochs: 10.9155\n",
      "loss avg after 200 epochs: 7.55583\n",
      "loss avg after 300 epochs: 2.98292\n",
      "loss avg after 400 epochs: 2.11261\n",
      "loss avg after 500 epochs: 2.05825\n",
      "loss avg after 600 epochs: 1.38641\n",
      "loss avg after 700 epochs: 1.43654\n",
      "loss avg after 800 epochs: 1.02011\n",
      "loss avg after 900 epochs: 1.04995\n",
      "loss avg after 1000 epochs: 0.693391\n",
      "loss avg after 1100 epochs: 0.561351\n",
      "loss avg after 1200 epochs: 0.722653\n",
      "loss avg after 1300 epochs: 0.477041\n",
      "loss avg after 1400 epochs: 0.492769\n",
      "loss avg after 1500 epochs: 0.364497\n",
      "loss avg after 1600 epochs: 0.319009\n",
      "loss avg after 1700 epochs: 0.356701\n",
      "loss avg after 1800 epochs: 0.320573\n",
      "loss avg after 1900 epochs: 0.254976\n",
      "loss avg after 2000 epochs: 0.305126\n",
      "loss avg after 2100 epochs: 0.338606\n",
      "loss avg after 2200 epochs: 0.208879\n",
      "loss avg after 2300 epochs: 0.239228\n",
      "loss avg after 2400 epochs: 0.185241\n",
      "loss avg after 2500 epochs: 0.155826\n",
      "loss avg after 2600 epochs: 0.115666\n",
      "loss avg after 2700 epochs: 0.0960938\n",
      "loss avg after 2800 epochs: 0.727044\n",
      "loss avg after 2900 epochs: 0.102013\n",
      "loss avg after 3000 epochs: 0.0607593\n",
      "loss avg after 3100 epochs: 0.0790035\n",
      "loss avg after 3200 epochs: 0.0413402\n",
      "loss avg after 3300 epochs: 0.0676869\n",
      "loss avg after 3400 epochs: 0.0995017\n",
      "loss avg after 3500 epochs: 0.0950155\n",
      "loss avg after 3600 epochs: 0.0345256\n",
      "loss avg after 3700 epochs: 0.0521442\n",
      "loss avg after 3800 epochs: 0.0442452\n",
      "loss avg after 3900 epochs: 0.0285229\n",
      "loss avg after 4000 epochs: 0.0557959\n",
      "loss avg after 4100 epochs: 0.136277\n",
      "loss avg after 4200 epochs: 0.0191445\n",
      "loss avg after 4300 epochs: 0.0414565\n",
      "loss avg after 4400 epochs: 0.0443345\n",
      "loss avg after 4500 epochs: 0.0179141\n",
      "loss avg after 4600 epochs: 0.0107532\n",
      "loss avg after 4700 epochs: 0.0307581\n",
      "loss avg after 4800 epochs: 0.0109995\n",
      "loss avg after 4900 epochs: 0.0328139\n",
      "loss avg after 5000 epochs: 0.015501\n",
      "loss avg after 5100 epochs: 0.0555991\n",
      "loss avg after 5200 epochs: 0.0116589\n",
      "loss avg after 5300 epochs: 0.0167415\n",
      "loss avg after 5400 epochs: 0.013427\n",
      "loss avg after 5500 epochs: 0.00721056\n",
      "loss avg after 5600 epochs: 0.00983036\n",
      "loss avg after 5700 epochs: 0.0295205\n",
      "loss avg after 5800 epochs: 0.0109883\n",
      "loss avg after 5900 epochs: 0.0121141\n",
      "loss avg after 6000 epochs: 0.0156846\n",
      "loss avg after 6100 epochs: 0.00445747\n",
      "loss avg after 6200 epochs: 0.00393473\n",
      "loss avg after 6300 epochs: 0.0057211\n",
      "loss avg after 6400 epochs: 0.0121392\n",
      "loss avg after 6500 epochs: 0.0285515\n",
      "loss avg after 6600 epochs: 0.0205525\n",
      "loss avg after 6700 epochs: 0.0294331\n",
      "loss avg after 6800 epochs: 0.116876\n",
      "loss avg after 6900 epochs: 0.00503408\n",
      "loss avg after 7000 epochs: 0.00861059\n",
      "loss avg after 7100 epochs: 0.00352287\n",
      "loss avg after 7200 epochs: 0.00252485\n",
      "loss avg after 7300 epochs: 0.00342031\n",
      "loss avg after 7400 epochs: 0.00222541\n",
      "loss avg after 7500 epochs: 0.00343288\n",
      "loss avg after 7600 epochs: 0.00388004\n",
      "loss avg after 7700 epochs: 0.00535361\n",
      "loss avg after 7800 epochs: 0.00120976\n",
      "accuracy: 1\n",
      "--- repeat 2 of 5 ---\n",
      "loss avg after 100 epochs: 11.4888\n",
      "loss avg after 200 epochs: 5.30049\n",
      "loss avg after 300 epochs: 2.54879\n",
      "loss avg after 400 epochs: 1.99162\n",
      "loss avg after 500 epochs: 1.5784\n",
      "loss avg after 600 epochs: 1.28287\n",
      "loss avg after 700 epochs: 1.14554\n",
      "loss avg after 800 epochs: 0.82129\n",
      "loss avg after 900 epochs: 1.00857\n",
      "loss avg after 1000 epochs: 0.956465\n",
      "loss avg after 1100 epochs: 0.843515\n",
      "loss avg after 1200 epochs: 0.650758\n",
      "loss avg after 1300 epochs: 0.411524\n",
      "loss avg after 1400 epochs: 0.481285\n",
      "loss avg after 1500 epochs: 0.386655\n",
      "loss avg after 1600 epochs: 0.494299\n",
      "loss avg after 1700 epochs: 0.283991\n",
      "loss avg after 1800 epochs: 0.307339\n",
      "loss avg after 1900 epochs: 0.552039\n",
      "loss avg after 2000 epochs: 0.271725\n",
      "loss avg after 2100 epochs: 0.447227\n",
      "loss avg after 2200 epochs: 0.223221\n",
      "loss avg after 2300 epochs: 0.191377\n",
      "loss avg after 2400 epochs: 0.218144\n",
      "loss avg after 2500 epochs: 0.13937\n",
      "loss avg after 2600 epochs: 0.21009\n",
      "loss avg after 2700 epochs: 0.139978\n",
      "loss avg after 2800 epochs: 0.147977\n",
      "loss avg after 2900 epochs: 0.0863893\n",
      "loss avg after 3000 epochs: 0.0631801\n",
      "loss avg after 3100 epochs: 0.0801798\n",
      "loss avg after 3200 epochs: 0.156414\n",
      "loss avg after 3300 epochs: 0.0683408\n",
      "loss avg after 3400 epochs: 0.0696973\n",
      "loss avg after 3500 epochs: 0.0718392\n",
      "loss avg after 3600 epochs: 0.0433344\n",
      "loss avg after 3700 epochs: 0.0786118\n",
      "loss avg after 3800 epochs: 0.0512478\n",
      "loss avg after 3900 epochs: 0.0338615\n",
      "loss avg after 4000 epochs: 0.0601178\n",
      "loss avg after 4100 epochs: 0.0954047\n",
      "loss avg after 4200 epochs: 0.0379732\n",
      "loss avg after 4300 epochs: 0.045251\n",
      "loss avg after 4400 epochs: 0.0712116\n",
      "loss avg after 4500 epochs: 0.0397303\n",
      "loss avg after 4600 epochs: 0.178192\n",
      "loss avg after 4700 epochs: 0.0338208\n",
      "loss avg after 4800 epochs: 0.0289739\n",
      "loss avg after 4900 epochs: 0.0218596\n",
      "loss avg after 5000 epochs: 0.0332371\n",
      "loss avg after 5100 epochs: 0.0167236\n",
      "loss avg after 5200 epochs: 0.0121974\n",
      "loss avg after 5300 epochs: 0.087221\n",
      "loss avg after 5400 epochs: 0.0260459\n",
      "loss avg after 5500 epochs: 0.0422089\n",
      "loss avg after 5600 epochs: 0.00576627\n",
      "loss avg after 5700 epochs: 0.0127134\n",
      "loss avg after 5800 epochs: 0.00701728\n",
      "loss avg after 5900 epochs: 0.0210301\n",
      "loss avg after 6000 epochs: 0.0375724\n",
      "loss avg after 6100 epochs: 0.00747693\n",
      "loss avg after 6200 epochs: 0.00942009\n",
      "loss avg after 6300 epochs: 0.00732671\n",
      "loss avg after 6400 epochs: 0.0837769\n",
      "loss avg after 6500 epochs: 0.012343\n",
      "loss avg after 6600 epochs: 0.0265493\n",
      "loss avg after 6700 epochs: 0.00529281\n",
      "loss avg after 6800 epochs: 0.051007\n",
      "loss avg after 6900 epochs: 0.0171446\n",
      "loss avg after 7000 epochs: 0.0252793\n",
      "loss avg after 7100 epochs: 0.0108247\n",
      "loss avg after 7200 epochs: 0.00681333\n",
      "loss avg after 7300 epochs: 0.00373822\n",
      "loss avg after 7400 epochs: 0.00357433\n",
      "loss avg after 7500 epochs: 0.0175899\n",
      "loss avg after 7600 epochs: 0.00241897\n",
      "loss avg after 7700 epochs: 0.0185683\n",
      "loss avg after 7800 epochs: 0.00665568\n",
      "loss avg after 7900 epochs: 0.0169344\n",
      "loss avg after 8000 epochs: 0.00320542\n",
      "loss avg after 8100 epochs: 0.00439014\n",
      "loss avg after 8200 epochs: 0.00503776\n",
      "loss avg after 8300 epochs: 0.00863832\n",
      "loss avg after 8400 epochs: 0.00152822\n",
      "loss avg after 8500 epochs: 0.00471318\n",
      "loss avg after 8600 epochs: 0.0285923\n",
      "loss avg after 8700 epochs: 0.0405014\n",
      "loss avg after 8800 epochs: 0.00169243\n",
      "loss avg after 8900 epochs: 0.00231012\n",
      "loss avg after 9000 epochs: 0.00307749\n",
      "loss avg after 9100 epochs: 0.0446183\n",
      "loss avg after 9200 epochs: 0.00348601\n",
      "loss avg after 9300 epochs: 0.00232584\n",
      "accuracy: 1\n",
      "--- repeat 3 of 5 ---\n",
      "loss avg after 100 epochs: 13.5105\n",
      "loss avg after 200 epochs: 5.96073\n",
      "loss avg after 300 epochs: 3.39345\n",
      "loss avg after 400 epochs: 1.79905\n",
      "loss avg after 500 epochs: 2.0174\n",
      "loss avg after 600 epochs: 1.40476\n",
      "loss avg after 700 epochs: 1.32111\n",
      "loss avg after 800 epochs: 1.42873\n",
      "loss avg after 900 epochs: 0.955362\n",
      "loss avg after 1000 epochs: 0.965209\n",
      "loss avg after 1100 epochs: 1.21338\n",
      "loss avg after 1200 epochs: 0.861347\n",
      "loss avg after 1300 epochs: 0.807792\n",
      "loss avg after 1400 epochs: 0.768029\n",
      "loss avg after 1500 epochs: 0.497515\n",
      "loss avg after 1600 epochs: 0.496747\n",
      "loss avg after 1700 epochs: 0.462106\n",
      "loss avg after 1800 epochs: 0.338189\n",
      "loss avg after 1900 epochs: 0.287399\n",
      "loss avg after 2000 epochs: 0.303502\n",
      "loss avg after 2100 epochs: 0.226083\n",
      "loss avg after 2200 epochs: 0.205076\n",
      "loss avg after 2300 epochs: 0.21688\n",
      "loss avg after 2400 epochs: 0.139015\n",
      "loss avg after 2500 epochs: 0.1303\n",
      "loss avg after 2600 epochs: 0.0672693\n",
      "loss avg after 2700 epochs: 0.129459\n",
      "loss avg after 2800 epochs: 0.140854\n",
      "loss avg after 2900 epochs: 0.0911167\n",
      "loss avg after 3000 epochs: 0.0760005\n",
      "loss avg after 3100 epochs: 0.0878615\n",
      "loss avg after 3200 epochs: 0.0880494\n",
      "loss avg after 3300 epochs: 0.121215\n",
      "loss avg after 3400 epochs: 0.0403589\n",
      "loss avg after 3500 epochs: 0.0335327\n",
      "loss avg after 3600 epochs: 0.0259313\n",
      "loss avg after 3700 epochs: 0.0424105\n",
      "loss avg after 3800 epochs: 0.072894\n",
      "loss avg after 3900 epochs: 0.196263\n",
      "loss avg after 4000 epochs: 0.141366\n",
      "loss avg after 4100 epochs: 0.0329443\n",
      "loss avg after 4200 epochs: 0.0327217\n",
      "loss avg after 4300 epochs: 0.0264825\n",
      "loss avg after 4400 epochs: 0.0100329\n",
      "loss avg after 4500 epochs: 0.0105332\n",
      "loss avg after 4600 epochs: 0.0228638\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss avg after 4700 epochs: 0.0116111\n",
      "loss avg after 4800 epochs: 0.0112421\n",
      "loss avg after 4900 epochs: 0.0289628\n",
      "loss avg after 5000 epochs: 0.00699474\n",
      "loss avg after 5100 epochs: 0.00779304\n",
      "loss avg after 5200 epochs: 0.0200198\n",
      "loss avg after 5300 epochs: 0.016127\n",
      "loss avg after 5400 epochs: 0.00841532\n",
      "loss avg after 5500 epochs: 0.0057343\n",
      "loss avg after 5600 epochs: 0.0142305\n",
      "loss avg after 5700 epochs: 0.0100874\n",
      "loss avg after 5800 epochs: 0.0277419\n",
      "loss avg after 5900 epochs: 0.0229211\n",
      "loss avg after 6000 epochs: 0.0122006\n",
      "loss avg after 6100 epochs: 0.00522611\n",
      "loss avg after 6200 epochs: 0.0108277\n",
      "loss avg after 6300 epochs: 0.00559923\n",
      "loss avg after 6400 epochs: 0.00784556\n",
      "loss avg after 6500 epochs: 0.002899\n",
      "loss avg after 6600 epochs: 0.0494287\n",
      "loss avg after 6700 epochs: 0.19036\n",
      "loss avg after 6800 epochs: 0.0132539\n",
      "loss avg after 6900 epochs: 0.00730077\n",
      "loss avg after 7000 epochs: 0.00581614\n",
      "loss avg after 7100 epochs: 0.00262644\n",
      "loss avg after 7200 epochs: 0.00924977\n",
      "loss avg after 7300 epochs: 0.00967753\n",
      "loss avg after 7400 epochs: 0.00257423\n",
      "loss avg after 7500 epochs: 0.0090526\n",
      "loss avg after 7600 epochs: 0.00125577\n",
      "accuracy: 1\n",
      "--- repeat 4 of 5 ---\n",
      "loss avg after 100 epochs: 11.3265\n",
      "loss avg after 200 epochs: 5.02121\n",
      "loss avg after 300 epochs: 2.48308\n",
      "loss avg after 400 epochs: 2.05421\n",
      "loss avg after 500 epochs: 1.8412\n",
      "loss avg after 600 epochs: 1.69219\n",
      "loss avg after 700 epochs: 1.06963\n",
      "loss avg after 800 epochs: 0.827482\n",
      "loss avg after 900 epochs: 0.819082\n",
      "loss avg after 1000 epochs: 1.21167\n",
      "loss avg after 1100 epochs: 0.569988\n",
      "loss avg after 1200 epochs: 0.561763\n",
      "loss avg after 1300 epochs: 0.348291\n",
      "loss avg after 1400 epochs: 0.355184\n",
      "loss avg after 1500 epochs: 0.228032\n",
      "loss avg after 1600 epochs: 0.318641\n",
      "loss avg after 1700 epochs: 0.271809\n",
      "loss avg after 1800 epochs: 0.192228\n",
      "loss avg after 1900 epochs: 0.184694\n",
      "loss avg after 2000 epochs: 0.14973\n",
      "loss avg after 2100 epochs: 0.19708\n",
      "loss avg after 2200 epochs: 0.166011\n",
      "loss avg after 2300 epochs: 0.101891\n",
      "loss avg after 2400 epochs: 0.133813\n",
      "loss avg after 2500 epochs: 0.0874422\n",
      "loss avg after 2600 epochs: 0.0933496\n",
      "loss avg after 2700 epochs: 0.0645227\n",
      "loss avg after 2800 epochs: 0.0997345\n",
      "loss avg after 2900 epochs: 0.0439768\n",
      "loss avg after 3000 epochs: 0.0654222\n",
      "loss avg after 3100 epochs: 0.0454788\n",
      "loss avg after 3200 epochs: 0.0564456\n",
      "loss avg after 3300 epochs: 0.0354897\n",
      "loss avg after 3400 epochs: 0.0379229\n",
      "loss avg after 3500 epochs: 0.0482466\n",
      "loss avg after 3600 epochs: 0.0391452\n",
      "loss avg after 3700 epochs: 0.0494711\n",
      "loss avg after 3800 epochs: 0.0252161\n",
      "loss avg after 3900 epochs: 0.0263885\n",
      "loss avg after 4000 epochs: 0.0266838\n",
      "loss avg after 4100 epochs: 0.0189397\n",
      "loss avg after 4200 epochs: 0.0134506\n",
      "loss avg after 4300 epochs: 0.0205658\n",
      "loss avg after 4400 epochs: 0.028107\n",
      "loss avg after 4500 epochs: 1.93917\n",
      "loss avg after 4600 epochs: 0.0485346\n",
      "loss avg after 4700 epochs: 0.0268769\n",
      "loss avg after 4800 epochs: 0.0358941\n",
      "loss avg after 4900 epochs: 0.0145154\n",
      "loss avg after 5000 epochs: 0.0196243\n",
      "loss avg after 5100 epochs: 0.0170997\n",
      "loss avg after 5200 epochs: 0.0133303\n",
      "loss avg after 5300 epochs: 0.00723738\n",
      "loss avg after 5400 epochs: 0.00506452\n",
      "loss avg after 5500 epochs: 0.00921766\n",
      "loss avg after 5600 epochs: 0.00286466\n",
      "loss avg after 5700 epochs: 0.00713625\n",
      "loss avg after 5800 epochs: 0.0146656\n",
      "loss avg after 5900 epochs: 0.00936069\n",
      "loss avg after 6000 epochs: 0.00704049\n",
      "loss avg after 6100 epochs: 0.00209011\n",
      "loss avg after 6200 epochs: 0.00931545\n",
      "loss avg after 6300 epochs: 0.00714863\n",
      "loss avg after 6400 epochs: 0.00173482\n",
      "loss avg after 6500 epochs: 0.00936167\n",
      "loss avg after 6600 epochs: 0.0140557\n",
      "loss avg after 6700 epochs: 0.00616755\n",
      "loss avg after 6800 epochs: 0.0115879\n",
      "loss avg after 6900 epochs: 0.0248837\n",
      "loss avg after 7000 epochs: 0.00331168\n",
      "loss avg after 7100 epochs: 0.0154824\n",
      "loss avg after 7200 epochs: 0.00359694\n",
      "loss avg after 7300 epochs: 0.00162773\n",
      "loss avg after 7400 epochs: 0.0181334\n",
      "loss avg after 7500 epochs: 0.0546204\n",
      "loss avg after 7600 epochs: 0.00356946\n",
      "loss avg after 7700 epochs: 0.00980084\n",
      "accuracy: 1\n",
      "--- repeat 5 of 5 ---\n",
      "loss avg after 100 epochs: 11.446\n",
      "loss avg after 200 epochs: 5.38944\n",
      "loss avg after 300 epochs: 2.42651\n",
      "loss avg after 400 epochs: 1.78014\n",
      "loss avg after 500 epochs: 1.43295\n",
      "loss avg after 600 epochs: 2.09184\n",
      "loss avg after 700 epochs: 1.26756\n",
      "loss avg after 800 epochs: 1.29142\n",
      "loss avg after 900 epochs: 0.996683\n",
      "loss avg after 1000 epochs: 0.981951\n",
      "loss avg after 1100 epochs: 0.726381\n",
      "loss avg after 1200 epochs: 0.544536\n",
      "loss avg after 1300 epochs: 0.718471\n",
      "loss avg after 1400 epochs: 0.812935\n",
      "loss avg after 1500 epochs: 0.567692\n",
      "loss avg after 1600 epochs: 0.419097\n",
      "loss avg after 1700 epochs: 0.391555\n",
      "loss avg after 1800 epochs: 0.357878\n",
      "loss avg after 1900 epochs: 0.524041\n",
      "loss avg after 2000 epochs: 0.397289\n",
      "loss avg after 2100 epochs: 0.228433\n",
      "loss avg after 2200 epochs: 0.142225\n",
      "loss avg after 2300 epochs: 0.165055\n",
      "loss avg after 2400 epochs: 0.197781\n",
      "loss avg after 2500 epochs: 0.201408\n",
      "loss avg after 2600 epochs: 0.0977608\n",
      "loss avg after 2700 epochs: 0.185872\n",
      "loss avg after 2800 epochs: 0.12494\n",
      "loss avg after 2900 epochs: 0.102602\n",
      "loss avg after 3000 epochs: 0.0764152\n",
      "loss avg after 3100 epochs: 0.117364\n",
      "loss avg after 3200 epochs: 0.115275\n",
      "loss avg after 3300 epochs: 0.0900104\n",
      "loss avg after 3400 epochs: 0.0545822\n",
      "loss avg after 3500 epochs: 0.381023\n",
      "loss avg after 3600 epochs: 0.0608782\n",
      "loss avg after 3700 epochs: 0.0376809\n",
      "loss avg after 3800 epochs: 0.0326493\n",
      "loss avg after 3900 epochs: 0.0360038\n",
      "loss avg after 4000 epochs: 0.0541056\n",
      "loss avg after 4100 epochs: 0.0300824\n",
      "loss avg after 4200 epochs: 0.0158893\n",
      "loss avg after 4300 epochs: 0.0357145\n",
      "loss avg after 4400 epochs: 0.0223643\n",
      "loss avg after 4500 epochs: 0.0224189\n",
      "loss avg after 4600 epochs: 0.0211353\n",
      "loss avg after 4700 epochs: 0.0396178\n",
      "loss avg after 4800 epochs: 0.0205971\n",
      "loss avg after 4900 epochs: 0.0839435\n",
      "loss avg after 5000 epochs: 0.0204539\n",
      "loss avg after 5100 epochs: 0.0206597\n",
      "loss avg after 5200 epochs: 0.0192425\n",
      "loss avg after 5300 epochs: 0.185705\n",
      "loss avg after 5400 epochs: 0.051537\n",
      "loss avg after 5500 epochs: 0.0302943\n",
      "loss avg after 5600 epochs: 0.0182158\n",
      "loss avg after 5700 epochs: 0.0186634\n",
      "loss avg after 5800 epochs: 0.00548914\n",
      "loss avg after 5900 epochs: 0.0160483\n",
      "loss avg after 6000 epochs: 0.00959084\n",
      "loss avg after 6100 epochs: 0.0253561\n",
      "loss avg after 6200 epochs: 0.0071727\n",
      "loss avg after 6300 epochs: 0.0190434\n",
      "loss avg after 6400 epochs: 0.00990565\n",
      "loss avg after 6500 epochs: 0.0194867\n",
      "loss avg after 6600 epochs: 0.0365005\n",
      "loss avg after 6700 epochs: 0.00414416\n",
      "loss avg after 6800 epochs: 0.00693515\n",
      "loss avg after 6900 epochs: 0.00946755\n",
      "loss avg after 7000 epochs: 0.00314452\n",
      "loss avg after 7100 epochs: 0.00606915\n",
      "loss avg after 7200 epochs: 0.00291259\n",
      "loss avg after 7300 epochs: 0.0341619\n",
      "loss avg after 7400 epochs: 0.0238266\n",
      "loss avg after 7500 epochs: 0.0164465\n",
      "loss avg after 7600 epochs: 0.00308536\n",
      "loss avg after 7700 epochs: 0.00515734\n",
      "loss avg after 7800 epochs: 0.00388667\n",
      "loss avg after 7900 epochs: 0.00210545\n",
      "loss avg after 8000 epochs: 0.00177589\n",
      "loss avg after 8100 epochs: 0.0237931\n",
      "loss avg after 8200 epochs: 0.00336515\n",
      "loss avg after 8300 epochs: 0.00227493\n",
      "loss avg after 8400 epochs: 0.0294156\n",
      "loss avg after 8500 epochs: 0.00417529\n",
      "loss avg after 8600 epochs: 0.00195361\n",
      "loss avg after 8700 epochs: 0.0157526\n",
      "loss avg after 8800 epochs: 0.0017121\n",
      "loss avg after 8900 epochs: 0.018661\n",
      "loss avg after 9000 epochs: 0.00199266\n",
      "accuracy: 1\n"
     ]
    }
   ],
   "source": [
    "# learn\n",
    "import edist.tree_utils as tu\n",
    "import boolean_formulae\n",
    "\n",
    "for r in range(R):\n",
    "    print('--- repeat %d of %d ---' % (r+1, R))\n",
    "    # instantiate network and optimizer\n",
    "    net = ten.TEN(num_layers = num_layers, alphabet = boolean_formulae.alphabet,\n",
    "                  dim_hid = dim_hid, skip_connections = skip_connections, nonlin = nonlin,\n",
    "                  dim_in_extra = max_degree + 1)\n",
    "    optimizer = torch.optim.Adam(net.parameters(), lr=learning_rate, weight_decay=weight_decay)\n",
    "    # start training\n",
    "    loss_avg = None\n",
    "    learning_curve = []\n",
    "    epochs = 0\n",
    "    while epochs < max_epochs:\n",
    "        optimizer.zero_grad()\n",
    "        # sample a nontrivial time series\n",
    "        time_series = boolean_formulae.generate_time_series()\n",
    "        if len(time_series) < 2:\n",
    "            continue\n",
    "        # compute the prediction loss\n",
    "        loss = boolean_formulae.compute_loss(net, time_series)\n",
    "        # compute the gradient\n",
    "        loss.backward()\n",
    "        # perform an optimizer step\n",
    "        optimizer.step()\n",
    "        # compute a new moving average over the loss\n",
    "        if loss_avg is None:\n",
    "            loss_avg = loss.item()\n",
    "        else:\n",
    "            loss_avg = loss_avg * 0.9 + 0.1 * loss.item()\n",
    "        learning_curve.append(loss.item())\n",
    "        if((epochs+1) % 100 == 0):\n",
    "            print('loss avg after %d epochs: %g' % (epochs+1, loss_avg))\n",
    "        epochs += 1\n",
    "        if loss_avg < loss_threshold:\n",
    "            break\n",
    "    learning_curves.append(learning_curve)\n",
    "    # after training is completed, evaluate\n",
    "    j = 0\n",
    "    T = 0\n",
    "    while j < N_test:\n",
    "        # sample a random time series\n",
    "        time_series = boolean_formulae.generate_time_series()\n",
    "        if len(time_series) < 2:\n",
    "            continue\n",
    "        # iterate over the time series\n",
    "        for t in range(len(time_series)-1):\n",
    "            # perform the prediction\n",
    "            nodes, adj = time_series[t]\n",
    "            try:\n",
    "                _, nodes_actual, adj_actual = boolean_formulae.predict_step(net, nodes, adj)\n",
    "                nodes_expected, adj_expected = time_series[t+1]\n",
    "                if nodes_actual == nodes_expected and adj_actual == adj_expected:\n",
    "                    accs[r] += 1\n",
    "                else:\n",
    "                    print('expected tree %s but was actually %s' % (tu.tree_to_string(nodes_expected, adj_expected), tu.tree_to_string(nodes_actual, adj_actual)))\n",
    "            except Exception as ex:\n",
    "                try:\n",
    "                    boolean_formulae.predict_step(net, nodes, adj, verbose = True)\n",
    "                except Exception as ex2:\n",
    "                    pass\n",
    "                print('Exception for input tree %s and network output %s\\nexception was %s' % (tu.tree_to_string(nodes, adj, indent = True, with_indices = True), deltaX, str(ex)))\n",
    "        T += len(time_series)-1\n",
    "        j += 1\n",
    "    accs[r] /= T\n",
    "    print('accuracy: %g' % accs[r])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: 1 +- 0\n",
      "Epochs: 8320.2 +- 716.397\n"
     ]
    }
   ],
   "source": [
    "# print results\n",
    "print('Accuracy: %g +- %g' % (np.mean(accs), np.std(accs)))\n",
    "num_epochs = np.array(list(map(len, learning_curves)))\n",
    "print('Epochs: %g +- %g' % (np.mean(num_epochs), np.std(num_epochs)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "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": [
    "# visualize learning curves\n",
    "import matplotlib.pyplot as plt\n",
    "smoothing_steps = 10\n",
    "for r in range(R):\n",
    "    # compute a moving average before visualization\n",
    "    acum = np.cumsum(learning_curves[r])\n",
    "    plt.semilogy((acum[smoothing_steps:] - acum[:-smoothing_steps])/smoothing_steps)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Task 2: Addition\n",
    "\n",
    "We generate an addition formula of at most four numbers in the range 1-3 and then use the Peano addition axiom to compute the addition. In particular, the following four rules apply.\n",
    "\n",
    "1. +(m, 0) for any m can be replaced with m.\n",
    "2. +(m, succ(n)) can be replaced with succ(+(m, n)).\n",
    "3. +(m, n) for n in the range 0-9 can be replaced with +(m, succ(n-1)) where -1 refers to the numeric subtraction of 1.\n",
    "4. succ(n) for n in the range 0-9 can be replaced with n+1 (mod 10 because we don't permit two-digit numbers).\n",
    "\n",
    "A time series arises by applying to a current tree every rule that is applicable until a tree results which is only a single number."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# set up the model\n",
    "import torch\n",
    "import numpy as np\n",
    "import pytorch_tree_edit_networks as ten\n",
    "\n",
    "# the number of experimental repititions\n",
    "R = 5\n",
    "# the number of test trees\n",
    "N_test = 10\n",
    "\n",
    "# training hyperparameters\n",
    "max_epochs     = 30000\n",
    "learning_rate  = 1E-3\n",
    "weight_decay   = 1E-5\n",
    "loss_threshold = 1E-3\n",
    "\n",
    "# model hyperparameters\n",
    "# a single layer with sufficient neurons should suffice here, because we only need to\n",
    "# check immediate parents and children to check whether a rule applies\n",
    "num_layers = 2\n",
    "dim_hid = 64\n",
    "skip_connections = False\n",
    "nonlin = torch.nn.ReLU()\n",
    "max_degree = 2\n",
    "\n",
    "accs = np.zeros(R)\n",
    "learning_curves = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--- repeat 1 of 5 ---\n",
      "loss avg after 100 epochs: 412.88\n",
      "loss avg after 200 epochs: 188.633\n",
      "loss avg after 300 epochs: 108.75\n",
      "loss avg after 400 epochs: 83.5314\n",
      "loss avg after 500 epochs: 39.0241\n",
      "loss avg after 600 epochs: 28.1514\n",
      "loss avg after 700 epochs: 35.2105\n",
      "loss avg after 800 epochs: 15.8604\n",
      "loss avg after 900 epochs: 13.0377\n",
      "loss avg after 1000 epochs: 7.27475\n",
      "loss avg after 1100 epochs: 7.44812\n",
      "loss avg after 1200 epochs: 3.56508\n",
      "loss avg after 1300 epochs: 2.90106\n",
      "loss avg after 1400 epochs: 2.30318\n",
      "loss avg after 1500 epochs: 1.77996\n",
      "loss avg after 1600 epochs: 1.60777\n",
      "loss avg after 1700 epochs: 1.69116\n",
      "loss avg after 1800 epochs: 1.51791\n",
      "loss avg after 1900 epochs: 1.08536\n",
      "loss avg after 2000 epochs: 1.07082\n",
      "loss avg after 2100 epochs: 0.897198\n",
      "loss avg after 2200 epochs: 0.626463\n",
      "loss avg after 2300 epochs: 0.485799\n",
      "loss avg after 2400 epochs: 0.471478\n",
      "loss avg after 2500 epochs: 0.611787\n",
      "loss avg after 2600 epochs: 0.468785\n",
      "loss avg after 2700 epochs: 0.379\n",
      "loss avg after 2800 epochs: 0.527805\n",
      "loss avg after 2900 epochs: 5.23085\n",
      "loss avg after 3000 epochs: 0.530921\n",
      "loss avg after 3100 epochs: 0.406706\n",
      "loss avg after 3200 epochs: 0.329081\n",
      "loss avg after 3300 epochs: 0.260822\n",
      "loss avg after 3400 epochs: 0.226553\n",
      "loss avg after 3500 epochs: 0.184178\n",
      "loss avg after 3600 epochs: 0.1926\n",
      "loss avg after 3700 epochs: 0.135163\n",
      "loss avg after 3800 epochs: 0.151504\n",
      "loss avg after 3900 epochs: 0.178802\n",
      "loss avg after 4000 epochs: 0.104935\n",
      "loss avg after 4100 epochs: 0.110547\n",
      "loss avg after 4200 epochs: 0.102749\n",
      "loss avg after 4300 epochs: 0.099885\n",
      "loss avg after 4400 epochs: 0.0923101\n",
      "loss avg after 4500 epochs: 0.0936849\n",
      "loss avg after 4600 epochs: 0.144949\n",
      "loss avg after 4700 epochs: 0.11891\n",
      "loss avg after 4800 epochs: 0.0705811\n",
      "loss avg after 4900 epochs: 0.0627021\n",
      "loss avg after 5000 epochs: 0.060002\n",
      "loss avg after 5100 epochs: 0.0478377\n",
      "loss avg after 5200 epochs: 0.0466103\n",
      "loss avg after 5300 epochs: 0.0473712\n",
      "loss avg after 5400 epochs: 0.0396234\n",
      "loss avg after 5500 epochs: 0.0498376\n",
      "loss avg after 5600 epochs: 0.262294\n",
      "loss avg after 5700 epochs: 0.0478006\n",
      "loss avg after 5800 epochs: 0.0467791\n",
      "loss avg after 5900 epochs: 0.039324\n",
      "loss avg after 6000 epochs: 0.0655487\n",
      "loss avg after 6100 epochs: 0.0376201\n",
      "loss avg after 6200 epochs: 0.0235395\n",
      "loss avg after 6300 epochs: 0.0283559\n",
      "loss avg after 6400 epochs: 0.0226084\n",
      "loss avg after 6500 epochs: 0.0210969\n",
      "loss avg after 6600 epochs: 0.0187365\n",
      "loss avg after 6700 epochs: 0.031763\n",
      "loss avg after 6800 epochs: 0.0271502\n",
      "loss avg after 6900 epochs: 0.0123821\n",
      "loss avg after 7000 epochs: 0.0122766\n",
      "loss avg after 7100 epochs: 0.0156882\n",
      "loss avg after 7200 epochs: 0.0122973\n",
      "loss avg after 7300 epochs: 0.0105996\n",
      "loss avg after 7400 epochs: 0.0116457\n",
      "loss avg after 7500 epochs: 0.0140666\n",
      "loss avg after 7600 epochs: 0.00886211\n",
      "loss avg after 7700 epochs: 0.103019\n",
      "loss avg after 7800 epochs: 0.0202482\n",
      "loss avg after 7900 epochs: 0.0434321\n",
      "loss avg after 8000 epochs: 0.0499878\n",
      "loss avg after 8100 epochs: 0.0256972\n",
      "loss avg after 8200 epochs: 0.0163156\n",
      "loss avg after 8300 epochs: 0.0120343\n",
      "loss avg after 8400 epochs: 0.0126455\n",
      "loss avg after 8500 epochs: 0.0087847\n",
      "loss avg after 8600 epochs: 0.00958538\n",
      "loss avg after 8700 epochs: 0.00709604\n",
      "loss avg after 8800 epochs: 0.00741386\n",
      "loss avg after 8900 epochs: 0.00583881\n",
      "loss avg after 9000 epochs: 0.00646968\n",
      "loss avg after 9100 epochs: 0.318924\n",
      "loss avg after 9200 epochs: 0.103639\n",
      "loss avg after 9300 epochs: 0.0488021\n",
      "loss avg after 9400 epochs: 0.0166474\n",
      "loss avg after 9500 epochs: 0.0152369\n",
      "loss avg after 9600 epochs: 0.00822112\n",
      "loss avg after 9700 epochs: 0.00886673\n",
      "loss avg after 9800 epochs: 0.00615617\n",
      "loss avg after 9900 epochs: 0.0056508\n",
      "loss avg after 10000 epochs: 0.00479412\n",
      "loss avg after 10100 epochs: 0.00350782\n",
      "loss avg after 10200 epochs: 0.00396492\n",
      "loss avg after 10300 epochs: 0.00416782\n",
      "loss avg after 10400 epochs: 0.00359478\n",
      "loss avg after 10500 epochs: 0.00287283\n",
      "loss avg after 10600 epochs: 0.00287358\n",
      "loss avg after 10700 epochs: 0.00233028\n",
      "loss avg after 10800 epochs: 0.00300293\n",
      "loss avg after 10900 epochs: 0.0022986\n",
      "loss avg after 11000 epochs: 0.00224836\n",
      "loss avg after 11100 epochs: 0.00162066\n",
      "loss avg after 11200 epochs: 0.00317364\n",
      "loss avg after 11300 epochs: 0.00180454\n",
      "loss avg after 11400 epochs: 0.00164786\n",
      "loss avg after 11500 epochs: 0.00201696\n",
      "loss avg after 11600 epochs: 0.0015775\n",
      "loss avg after 11700 epochs: 0.00155404\n",
      "loss avg after 11800 epochs: 0.00133247\n",
      "accuracy: 1\n",
      "--- repeat 2 of 5 ---\n",
      "loss avg after 100 epochs: 306.59\n",
      "loss avg after 200 epochs: 194.287\n",
      "loss avg after 300 epochs: 126.829\n",
      "loss avg after 400 epochs: 106.359\n",
      "loss avg after 500 epochs: 73.0527\n",
      "loss avg after 600 epochs: 55.6027\n",
      "loss avg after 700 epochs: 42.536\n",
      "loss avg after 800 epochs: 29.6066\n",
      "loss avg after 900 epochs: 18.7561\n",
      "loss avg after 1000 epochs: 11.7081\n",
      "loss avg after 1100 epochs: 8.59693\n",
      "loss avg after 1200 epochs: 6.31473\n",
      "loss avg after 1300 epochs: 3.71565\n",
      "loss avg after 1400 epochs: 3.66523\n",
      "loss avg after 1500 epochs: 2.47002\n",
      "loss avg after 1600 epochs: 1.88251\n",
      "loss avg after 1700 epochs: 1.57493\n",
      "loss avg after 1800 epochs: 1.28861\n",
      "loss avg after 1900 epochs: 1.13581\n",
      "loss avg after 2000 epochs: 0.971084\n",
      "loss avg after 2100 epochs: 1.0272\n",
      "loss avg after 2200 epochs: 0.605453\n",
      "loss avg after 2300 epochs: 0.594657\n",
      "loss avg after 2400 epochs: 0.499224\n",
      "loss avg after 2500 epochs: 0.702599\n",
      "loss avg after 2600 epochs: 0.439641\n",
      "loss avg after 2700 epochs: 0.400165\n",
      "loss avg after 2800 epochs: 0.3214\n",
      "loss avg after 2900 epochs: 0.275167\n",
      "loss avg after 3000 epochs: 0.334623\n",
      "loss avg after 3100 epochs: 0.301521\n",
      "loss avg after 3200 epochs: 0.244919\n",
      "loss avg after 3300 epochs: 0.154518\n",
      "loss avg after 3400 epochs: 0.175046\n",
      "loss avg after 3500 epochs: 0.138121\n",
      "loss avg after 3600 epochs: 1.82087\n",
      "loss avg after 3700 epochs: 0.420962\n",
      "loss avg after 3800 epochs: 0.268338\n",
      "loss avg after 3900 epochs: 0.292654\n",
      "loss avg after 4000 epochs: 0.1997\n",
      "loss avg after 4100 epochs: 0.146497\n",
      "loss avg after 4200 epochs: 0.171923\n",
      "loss avg after 4300 epochs: 0.159382\n",
      "loss avg after 4400 epochs: 0.137519\n",
      "loss avg after 4500 epochs: 0.10121\n",
      "loss avg after 4600 epochs: 0.109573\n",
      "loss avg after 4700 epochs: 0.0914864\n",
      "loss avg after 4800 epochs: 0.0781399\n",
      "loss avg after 4900 epochs: 0.113595\n",
      "loss avg after 5000 epochs: 0.0591917\n",
      "loss avg after 5100 epochs: 0.0612883\n",
      "loss avg after 5200 epochs: 0.0544855\n",
      "loss avg after 5300 epochs: 0.0537674\n",
      "loss avg after 5400 epochs: 0.0493129\n",
      "loss avg after 5500 epochs: 0.0491209\n",
      "loss avg after 5600 epochs: 0.0522934\n",
      "loss avg after 5700 epochs: 0.0368549\n",
      "loss avg after 5800 epochs: 0.0530985\n",
      "loss avg after 5900 epochs: 0.0291767\n",
      "loss avg after 6000 epochs: 0.0270458\n",
      "loss avg after 6100 epochs: 0.0195577\n",
      "loss avg after 6200 epochs: 0.0341961\n",
      "loss avg after 6300 epochs: 0.0216746\n",
      "loss avg after 6400 epochs: 0.0580692\n",
      "loss avg after 6500 epochs: 0.0434982\n",
      "loss avg after 6600 epochs: 0.0284904\n",
      "loss avg after 6700 epochs: 0.0281213\n",
      "loss avg after 6800 epochs: 0.0263801\n",
      "loss avg after 6900 epochs: 0.0244184\n",
      "loss avg after 7000 epochs: 0.020464\n",
      "loss avg after 7100 epochs: 0.0151405\n",
      "loss avg after 7200 epochs: 0.0170975\n",
      "loss avg after 7300 epochs: 0.0215934\n",
      "loss avg after 7400 epochs: 0.0118934\n",
      "loss avg after 7500 epochs: 0.0124208\n",
      "loss avg after 7600 epochs: 0.0108852\n",
      "loss avg after 7700 epochs: 0.0138137\n",
      "loss avg after 7800 epochs: 0.0138418\n",
      "loss avg after 7900 epochs: 0.00983968\n",
      "loss avg after 8000 epochs: 0.0117861\n",
      "loss avg after 8100 epochs: 0.0114887\n",
      "loss avg after 8200 epochs: 0.00811474\n",
      "loss avg after 8300 epochs: 0.00687795\n",
      "loss avg after 8400 epochs: 0.00705652\n",
      "loss avg after 8500 epochs: 0.00549054\n",
      "loss avg after 8600 epochs: 0.00756613\n",
      "loss avg after 8700 epochs: 0.00476697\n",
      "loss avg after 8800 epochs: 0.00724469\n",
      "loss avg after 8900 epochs: 0.00728897\n",
      "loss avg after 9000 epochs: 0.00413122\n",
      "loss avg after 9100 epochs: 0.00492206\n",
      "loss avg after 9200 epochs: 0.00580947\n",
      "loss avg after 9300 epochs: 0.00460092\n",
      "loss avg after 9400 epochs: 0.00304594\n",
      "loss avg after 9500 epochs: 0.0044775\n",
      "loss avg after 9600 epochs: 0.00368678\n",
      "loss avg after 9700 epochs: 0.00461678\n",
      "loss avg after 9800 epochs: 0.00368918\n",
      "loss avg after 9900 epochs: 0.00427637\n",
      "loss avg after 10000 epochs: 0.00641755\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss avg after 10100 epochs: 0.00304314\n",
      "loss avg after 10200 epochs: 0.00254328\n",
      "loss avg after 10300 epochs: 0.00319448\n",
      "loss avg after 10400 epochs: 0.00743212\n",
      "loss avg after 10500 epochs: 0.89663\n",
      "loss avg after 10600 epochs: 0.116096\n",
      "loss avg after 10700 epochs: 0.0458062\n",
      "loss avg after 10800 epochs: 0.0319004\n",
      "loss avg after 10900 epochs: 0.0272386\n",
      "loss avg after 11000 epochs: 0.0202005\n",
      "loss avg after 11100 epochs: 0.015735\n",
      "loss avg after 11200 epochs: 0.0137749\n",
      "loss avg after 11300 epochs: 0.0103377\n",
      "loss avg after 11400 epochs: 0.0133205\n",
      "loss avg after 11500 epochs: 0.0113656\n",
      "loss avg after 11600 epochs: 0.00813533\n",
      "loss avg after 11700 epochs: 0.00703814\n",
      "loss avg after 11800 epochs: 0.00687856\n",
      "loss avg after 11900 epochs: 0.00581565\n",
      "loss avg after 12000 epochs: 0.00722786\n",
      "loss avg after 12100 epochs: 0.00586422\n",
      "loss avg after 12200 epochs: 0.00578525\n",
      "loss avg after 12300 epochs: 0.00514659\n",
      "loss avg after 12400 epochs: 0.00450083\n",
      "loss avg after 12500 epochs: 0.00362492\n",
      "loss avg after 12600 epochs: 0.00381286\n",
      "loss avg after 12700 epochs: 0.00327498\n",
      "loss avg after 12800 epochs: 0.0032137\n",
      "loss avg after 12900 epochs: 0.00301647\n",
      "loss avg after 13000 epochs: 0.00298473\n",
      "loss avg after 13100 epochs: 0.00244536\n",
      "loss avg after 13200 epochs: 0.00308008\n",
      "loss avg after 13300 epochs: 0.00271133\n",
      "loss avg after 13400 epochs: 0.00187347\n",
      "loss avg after 13500 epochs: 0.0022709\n",
      "loss avg after 13600 epochs: 0.00200167\n",
      "loss avg after 13700 epochs: 0.00160994\n",
      "loss avg after 13800 epochs: 0.00115943\n",
      "loss avg after 13900 epochs: 0.00170608\n",
      "loss avg after 14000 epochs: 0.00190207\n",
      "loss avg after 14100 epochs: 0.00118264\n",
      "loss avg after 14200 epochs: 0.00131873\n",
      "accuracy: 1\n",
      "--- repeat 3 of 5 ---\n",
      "loss avg after 100 epochs: 307.992\n",
      "loss avg after 200 epochs: 225.896\n",
      "loss avg after 300 epochs: 105.92\n",
      "loss avg after 400 epochs: 82.7527\n",
      "loss avg after 500 epochs: 67.815\n",
      "loss avg after 600 epochs: 43.624\n",
      "loss avg after 700 epochs: 19.4149\n",
      "loss avg after 800 epochs: 13.5625\n",
      "loss avg after 900 epochs: 8.792\n",
      "loss avg after 1000 epochs: 7.47581\n",
      "loss avg after 1100 epochs: 4.14373\n",
      "loss avg after 1200 epochs: 3.34771\n",
      "loss avg after 1300 epochs: 2.0407\n",
      "loss avg after 1400 epochs: 1.76571\n",
      "loss avg after 1500 epochs: 1.59492\n",
      "loss avg after 1600 epochs: 1.33564\n",
      "loss avg after 1700 epochs: 1.05972\n",
      "loss avg after 1800 epochs: 0.797376\n",
      "loss avg after 1900 epochs: 0.755887\n",
      "loss avg after 2000 epochs: 0.432746\n",
      "loss avg after 2100 epochs: 0.519732\n",
      "loss avg after 2200 epochs: 10.1849\n",
      "loss avg after 2300 epochs: 0.562655\n",
      "loss avg after 2400 epochs: 0.583056\n",
      "loss avg after 2500 epochs: 0.355322\n",
      "loss avg after 2600 epochs: 0.295993\n",
      "loss avg after 2700 epochs: 0.280805\n",
      "loss avg after 2800 epochs: 0.240073\n",
      "loss avg after 2900 epochs: 0.215042\n",
      "loss avg after 3000 epochs: 0.176981\n",
      "loss avg after 3100 epochs: 0.167239\n",
      "loss avg after 3200 epochs: 0.134042\n",
      "loss avg after 3300 epochs: 0.141168\n",
      "loss avg after 3400 epochs: 0.0973034\n",
      "loss avg after 3500 epochs: 0.103043\n",
      "loss avg after 3600 epochs: 0.113277\n",
      "loss avg after 3700 epochs: 0.0936941\n",
      "loss avg after 3800 epochs: 0.0909402\n",
      "loss avg after 3900 epochs: 0.0652128\n",
      "loss avg after 4000 epochs: 0.0729462\n",
      "loss avg after 4100 epochs: 0.0696905\n",
      "loss avg after 4200 epochs: 0.0662879\n",
      "loss avg after 4300 epochs: 0.077898\n",
      "loss avg after 4400 epochs: 1.21335\n",
      "loss avg after 4500 epochs: 0.108664\n",
      "loss avg after 4600 epochs: 0.155302\n",
      "loss avg after 4700 epochs: 0.069796\n",
      "loss avg after 4800 epochs: 0.0565884\n",
      "loss avg after 4900 epochs: 0.245284\n",
      "loss avg after 5000 epochs: 0.337938\n",
      "loss avg after 5100 epochs: 0.103555\n",
      "loss avg after 5200 epochs: 0.104329\n",
      "loss avg after 5300 epochs: 0.0803449\n",
      "loss avg after 5400 epochs: 0.0507251\n",
      "loss avg after 5500 epochs: 0.0411673\n",
      "loss avg after 5600 epochs: 0.0298195\n",
      "loss avg after 5700 epochs: 0.029751\n",
      "loss avg after 5800 epochs: 0.0229403\n",
      "loss avg after 5900 epochs: 0.0278198\n",
      "loss avg after 6000 epochs: 0.0279557\n",
      "loss avg after 6100 epochs: 0.0210471\n",
      "loss avg after 6200 epochs: 0.0204519\n",
      "loss avg after 6300 epochs: 0.0294071\n",
      "loss avg after 6400 epochs: 0.0122731\n",
      "loss avg after 6500 epochs: 0.0276985\n",
      "loss avg after 6600 epochs: 0.0179792\n",
      "loss avg after 6700 epochs: 0.0117351\n",
      "loss avg after 6800 epochs: 0.0130694\n",
      "loss avg after 6900 epochs: 0.013003\n",
      "loss avg after 7000 epochs: 0.0486947\n",
      "loss avg after 7100 epochs: 0.0126524\n",
      "loss avg after 7200 epochs: 0.00979478\n",
      "loss avg after 7300 epochs: 0.00970934\n",
      "loss avg after 7400 epochs: 0.0108935\n",
      "loss avg after 7500 epochs: 0.0081542\n",
      "loss avg after 7600 epochs: 0.00615725\n",
      "loss avg after 7700 epochs: 0.00919059\n",
      "loss avg after 7800 epochs: 0.00549737\n",
      "loss avg after 7900 epochs: 0.00894169\n",
      "loss avg after 8000 epochs: 0.00617146\n",
      "loss avg after 8100 epochs: 0.00612411\n",
      "loss avg after 8200 epochs: 0.00558515\n",
      "loss avg after 8300 epochs: 0.00422449\n",
      "loss avg after 8400 epochs: 0.00365082\n",
      "loss avg after 8500 epochs: 0.00405952\n",
      "loss avg after 8600 epochs: 0.00720109\n",
      "loss avg after 8700 epochs: 0.00795822\n",
      "loss avg after 8800 epochs: 10.4098\n",
      "loss avg after 8900 epochs: 0.11055\n",
      "loss avg after 9000 epochs: 0.0372839\n",
      "loss avg after 9100 epochs: 0.0184548\n",
      "loss avg after 9200 epochs: 0.0136122\n",
      "loss avg after 9300 epochs: 0.0133214\n",
      "loss avg after 9400 epochs: 0.0129294\n",
      "loss avg after 9500 epochs: 0.0089885\n",
      "loss avg after 9600 epochs: 0.0134619\n",
      "loss avg after 9700 epochs: 0.00806155\n",
      "loss avg after 9800 epochs: 0.00711267\n",
      "loss avg after 9900 epochs: 0.00521945\n",
      "loss avg after 10000 epochs: 0.00671276\n",
      "loss avg after 10100 epochs: 0.005224\n",
      "loss avg after 10200 epochs: 0.00736652\n",
      "loss avg after 10300 epochs: 0.00530029\n",
      "loss avg after 10400 epochs: 0.00370421\n",
      "loss avg after 10500 epochs: 0.00353219\n",
      "loss avg after 10600 epochs: 0.00294935\n",
      "loss avg after 10700 epochs: 0.0061139\n",
      "loss avg after 10800 epochs: 0.00366204\n",
      "loss avg after 10900 epochs: 0.00255401\n",
      "loss avg after 11000 epochs: 0.00367057\n",
      "loss avg after 11100 epochs: 0.00198446\n",
      "loss avg after 11200 epochs: 0.00235707\n",
      "loss avg after 11300 epochs: 0.00158828\n",
      "loss avg after 11400 epochs: 0.00204561\n",
      "loss avg after 11500 epochs: 0.00172785\n",
      "loss avg after 11600 epochs: 0.00621426\n",
      "loss avg after 11700 epochs: 0.00268978\n",
      "accuracy: 1\n",
      "--- repeat 4 of 5 ---\n",
      "loss avg after 100 epochs: 343.381\n",
      "loss avg after 200 epochs: 215.642\n",
      "loss avg after 300 epochs: 98.5257\n",
      "loss avg after 400 epochs: 87.1797\n",
      "loss avg after 500 epochs: 78.1456\n",
      "loss avg after 600 epochs: 47\n",
      "loss avg after 700 epochs: 29.214\n",
      "loss avg after 800 epochs: 24.0121\n",
      "loss avg after 900 epochs: 11.9705\n",
      "loss avg after 1000 epochs: 7.71972\n",
      "loss avg after 1100 epochs: 5.92365\n",
      "loss avg after 1200 epochs: 4.26347\n",
      "loss avg after 1300 epochs: 3.28216\n",
      "loss avg after 1400 epochs: 2.07688\n",
      "loss avg after 1500 epochs: 1.55991\n",
      "loss avg after 1600 epochs: 1.36319\n",
      "loss avg after 1700 epochs: 1.21813\n",
      "loss avg after 1800 epochs: 1.07986\n",
      "loss avg after 1900 epochs: 0.964751\n",
      "loss avg after 2000 epochs: 0.81094\n",
      "loss avg after 2100 epochs: 0.599328\n",
      "loss avg after 2200 epochs: 0.677064\n",
      "loss avg after 2300 epochs: 0.679379\n",
      "loss avg after 2400 epochs: 0.494059\n",
      "loss avg after 2500 epochs: 0.510257\n",
      "loss avg after 2600 epochs: 0.528131\n",
      "loss avg after 2700 epochs: 0.512015\n",
      "loss avg after 2800 epochs: 1.71588\n",
      "loss avg after 2900 epochs: 0.499454\n",
      "loss avg after 3000 epochs: 0.286587\n",
      "loss avg after 3100 epochs: 0.287958\n",
      "loss avg after 3200 epochs: 0.22159\n",
      "loss avg after 3300 epochs: 0.172387\n",
      "loss avg after 3400 epochs: 0.379704\n",
      "loss avg after 3500 epochs: 0.201392\n",
      "loss avg after 3600 epochs: 0.218006\n",
      "loss avg after 3700 epochs: 0.175664\n",
      "loss avg after 3800 epochs: 0.115785\n",
      "loss avg after 3900 epochs: 0.123429\n",
      "loss avg after 4000 epochs: 7.18882\n",
      "loss avg after 4100 epochs: 1.17971\n",
      "loss avg after 4200 epochs: 0.445837\n",
      "loss avg after 4300 epochs: 0.329569\n",
      "loss avg after 4400 epochs: 0.221079\n",
      "loss avg after 4500 epochs: 0.198422\n",
      "loss avg after 4600 epochs: 0.196312\n",
      "loss avg after 4700 epochs: 0.146126\n",
      "loss avg after 4800 epochs: 0.122647\n",
      "loss avg after 4900 epochs: 0.104894\n",
      "loss avg after 5000 epochs: 0.0979064\n",
      "loss avg after 5100 epochs: 0.0994694\n",
      "loss avg after 5200 epochs: 0.335125\n",
      "loss avg after 5300 epochs: 0.0727841\n",
      "loss avg after 5400 epochs: 0.0579522\n",
      "loss avg after 5500 epochs: 0.0547573\n",
      "loss avg after 5600 epochs: 0.0457408\n",
      "loss avg after 5700 epochs: 0.0553254\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss avg after 5800 epochs: 0.0518236\n",
      "loss avg after 5900 epochs: 0.038906\n",
      "loss avg after 6000 epochs: 0.0489734\n",
      "loss avg after 6100 epochs: 0.029511\n",
      "loss avg after 6200 epochs: 0.0356287\n",
      "loss avg after 6300 epochs: 0.0272206\n",
      "loss avg after 6400 epochs: 0.0232502\n",
      "loss avg after 6500 epochs: 0.0462862\n",
      "loss avg after 6600 epochs: 0.0268557\n",
      "loss avg after 6700 epochs: 0.0172243\n",
      "loss avg after 6800 epochs: 0.0173206\n",
      "loss avg after 6900 epochs: 0.020154\n",
      "loss avg after 7000 epochs: 0.0166904\n",
      "loss avg after 7100 epochs: 0.0135547\n",
      "loss avg after 7200 epochs: 0.0131848\n",
      "loss avg after 7300 epochs: 0.0159419\n",
      "loss avg after 7400 epochs: 0.0146038\n",
      "loss avg after 7500 epochs: 0.0146657\n",
      "loss avg after 7600 epochs: 0.0108543\n",
      "loss avg after 7700 epochs: 0.0107235\n",
      "loss avg after 7800 epochs: 0.0110762\n",
      "loss avg after 7900 epochs: 0.00946748\n",
      "loss avg after 8000 epochs: 0.0127652\n",
      "loss avg after 8100 epochs: 0.00586374\n",
      "loss avg after 8200 epochs: 0.0122813\n",
      "loss avg after 8300 epochs: 0.0057035\n",
      "loss avg after 8400 epochs: 0.00484085\n",
      "loss avg after 8500 epochs: 0.00634808\n",
      "loss avg after 8600 epochs: 0.00587581\n",
      "loss avg after 8700 epochs: 0.00624232\n",
      "loss avg after 8800 epochs: 0.00515293\n",
      "loss avg after 8900 epochs: 0.00514352\n",
      "loss avg after 9000 epochs: 0.0051046\n",
      "loss avg after 9100 epochs: 0.00534406\n",
      "loss avg after 9200 epochs: 0.00384974\n",
      "loss avg after 9300 epochs: 0.00428682\n",
      "loss avg after 9400 epochs: 0.00381719\n",
      "loss avg after 9500 epochs: 0.00443952\n",
      "loss avg after 9600 epochs: 10.593\n",
      "loss avg after 9700 epochs: 0.0809308\n",
      "loss avg after 9800 epochs: 0.045218\n",
      "loss avg after 9900 epochs: 0.0217758\n",
      "loss avg after 10000 epochs: 0.0297442\n",
      "loss avg after 10100 epochs: 0.0175848\n",
      "loss avg after 10200 epochs: 0.0134565\n",
      "loss avg after 10300 epochs: 0.0125137\n",
      "loss avg after 10400 epochs: 0.0194393\n",
      "loss avg after 10500 epochs: 0.0113458\n",
      "loss avg after 10600 epochs: 0.00903355\n",
      "loss avg after 10700 epochs: 0.00928398\n",
      "loss avg after 10800 epochs: 0.0116305\n",
      "loss avg after 10900 epochs: 0.005567\n",
      "loss avg after 11000 epochs: 0.00671104\n",
      "loss avg after 11100 epochs: 0.00600108\n",
      "loss avg after 11200 epochs: 0.00591422\n",
      "loss avg after 11300 epochs: 0.0869175\n",
      "loss avg after 11400 epochs: 0.015096\n",
      "loss avg after 11500 epochs: 0.0071896\n",
      "loss avg after 11600 epochs: 0.00914093\n",
      "loss avg after 11700 epochs: 0.0056416\n",
      "loss avg after 11800 epochs: 0.00372027\n",
      "loss avg after 11900 epochs: 0.00459193\n",
      "loss avg after 12000 epochs: 0.00405963\n",
      "loss avg after 12100 epochs: 0.00382451\n",
      "loss avg after 12200 epochs: 0.00265431\n",
      "loss avg after 12300 epochs: 0.00217775\n",
      "loss avg after 12400 epochs: 0.0028116\n",
      "loss avg after 12500 epochs: 0.00281545\n",
      "loss avg after 12600 epochs: 0.00196123\n",
      "loss avg after 12700 epochs: 0.00228443\n",
      "loss avg after 12800 epochs: 0.00178231\n",
      "loss avg after 12900 epochs: 0.00205352\n",
      "loss avg after 13000 epochs: 0.00256903\n",
      "loss avg after 13100 epochs: 0.00198655\n",
      "loss avg after 13200 epochs: 0.00166064\n",
      "loss avg after 13300 epochs: 0.00308632\n",
      "accuracy: 1\n",
      "--- repeat 5 of 5 ---\n",
      "loss avg after 100 epochs: 402.918\n",
      "loss avg after 200 epochs: 242.54\n",
      "loss avg after 300 epochs: 121.516\n",
      "loss avg after 400 epochs: 72.7671\n",
      "loss avg after 500 epochs: 49.5541\n",
      "loss avg after 600 epochs: 37.6774\n",
      "loss avg after 700 epochs: 22.9249\n",
      "loss avg after 800 epochs: 15.3667\n",
      "loss avg after 900 epochs: 8.16013\n",
      "loss avg after 1000 epochs: 5.48884\n",
      "loss avg after 1100 epochs: 4.57461\n",
      "loss avg after 1200 epochs: 3.48951\n",
      "loss avg after 1300 epochs: 2.9813\n",
      "loss avg after 1400 epochs: 2.06221\n",
      "loss avg after 1500 epochs: 2.05243\n",
      "loss avg after 1600 epochs: 1.55188\n",
      "loss avg after 1700 epochs: 1.26721\n",
      "loss avg after 1800 epochs: 3.14951\n",
      "loss avg after 1900 epochs: 4.45136\n",
      "loss avg after 2000 epochs: 2.03166\n",
      "loss avg after 2100 epochs: 1.43903\n",
      "loss avg after 2200 epochs: 1.00415\n",
      "loss avg after 2300 epochs: 0.654694\n",
      "loss avg after 2400 epochs: 0.869415\n",
      "loss avg after 2500 epochs: 0.522935\n",
      "loss avg after 2600 epochs: 0.539285\n",
      "loss avg after 2700 epochs: 0.422361\n",
      "loss avg after 2800 epochs: 0.525909\n",
      "loss avg after 2900 epochs: 0.34074\n",
      "loss avg after 3000 epochs: 0.282982\n",
      "loss avg after 3100 epochs: 0.317226\n",
      "loss avg after 3200 epochs: 0.253619\n",
      "loss avg after 3300 epochs: 0.37269\n",
      "loss avg after 3400 epochs: 0.440419\n",
      "loss avg after 3500 epochs: 0.247335\n",
      "loss avg after 3600 epochs: 0.209673\n",
      "loss avg after 3700 epochs: 0.19353\n",
      "loss avg after 3800 epochs: 0.179174\n",
      "loss avg after 3900 epochs: 0.172397\n",
      "loss avg after 4000 epochs: 0.138774\n",
      "loss avg after 4100 epochs: 0.185067\n",
      "loss avg after 4200 epochs: 0.113333\n",
      "loss avg after 4300 epochs: 0.0995501\n",
      "loss avg after 4400 epochs: 0.0836032\n",
      "loss avg after 4500 epochs: 0.0728257\n",
      "loss avg after 4600 epochs: 0.0858081\n",
      "loss avg after 4700 epochs: 0.0808157\n",
      "loss avg after 4800 epochs: 0.0697517\n",
      "loss avg after 4900 epochs: 0.0771096\n",
      "loss avg after 5000 epochs: 0.0582388\n",
      "loss avg after 5100 epochs: 0.0571821\n",
      "loss avg after 5200 epochs: 0.0743085\n",
      "loss avg after 5300 epochs: 0.0531444\n",
      "loss avg after 5400 epochs: 0.0470299\n",
      "loss avg after 5500 epochs: 0.0748859\n",
      "loss avg after 5600 epochs: 0.0479522\n",
      "loss avg after 5700 epochs: 0.0384735\n",
      "loss avg after 5800 epochs: 0.0578657\n",
      "loss avg after 5900 epochs: 0.0448238\n",
      "loss avg after 6000 epochs: 0.0430187\n",
      "loss avg after 6100 epochs: 0.0389335\n",
      "loss avg after 6200 epochs: 1.78089\n",
      "loss avg after 6300 epochs: 0.354905\n",
      "loss avg after 6400 epochs: 0.17396\n",
      "loss avg after 6500 epochs: 0.176111\n",
      "loss avg after 6600 epochs: 0.125432\n",
      "loss avg after 6700 epochs: 0.0879134\n",
      "loss avg after 6800 epochs: 0.0912296\n",
      "loss avg after 6900 epochs: 0.070152\n",
      "loss avg after 7000 epochs: 0.0404161\n",
      "loss avg after 7100 epochs: 0.045737\n",
      "loss avg after 7200 epochs: 0.0492855\n",
      "loss avg after 7300 epochs: 0.0421553\n",
      "loss avg after 7400 epochs: 0.0354056\n",
      "loss avg after 7500 epochs: 0.0317329\n",
      "loss avg after 7600 epochs: 0.0326086\n",
      "loss avg after 7700 epochs: 0.0269378\n",
      "loss avg after 7800 epochs: 0.023479\n",
      "loss avg after 7900 epochs: 0.0212517\n",
      "loss avg after 8000 epochs: 0.0223801\n",
      "loss avg after 8100 epochs: 0.0182418\n",
      "loss avg after 8200 epochs: 0.0150364\n",
      "loss avg after 8300 epochs: 0.0198367\n",
      "loss avg after 8400 epochs: 0.0133095\n",
      "loss avg after 8500 epochs: 0.0147044\n",
      "loss avg after 8600 epochs: 0.0124436\n",
      "loss avg after 8700 epochs: 0.0113854\n",
      "loss avg after 8800 epochs: 0.0140554\n",
      "loss avg after 8900 epochs: 0.0105641\n",
      "loss avg after 9000 epochs: 0.0125666\n",
      "loss avg after 9100 epochs: 0.00869003\n",
      "loss avg after 9200 epochs: 0.0107848\n",
      "loss avg after 9300 epochs: 0.0118817\n",
      "loss avg after 9400 epochs: 0.00848228\n",
      "loss avg after 9500 epochs: 0.00603964\n",
      "loss avg after 9600 epochs: 0.00538563\n",
      "loss avg after 9700 epochs: 0.00631068\n",
      "loss avg after 9800 epochs: 0.00598467\n",
      "loss avg after 9900 epochs: 0.00446347\n",
      "loss avg after 10000 epochs: 0.00436668\n",
      "loss avg after 10100 epochs: 0.0038753\n",
      "loss avg after 10200 epochs: 0.00333858\n",
      "loss avg after 10300 epochs: 0.00278764\n",
      "loss avg after 10400 epochs: 0.00310426\n",
      "loss avg after 10500 epochs: 0.0163032\n",
      "loss avg after 10600 epochs: 0.00800157\n",
      "loss avg after 10700 epochs: 0.00795342\n",
      "loss avg after 10800 epochs: 0.00464918\n",
      "loss avg after 10900 epochs: 0.00607866\n",
      "loss avg after 11000 epochs: 0.00384888\n",
      "loss avg after 11100 epochs: 0.00300451\n",
      "loss avg after 11200 epochs: 0.00473065\n",
      "loss avg after 11300 epochs: 0.00253779\n",
      "loss avg after 11400 epochs: 0.00250083\n",
      "loss avg after 11500 epochs: 0.00238383\n",
      "loss avg after 11600 epochs: 0.00336375\n",
      "loss avg after 11700 epochs: 0.00226925\n",
      "loss avg after 11800 epochs: 0.00232467\n",
      "loss avg after 11900 epochs: 0.00228975\n",
      "loss avg after 12000 epochs: 0.00255327\n",
      "loss avg after 12100 epochs: 0.00230531\n",
      "loss avg after 12200 epochs: 0.00189048\n",
      "loss avg after 12300 epochs: 0.00182686\n",
      "loss avg after 12400 epochs: 0.00138675\n",
      "loss avg after 12500 epochs: 0.00895981\n",
      "loss avg after 12600 epochs: 0.0016466\n",
      "accuracy: 1\n"
     ]
    }
   ],
   "source": [
    "# learn\n",
    "import edist.tree_utils as tu\n",
    "import peano_addition\n",
    "\n",
    "for r in range(R):\n",
    "    print('--- repeat %d of %d ---' % (r+1, R))\n",
    "    # instantiate network and optimizer\n",
    "    net = ten.TEN(num_layers = num_layers, alphabet = peano_addition.alphabet,\n",
    "                  dim_hid = dim_hid, skip_connections = skip_connections, nonlin = nonlin,\n",
    "                  dim_in_extra = max_degree + 1)\n",
    "    optimizer = torch.optim.Adam(net.parameters(), lr=learning_rate, weight_decay=weight_decay)\n",
    "    # start training\n",
    "    loss_avg = None\n",
    "    learning_curve = []\n",
    "    epochs = 0\n",
    "    while epochs < max_epochs:\n",
    "        optimizer.zero_grad()\n",
    "        # sample a nontrivial time series\n",
    "        time_series = peano_addition.generate_time_series()\n",
    "        if len(time_series) < 2:\n",
    "            continue\n",
    "        # compute the prediction loss\n",
    "        loss = peano_addition.compute_loss(net, time_series)\n",
    "        # compute the gradient\n",
    "        loss.backward()\n",
    "        # perform an optimizer step\n",
    "        optimizer.step()\n",
    "        # compute a new moving average over the loss\n",
    "        if loss_avg is None:\n",
    "            loss_avg = loss.item()\n",
    "        else:\n",
    "            loss_avg = loss_avg * 0.9 + 0.1 * loss.item()\n",
    "        learning_curve.append(loss.item())\n",
    "        if((epochs+1) % 100 == 0):\n",
    "            print('loss avg after %d epochs: %g' % (epochs+1, loss_avg))\n",
    "        epochs += 1\n",
    "        if loss_avg < loss_threshold:\n",
    "            break\n",
    "    learning_curves.append(learning_curve)\n",
    "    # after training is completed, evaluate\n",
    "    j = 0\n",
    "    T = 0\n",
    "    while j < N_test:\n",
    "        # sample a random time series\n",
    "        time_series = peano_addition.generate_time_series()\n",
    "        if len(time_series) < 2:\n",
    "            continue\n",
    "        # iterate over the time series\n",
    "        for t in range(len(time_series)-1):\n",
    "            # perform the prediction\n",
    "            nodes, adj = time_series[t]\n",
    "            try:\n",
    "                _, nodes_actual, adj_actual = peano_addition.predict_step(net, nodes, adj)\n",
    "                nodes_expected, adj_expected = time_series[t+1]\n",
    "                if nodes_actual == nodes_expected and adj_actual == adj_expected:\n",
    "                    accs[r] += 1\n",
    "                else:\n",
    "                    print('expected tree %s but was actually %s' % (tu.tree_to_string(nodes_expected, adj_expected), tu.tree_to_string(nodes_actual, adj_actual)))\n",
    "            except Exception as ex:\n",
    "                try:\n",
    "                    peano_addition.predict_step(net, nodes, adj, verbose = True)\n",
    "                except Exception as ex2:\n",
    "                    pass\n",
    "                print('Exception for input tree %s and network output %s\\nexception was %s' % (tu.tree_to_string(nodes, adj, indent = True, with_indices = True), deltaX, str(ex)))\n",
    "        T += len(time_series)-1\n",
    "        j += 1\n",
    "    accs[r] /= T\n",
    "    print('accuracy: %g' % accs[r])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: 1 +- 0\n",
      "Epochs: 12768.4 +- 933.092\n"
     ]
    }
   ],
   "source": [
    "# print results\n",
    "print('Accuracy: %g +- %g' % (np.mean(accs), np.std(accs)))\n",
    "num_epochs = np.array(list(map(len, learning_curves)))\n",
    "print('Epochs: %g +- %g' % (np.mean(num_epochs), np.std(num_epochs)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "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": [
    "# visualize learning curves\n",
    "import matplotlib.pyplot as plt\n",
    "smoothing_steps = 10\n",
    "for r in range(R):\n",
    "    # compute a moving average before visualization\n",
    "    acum = np.cumsum(learning_curves[r])\n",
    "    plt.semilogy((acum[smoothing_steps:] - acum[:-smoothing_steps])/smoothing_steps)\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
}
