{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Table of Contents\n",
    "\n",
    "<div id=\"toc\"></div>"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/javascript": [
       "$.getScript('https://kmahelona.github.io/ipython_notebook_goodies/ipython_notebook_toc.js')\n"
      ],
      "text/plain": [
       "<IPython.core.display.Javascript object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "%%javascript\n",
    "$.getScript('https://kmahelona.github.io/ipython_notebook_goodies/ipython_notebook_toc.js')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Data import and parsing"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import gtar\n",
    "import numpy as np\n",
    "import matplotlib, matplotlib.pyplot as pp\n",
    "import os\n",
    "# This line will disable GPU acceleration if desired\n",
    "# os.environ['CUDA_VISIBLE_DEVICES'] = '-1'\n",
    "import json\n",
    "import collections\n",
    "import functools\n",
    "import itertools\n",
    "import ipywidgets\n",
    "\n",
    "import sqlite3\n",
    "import numpy as np\n",
    "import io\n",
    "\n",
    "import flowws\n",
    "import keras_gtar\n",
    "\n",
    "def adapt_array(arr):\n",
    "    \"\"\"\n",
    "    http://stackoverflow.com/a/31312102/190597 (SoulNibbler)\n",
    "    \"\"\"\n",
    "    out = io.BytesIO()\n",
    "    np.save(out, arr)\n",
    "    out.seek(0)\n",
    "    return sqlite3.Binary(out.read())\n",
    "\n",
    "def convert_array(text):\n",
    "    out = io.BytesIO(text)\n",
    "    out.seek(0)\n",
    "    return np.load(out)\n",
    "\n",
    "# Converts np.array to TEXT when inserting\n",
    "sqlite3.register_adapter(np.ndarray, adapt_array)\n",
    "\n",
    "# Converts TEXT to np.array when selecting\n",
    "sqlite3.register_converter(\"ndarray\", convert_array)\n",
    "\n",
    "from tensorflow.keras.backend import clear_session\n",
    "import gc\n",
    "import multiprocessing\n",
    "from tqdm.notebook import tqdm\n",
    "\n",
    "class DataSeries:\n",
    "    def __init__(self):\n",
    "        self.data = collections.defaultdict(list)\n",
    "\n",
    "    def add(self, x, y):\n",
    "        self.data[x].append(y)\n",
    "\n",
    "    @property\n",
    "    def x(self):\n",
    "        return np.array(list(sorted(self.data)))\n",
    "\n",
    "    @property\n",
    "    def mean(self):\n",
    "        return np.array([np.mean(self.data[x], axis=0) for x in self.x])\n",
    "\n",
    "    @property\n",
    "    def std(self):\n",
    "        return np.array([np.std(self.data[x], axis=0) for x in self.x])\n",
    "\n",
    "    @property\n",
    "    def stderr(self):\n",
    "        return np.array([np.std(self.data[x], axis=0)/np.sqrt(len(self.data[x])) for x in self.x])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(145,)\n"
     ]
    }
   ],
   "source": [
    "dbases = []\n",
    "\n",
    "dbases.append('../data')\n",
    "# dbases.append('../data/temporary')\n",
    "\n",
    "conn = sqlite3.connect(\":memory:\", detect_types=sqlite3.PARSE_DECLTYPES)\n",
    "conn.execute('create table data '\n",
    "             '(fname, workflow_description, dataset, model_size, '\n",
    "             'depth, dropout, score_norm, value_norm, '\n",
    "             'distance_norm, residual, nonlin, optimizer)')\n",
    "conn.execute('create table md17 (data_id, molecules)')\n",
    "conn.execute('create table varyings (data_id, name, arr_value ndarray)')\n",
    "\n",
    "DATA_NAMES = ['MD17', 'PDBCoarseGrained', 'PyriodicDataset', 'RMD17']\n",
    "DATA_NAME_PREFIXES = []\n",
    "\n",
    "skipped_list = []\n",
    "\n",
    "with multiprocessing.Pool() as p:\n",
    "    for dbase in dbases:\n",
    "        for (dirpath, dirnames, fnames) in os.walk(dbase):\n",
    "            if 'SKIP_THIS_DIR' in fnames:\n",
    "                continue\n",
    "            for fname in sorted(fnames):\n",
    "                fname = os.path.join(dirpath, fname)\n",
    "                try:\n",
    "                    with gtar.GTAR(fname, 'r') as traj:\n",
    "                        dataset = None\n",
    "                        depth = None\n",
    "                        model_size = None\n",
    "                        dropout = None\n",
    "                        optimizer = None\n",
    "                        score_norm = None\n",
    "                        value_norm = None\n",
    "                        distance_norm = None\n",
    "                        residual = None\n",
    "                        nonlin = None\n",
    "                        molecules = None\n",
    "\n",
    "                        skip = False\n",
    "\n",
    "                        m = traj.readStr('workflow.json')\n",
    "\n",
    "                        if m is None:\n",
    "                            continue\n",
    "                        m = json.loads(m)\n",
    "                        for stage in m['stages']:\n",
    "                            if (stage['type'] in DATA_NAMES or any(\n",
    "                                stage['type'].startswith(prefix) for prefix in DATA_NAME_PREFIXES)):\n",
    "                                dataset = stage['type']\n",
    "                                molecules = stage['arguments'].get('molecules', None)\n",
    "                            elif stage['type'] == 'MoleculeForceRegression':\n",
    "                                depth = stage['arguments']['n_blocks']\n",
    "                                model_size = stage['arguments']['n_dim']\n",
    "                                dropout = stage['arguments']['dropout']\n",
    "                                score_norm = ','.join(stage['arguments']['score_normalization'])\n",
    "                                value_norm = ','.join(stage['arguments']['value_normalization'])\n",
    "                                distance_norm = stage['arguments'].get('normalize_distances', '')\n",
    "                                residual = stage['arguments']['residual']\n",
    "                                nonlin = stage['arguments']['block_nonlinearity']\n",
    "                            elif stage['type'].startswith('PDBInverseCoarseGrain'):\n",
    "                                depth = stage['arguments']['n_blocks_coarse'] + stage['arguments']['n_blocks_fine']\n",
    "                                model_size = stage['arguments']['n_dim']\n",
    "                                residual = stage['arguments']['residual']\n",
    "                                nonlin = stage['arguments']['block_nonlinearity']\n",
    "                            elif stage['type'] == 'CrystalStructureClassification':\n",
    "                                depth = stage['arguments']['n_blocks']\n",
    "                                model_size = stage['arguments']['n_dim']\n",
    "                                dropout = stage['arguments']['dropout']\n",
    "                                residual = stage['arguments']['residual']\n",
    "                                nonlin = stage['arguments']['block_nonlinearity']\n",
    "                            elif stage['type'].endswith('Train'):\n",
    "                                val_split = stage['arguments']['validation_split']\n",
    "                                optimizer = stage['arguments']['optimizer']\n",
    "\n",
    "                        if skip:\n",
    "                            continue\n",
    "\n",
    "                        assert dataset is not None\n",
    "\n",
    "                        vals = [fname, json.dumps(m), dataset,\n",
    "                               model_size, depth, dropout, score_norm, value_norm,\n",
    "                                distance_norm, residual, nonlin, optimizer]\n",
    "                        curs = conn.execute('insert into data values ({})'.format(','.join(len(vals)*'?')), vals)\n",
    "                        this_rowid = curs.lastrowid\n",
    "\n",
    "                        recs = [rec for rec in traj.getRecordTypes() if rec.getBehavior() == gtar.Behavior.Continuous]\n",
    "                        dset = {}\n",
    "                        for rec in recs:\n",
    "                            vals = np.concatenate([traj.getRecord(rec, frame) for frame in traj.queryFrames(rec)])\n",
    "                            conn.execute('insert into varyings values (?, ?, ?)', (this_rowid, rec.getName(), vals))\n",
    "\n",
    "                        if dataset.endswith('MD17'):\n",
    "                            molecule_str = '{}:{}'.format(dataset, ','.join(molecules))\n",
    "                            conn.execute('insert into md17 values (?, ?)', (this_rowid, molecule_str))\n",
    "\n",
    "                except RuntimeError:\n",
    "                    skipped_list.append(fname)\n",
    "                    if len(skipped_list) < 8:\n",
    "                        print('Skipping {}'.format(fname))\n",
    "                    continue\n",
    "\n",
    "print(next(conn.execute('select count(*) from data')))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Error calculation\n",
    "\n",
    "## Structure identification"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 7s 7ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 3s 2ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 3s 2ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 3s 2ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 3s 2ms/step\n",
      "0.9898270600203458 0.0008274547063138542\n"
     ]
    }
   ],
   "source": [
    "errs = []\n",
    "\n",
    "for (fname, desc) in conn.execute(\n",
    "    'select fname, workflow_description from data where dataset = \"PyriodicDataset\"'):\n",
    "\n",
    "    desc = json.loads(desc)\n",
    "    train_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'].endswith('Train')][0]\n",
    "    batch_size = max(1, desc['stages'][train_index]['arguments']['batch_size']//4)\n",
    "    desc['stages'] = desc['stages'][:train_index]\n",
    "\n",
    "    workflow = flowws.Workflow.from_JSON(desc)\n",
    "    scope = workflow.run()\n",
    "\n",
    "    with keras_gtar.Trajectory(fname, 'r') as traj:\n",
    "        model = traj.load()\n",
    "\n",
    "    prediction = model.predict(scope['x_test'], batch_size=batch_size, verbose=1)\n",
    "    prediction = np.argmax(prediction, axis=-1)\n",
    "\n",
    "    acc = np.mean(prediction == scope['y_test'])\n",
    "\n",
    "    errs.append(acc)\n",
    "\n",
    "print(np.mean(errs), np.std(errs)/np.sqrt(len(errs)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## MD17"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 19s 14ms/step\n",
      "../data/force_regression/dump.20210811155449.aspirin.26.sqlite 12.551308802176605\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 16s 15ms/step\n",
      "../data/force_regression/dump.20210815110554.aspirin.27.sqlite 13.429755429296236\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 16s 14ms/step\n",
      "../data/force_regression/dump.20210815110554.aspirin.28.sqlite 13.96924417463346\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 17s 14ms/step\n",
      "../data/force_regression/dump.20210815110554.aspirin.29.sqlite 15.514020282571593\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 16s 14ms/step\n",
      "../data/force_regression/dump.20210818093623.aspirin.30.sqlite 11.950861963240678\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 5ms/step\n",
      "../data/force_regression/dump.20210811155449.benzene.26.sqlite 6.505618764049095\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 5ms/step\n",
      "../data/force_regression/dump.20210815110554.benzene.27.sqlite 6.5044735457689855\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 5ms/step\n",
      "../data/force_regression/dump.20210815110554.benzene.28.sqlite 7.005997081373787\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 5ms/step\n",
      "../data/force_regression/dump.20210815110554.benzene.29.sqlite 6.667958954090425\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 5ms/step\n",
      "../data/force_regression/dump.20210815110554.benzene.30.sqlite 6.709255973573829\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 15s 6ms/step\n",
      "../data/force_regression/dump.20210811155449.ethanol.26.sqlite 4.792121684759935\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 13s 6ms/step\n",
      "../data/force_regression/dump.20210815110554.ethanol.27.sqlite 4.315667726880824\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 6ms/step\n",
      "../data/force_regression/dump.20210815110554.ethanol.28.sqlite 4.624417640310539\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 16s 6ms/step\n",
      "../data/force_regression/dump.20210815162308.ethanol.29.sqlite 4.330082443997123\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 13s 6ms/step\n",
      "../data/force_regression/dump.20210815172011.ethanol.30.sqlite 5.177654053308557\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 14s 6ms/step\n",
      "../data/force_regression/dump.20210811155449.malonaldehyde.26.sqlite 8.729175669404112\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 16s 6ms/step\n",
      "../data/force_regression/dump.20210815212753.malonaldehyde.27.sqlite 7.645525038301115\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 13s 5ms/step\n",
      "../data/force_regression/dump.20210815214538.malonaldehyde.28.sqlite 8.00644170129081\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 6ms/step\n",
      "../data/force_regression/dump.20210815214538.malonaldehyde.29.sqlite 8.349182317292577\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 15s 6ms/step\n",
      "../data/force_regression/dump.20210815232259.malonaldehyde.30.sqlite 8.59793761753682\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 18s 10ms/step\n",
      "../data/force_regression/dump.20210811155449.naphthalene.26.sqlite 4.232955655216333\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 15s 10ms/step\n",
      "../data/force_regression/dump.20210816031606.naphthalene.28.sqlite 3.9672438100651832\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 17s 10ms/step\n",
      "../data/force_regression/dump.20210816070435.naphthalene.27.sqlite 4.194330313436005\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 14s 10ms/step\n",
      "../data/force_regression/dump.20210816080514.naphthalene.29.sqlite 3.637103319648413\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 15s 10ms/step\n",
      "../data/force_regression/dump.20210816103207.naphthalene.30.sqlite 4.266713020561451\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 15s 8ms/step\n",
      "../data/force_regression/dump.20210811155449.salicylic_acid.26.sqlite 7.835735870768453\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 15s 8ms/step\n",
      "../data/force_regression/dump.20210816124308.salicylic_acid.28.sqlite 8.557641750713588\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 13s 8ms/step\n",
      "../data/force_regression/dump.20210816135310.salicylic_acid.29.sqlite 8.550498919539201\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 8ms/step\n",
      "../data/force_regression/dump.20210818110002.salicylic_acid.30.sqlite 7.315169632582268\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 26s 12ms/step\n",
      "../data/force_regression/dump.20210906110653.salicylic_acid.27.sqlite 7.493096373051023\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 14s 7ms/step\n",
      "../data/force_regression/dump.20210811155449.toluene.26.sqlite 4.526431430070258\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 7ms/step\n",
      "../data/force_regression/dump.20210816162714.toluene.27.sqlite 3.366090659627319\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 6ms/step\n",
      "../data/force_regression/dump.20210816181839.toluene.28.sqlite 3.857017966371815\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 7ms/step\n",
      "../data/force_regression/dump.20210816223047.toluene.29.sqlite 3.6550610092776648\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 7ms/step\n",
      "../data/force_regression/dump.20210816232149.toluene.30.sqlite 3.7320726225927476\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 13s 6ms/step\n",
      "../data/force_regression/dump.20210811155449.uracil.26.sqlite 6.2026325847118455\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 5ms/step\n",
      "../data/force_regression/dump.20210817021954.uracil.27.sqlite 6.143570377365137\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 17s 5ms/step\n",
      "../data/force_regression/dump.20210817150950.uracil.28.sqlite 6.020514928315043\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 7ms/step\n",
      "../data/force_regression/dump.20210817230445.uracil.30.sqlite 5.922962310077607\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 5ms/step\n",
      "../data/force_regression/dump.20210823122427.uracil.129.sqlite 5.853160770965174\n",
      "MD17:aspirin 13.483038130383715 0.5506289935849099\n",
      "MD17:benzene 6.678660863771225 0.08209369793061042\n",
      "MD17:ethanol 4.647988709851395 0.14330065113973667\n",
      "MD17:malonaldehyde 8.265652468765087 0.17708672058721694\n",
      "MD17:naphthalene 4.059669223785477 0.10547669517156125\n",
      "MD17:salicylic_acid 7.950428509330907 0.23277915219543988\n",
      "MD17:toluene 3.827334737587961 0.1721762453979963\n",
      "MD17:uracil 6.028568194286962 0.058483890484671175\n"
     ]
    }
   ],
   "source": [
    "iterator = itertools.groupby(conn.execute('select * from md17 order by molecules'), lambda x: x[1])\n",
    "\n",
    "md17_location = '/tmp'\n",
    "\n",
    "errs = collections.defaultdict(list)\n",
    "\n",
    "for (mols, selection) in iterator:\n",
    "    for (rowid, _) in selection:\n",
    "        for (fname, desc) in conn.execute(\n",
    "            'select fname, workflow_description from data where rowid = ?', (rowid,)):\n",
    "\n",
    "            desc = json.loads(desc)\n",
    "            train_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'].endswith('Train')][0]\n",
    "            batch_size = max(1, desc['stages'][train_index]['arguments']['batch_size']//4)\n",
    "            desc['stages'] = desc['stages'][:train_index]\n",
    "\n",
    "            dset_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'].endswith('MD17')][0]\n",
    "            desc['stages'][dset_index]['arguments']['cache_dir'] = md17_location\n",
    "\n",
    "            workflow = flowws.Workflow.from_JSON(desc)\n",
    "            scope = workflow.run()\n",
    "\n",
    "            with keras_gtar.Trajectory(fname, 'r') as traj:\n",
    "                model = traj.load()\n",
    "\n",
    "            prediction = model.predict(scope['x_test'], batch_size=batch_size, verbose=1)\n",
    "            err = np.mean(np.abs(prediction - scope['y_test']))*scope['y_scale']\n",
    "\n",
    "            errs[mols].append(err)\n",
    "            print(fname, err)\n",
    "\n",
    "for m, vals in errs.items():\n",
    "    print(m, np.mean(vals), np.std(vals)/np.sqrt(len(vals)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## PDB coarse grain backmapping"
   ]
  },
  {
   "cell_type": "raw",
   "metadata": {
    "scrolled": false
   },
   "source": [
    "errs = []\n",
    "\n",
    "pdb_cache_dir = '/tmp'\n",
    "\n",
    "for (fname, desc) in conn.execute(\n",
    "    'select fname, workflow_description from data where dataset = \"PDBCoarseGrained\"'):\n",
    "\n",
    "    desc = json.loads(desc)\n",
    "\n",
    "    cache_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'] == 'PDBCache'][0]\n",
    "    desc['stages'][cache_index]['arguments']['cache_directory'] = pdb_cache_dir\n",
    "\n",
    "    train_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'].endswith('Train')][0]\n",
    "    steps = desc['stages'][train_index]['arguments']['generator_train_steps']\n",
    "    desc['stages'] = desc['stages'][:train_index]\n",
    "\n",
    "    workflow = flowws.Workflow.from_JSON(desc)\n",
    "    scope = workflow.run()\n",
    "\n",
    "    with keras_gtar.Trajectory(fname, 'r') as traj:\n",
    "        model = traj.load()\n",
    "\n",
    "    model_errs = []\n",
    "    for (x, y) in itertools.islice(scope['train_generator'], steps):\n",
    "        pred = model.predict_on_batch(x)\n",
    "        err = np.abs((pred - y))*scope['y_scale']\n",
    "        model_errs.append(err)\n",
    "\n",
    "    errs.append(np.mean(model_errs))\n",
    "\n",
    "print(np.mean(errs), np.std(errs)/np.sqrt(len(errs)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {
    "scrolled": false
   },
   "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": [
    "ds = DataSeries()\n",
    "\n",
    "minlength = np.inf\n",
    "for (fname,) in conn.execute(\n",
    "    'select fname from data where dataset = \"PDBCoarseGrained\"'):\n",
    "\n",
    "    with gtar.GTAR(fname, 'r') as traj:\n",
    "        for (index, vals) in traj.recordsNamed('val_mean_absolute_error'):\n",
    "            minlength = min(minlength, len(vals))\n",
    "            for (i, v) in enumerate(vals):\n",
    "                ds.add(i, v)\n",
    "\n",
    "mu, sigma = ds.mean, 2*ds.stderr\n",
    "pp.fill_between(ds.x, mu - sigma, mu + sigma, color='gray', alpha=.5)\n",
    "pp.plot(ds.x, ds.mean, color='black')\n",
    "pp.gca().set_xlim(0, minlength)\n",
    "pp.xlabel('Epoch')\n",
    "pp.ylabel('MAE');"
   ]
  },
  {
   "cell_type": "raw",
   "metadata": {
    "scrolled": false
   },
   "source": [
    "model_series = collections.defaultdict(DataSeries)\n",
    "\n",
    "pdb_cache_dir = '/tmp'\n",
    "for (fname, desc) in conn.execute(\n",
    "    'select fname, workflow_description from data where dataset = \"PDBCoarseGrained\"'):\n",
    "    desc = json.loads(desc)\n",
    "\n",
    "    model_index = [i for (i, stage) in enumerate(desc['stages']) if 'PDBInverseCoarseGrain' in stage['type']][0]\n",
    "    model_type = desc['stages'][model_index]['type']\n",
    "    model_width = desc['stages'][model_index]['arguments']['n_dim']\n",
    "\n",
    "    cache_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'] == 'PDBCache'][0]\n",
    "    desc['stages'][cache_index]['arguments']['cache_directory'] = pdb_cache_dir\n",
    "\n",
    "    train_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'].endswith('Train')][0]\n",
    "    steps = desc['stages'][train_index]['arguments']['generator_train_steps']\n",
    "    desc['stages'] = desc['stages'][:train_index]\n",
    "    \n",
    "    try:\n",
    "        with keras_gtar.Trajectory(fname, 'r') as traj:\n",
    "            for (index, vals) in traj.handle.recordsNamed('val_mean_absolute_error'):\n",
    "                vals -= np.min(vals)\n",
    "                vals /= np.max(vals)\n",
    "                model_series[model_type].add(model_width, np.correlate(vals, vals)[0])\n",
    "    except Exception as e:\n",
    "        print('skipping: {}'.format(e))\n",
    "        continue\n",
    "\n",
    "colors = pp.rcParams['axes.prop_cycle'].by_key()['color']\n",
    "model_types = list(sorted(model_series))\n",
    "label_map = dict(PDBInverseCoarseGrainTransformer='Transformer', \n",
    "                 PDBInverseCoarseGrain='Geometric Algebra')\n",
    "for color, model_type in zip(colors, model_types):\n",
    "    ds = model_series[model_type]\n",
    "    mu, sigma = ds.mean, 2*ds.stderr\n",
    "    pp.fill_between(ds.x, mu - sigma, mu + sigma, color=color, alpha=.5)\n",
    "    pp.plot(ds.x, ds.mean, color=color, label=label_map[model_type])\n",
    "pp.legend()\n",
    "pp.xlabel('Model Width')\n",
    "pp.gca().set_xscale('log')\n",
    "# pp.gca().set_yscale('log')\n",
    "pp.ylabel('Autocorrelation');"
   ]
  },
  {
   "cell_type": "raw",
   "metadata": {
    "scrolled": false
   },
   "source": [
    "model_series = collections.defaultdict(DataSeries)\n",
    "\n",
    "pdb_cache_dir = '/tmp'\n",
    "for (fname, desc) in conn.execute(\n",
    "    'select fname, workflow_description from data where dataset = \"PDBCoarseGrained\"'):\n",
    "    desc = json.loads(desc)\n",
    "\n",
    "    model_index = [i for (i, stage) in enumerate(desc['stages']) if 'PDBInverseCoarseGrain' in stage['type']][0]\n",
    "    model_type = desc['stages'][model_index]['type']\n",
    "    model_width = desc['stages'][model_index]['arguments']['n_dim']\n",
    "\n",
    "    cache_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'] == 'PDBCache'][0]\n",
    "    desc['stages'][cache_index]['arguments']['cache_directory'] = pdb_cache_dir\n",
    "\n",
    "    train_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'].endswith('Train')][0]\n",
    "    steps = desc['stages'][train_index]['arguments']['generator_train_steps']\n",
    "    desc['stages'] = desc['stages'][:train_index]\n",
    "    \n",
    "    with keras_gtar.Trajectory(fname, 'r') as traj:\n",
    "        for (index, vals) in traj.handle.recordsNamed('val_mean_absolute_error'):\n",
    "            vals -= np.min(vals)\n",
    "            vals /= np.max(vals)\n",
    "            m = np.log(vals)/np.arange(1, len(vals) + 1)\n",
    "            m[np.logical_not(np.isfinite(m))] = np.nan\n",
    "            m = np.nanmean(m)\n",
    "            m = -1./m\n",
    "            model_series[model_type].add(model_width, m)\n",
    "\n",
    "colors = pp.rcParams['axes.prop_cycle'].by_key()['color']\n",
    "model_types = list(sorted(model_series))\n",
    "label_map = dict(PDBInverseCoarseGrainTransformer='Transformer', \n",
    "                 PDBInverseCoarseGrain='Geometric Algebra')\n",
    "for color, model_type in zip(colors, model_types):\n",
    "    ds = model_series[model_type]\n",
    "    mu, sigma = ds.mean, 2*ds.stderr\n",
    "    pp.fill_between(ds.x, mu - sigma, mu + sigma, color=color, alpha=.5)\n",
    "    pp.plot(ds.x, ds.mean, color=color, label=label_map[model_type])\n",
    "pp.legend()\n",
    "pp.xlabel('Model Width')\n",
    "pp.gca().set_xscale('log')\n",
    "# pp.gca().set_yscale('log')\n",
    "pp.ylabel('Correlation Time');"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "data": {
      "image/png": "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+Mc/ntd+XV1dvOUtb6G+vp7rr7+e5cuX09jYyE9/+lPOPPNM1q1bx+zZs/ttu2vXLq6//npOO+00NmzYwO23385TTz3FmjVrBs0MvvKVr2CaJp/97GdJJBL84Ac/4JJLLuEXv/gFN9xwA5/4xCf44Ac/yO9+9zu+8Y1vMHfuXK677rre/R955BGuvvpqFixYwBe+8AVKS0t5+eWX+cY3vsHGjRv5/e9/D8APf/hDvvrVr9La2sptt93Wu//SpX0tUMLhMOeffz5nn3023/3ud4e9aWf4+te/zs0338yyZct63aW7d+/mwQcf5Fvf+hbuHGYEd999N3PnzuXcc89FKcWpp57KPffck/P/6dOf/jR33HEHF1xwATfeeCMtLS380z/9E3Pnzh20bX19PWeddRaJRIIbbriB+fPns2vXLm6//Xaefvpp1q1bR1FRUb99PvzhD+NwOPjyl79MKBTif//3f7n00kt59NFHueiii/ptu27dOh588EE+fsMNXPee96LSn/FNmzfz0KOPcuXbLmFubS0p0+TvLzzP1//939m7bx8/ve2HSCpFKhS2+6VNM5KmsKeNvDvl5sqYnOppNfNyoCVbUSJfROQ5pdScAYtXArtEZE/6XL8B3gk8M8zyEQ2UUuom9Myvl1CWGvNQWJbFI1tMGjpH2sqd/smPniTc/Yo1+oZDUFsMlywe0668+eabFBYWMmvWLMysJ1XTNOnIaqEAEAgE8Pl0Bta//du/sWfPHp5//nlOPfXU3m0+9KEPccopp/CNb3yDe+65B4BbbrmFnTt38t///d988pOf7N32pJNO4jOf+Qy33HIL3/rWtwB9jTPnf/HFF3tv5kuWLOHqq6/m2muv5YUXXuCMM84A4KMf/Sjz5s3jf/7nf3j/+/WTeSwW4/rrr2flypU88cQTvTGij3/845x44onceOONPPnkk6xevZorr7yS2267jWg02rt/9jUQEdra2vjqV7/ab6Zmmmbv9bIsq/fvNWvWcPPNN7N69Woefvjhfi7A7373u737Wtbw/+tDhw7x2GOP8bWvfa13uw996EN8/vOf58033+xnPDOISO8YNm/ezB133MHb3vY2Hn744d6C4GuuuYYVK1YMGv+nP/1pkskk69ato7a2tveY11xzDWeffTY/+MEP+OY3v9nv/+NwOHj22Wd7/z8f+chHWL58Of/8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DFojgu8I6tbjOYqF5HOK3cCKEdZfyCgZdQMRkfcPs/wRhqipGm750UISCRALRBfjYiiMbGFYT//5rsNQLCifGkWvPp+Pv/71r1xxxRVcc801rF69mre97W1UVVXR3d3Ntm3b+O1vf4vD4aCurs+T+t3vfpcXX3yR973vffzhD39g1apVuN1u9u/fzyOPPMKKFSu49957AZ1g8Pvf/55PfepTvPbaa5x66qls2LCBu+++m8WLF/OlL31pXN9TIBDgvvvu46qrrmLx4sVcf/31LFiwgM7OTrZt28Yf/vAH/vjHP7J69WpAJ148/PDDfPrTn+Ytb3kLDoeDCy+8kBkz8q+QWLlyJV/+8pf53ve+x4oVK3jve99LVVUVe/fu5YEHHmDNmjXDxrN+/etfE41Gede73jXs8d/1rndx77338otf/IKvfGXokr/ly5fziU98gjvvvJOLLrqIq6++mpaWFn7yk59w6qmnsn79+n6zxttvv51zzjmH8847jw9/+MOceuqpWJbFnj17eOihh/jwhz88qNYrlUpx7rnn8v73v59QKMQdd9xBNBrlxz/+cc7X6tprr+V///d/+eD/+zgXnr+a5tZWfvF/91NaWtq7jdUTRTkcqKxZujIMXKUlSCpF9/rXCG/aTNFZK/EvnG9rXE4SkpE0Mo7MI+HJ0fLkY6B+BfybUup36EJdSCcpKKW+AFwKfDaP4007zGgMlKELdAHl0soRYpoop7Pfl2sqMm/ePNavX88999zDAw88wA9+8AO6uroIBAIsWLCAj3/849xwww39gu5FRUW8+OKL3HrrrTzwwAM89NBDOJ1OamtrOeecc/rV62S2/eY3v8mf//xnfv7zn1NZWck//uM/8u///u8Tkl10ySWXsHbtWm655Rbuv/9+WlpaKCkpYf78+Xz+85/v55763Oc+x549e3jggQe44447sCyLp59+ekwGCnQG38knn8z//M//8J//+Z9YlkVdXR2XXXbZiPGsu+++G6fTyTve8Y5ht7n44ospKCjg5z//+bAGCuCnP/0p1dXV3H333dx4440sXryY22+/nTVr1rB+/fp+caW6ujrWr1/P9773PR566CHuv/9+vF4vdXV1XHnllbznPYNDyPfddx933HEHt9xyC52dnZx00knce++9XHzxxTleJfiv//ovCgoK+N3vfsdf/vY36mpquOEj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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "thresh = .75\n",
    "\n",
    "model_series = collections.defaultdict(DataSeries)\n",
    "\n",
    "pdb_cache_dir = '/tmp'\n",
    "for (fname, desc) in conn.execute(\n",
    "    'select fname, workflow_description from data where dataset = \"PDBCoarseGrained\"'):\n",
    "    desc = json.loads(desc)\n",
    "\n",
    "    model_index = [i for (i, stage) in enumerate(desc['stages']) if 'PDBInverseCoarseGrain' in stage['type']][0]\n",
    "    model_type = desc['stages'][model_index]['type']\n",
    "    model_width = desc['stages'][model_index]['arguments']['n_dim']\n",
    "\n",
    "    cache_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'] == 'PDBCache'][0]\n",
    "    desc['stages'][cache_index]['arguments']['cache_directory'] = pdb_cache_dir\n",
    "\n",
    "    train_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'].endswith('Train')][0]\n",
    "    steps = desc['stages'][train_index]['arguments']['generator_train_steps']\n",
    "    desc['stages'] = desc['stages'][:train_index]\n",
    "\n",
    "    with keras_gtar.Trajectory(fname, 'r') as traj:\n",
    "        for (index, vals) in traj.handle.recordsNamed('val_loss'):\n",
    "            worst, best = vals[0], np.min(vals)\n",
    "            thresh_val = thresh*best + (1 - thresh)*worst\n",
    "            indices = np.where(vals < thresh_val)[0]\n",
    "            model_series[model_type].add(model_width, np.min(indices))\n",
    "\n",
    "colors = pp.rcParams['axes.prop_cycle'].by_key()['color']\n",
    "markers = itertools.cycle(['o-', 'X-'])\n",
    "model_types = list(sorted(model_series))\n",
    "label_map = dict(PDBInverseCoarseGrainTransformer='Transformer',\n",
    "                 PDBInverseCoarseGrain='Geometric Algebra')\n",
    "for color, marker, model_type in zip(colors, markers, model_types):\n",
    "    ds = model_series[model_type]\n",
    "    mu, sigma = ds.mean, ds.stderr\n",
    "    pp.fill_between(ds.x, mu - sigma, mu + sigma, color=color, alpha=.5)\n",
    "    pp.plot(ds.x, ds.mean, marker, color=color, label=label_map[model_type], markersize=8)\n",
    "pp.legend()\n",
    "pp.xlabel('Model Width')\n",
    "pp.gca().set_xscale('log')\n",
    "pp.minorticks_off()\n",
    "xticks = 2**np.arange(3, 9)\n",
    "pp.xticks(xticks, xticks)\n",
    "pp.xlim(min(ds.x), max(ds.x))\n",
    "pp.ylabel('{:d}% Progress Time'.format(int(thresh*100)));\n",
    "pp.savefig('/tmp/cg_progress_time.svg');"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "data": {
      "image/png": "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pUv52Mc+RiQyRI6gBL/73e9oaGjg/e9/P7feeisdHR088cQTw9oXdGTel770JaqqqvB6vaxevZonnnhiQB/S8ePH+eAHP8jcuXNxu91UVlZy55130tDQkPT4HR0dfOxjH6O8vLz7+vz1r3/t3t7ljyrIsvjYv97GC//4K5evv5CywhyuvuZaRIS6ujruuusuTjvtNAoKCvB6vSxfvpyvf/3rxONjiKyYgdgiNAY4KVAefS+JTAHzHqSw5PtM55ldcWraUn/czig8sHF0xd/m5MNVy0cXJvzrX/8agA984AMj2s/n83Heeedx9OhRbr/9dlasWMHx48f54Q9/yNlnn83mzZupqqrq1fbAgQPcfvvtnH766WzdupUf/ehH/O1vf2PTpk39Rgaf+cxniMfjfPzjHycSifCtb32LK664gp///Ofccccd3Hnnnbzvfe/jl7/8JV/4wheYP38+N998c/f+f/zjH7n++utZtGgRd911F4WFhbzyyit84QtfYNu2bfzqV78C4Lvf/S733HMPTU1NfOc73+nef9myZd3/BwIBLr74Ys4//3y+8pWvDHjT7uJzn/scX/3qV1m+fHm3ufTgwYP85je/4Ytf/CJu99CPsw888ADz58/nwgsvRCnFmjVrePDBB4d9nT7ykY/w4x//mEsuuYRPfvKTNDY28qEPfYj58+f3a3v06FEuuOACIpEId9xxBwsXLuTAgQP86Ec/4u9//zubN2/uF9p9yy234HA4uPvuu/H7/fzkJz/hyiuv5JlnnuHSSy8FTt4U39i6md89/Rtufv8HePf7bsFh6RvnG2+80T1qX7hwIdFolGeeeYbPfOYzHDp0iJ/85CfDeq8GqG0VAmEhHNPm1EyXFigBitPcvAfo+QQjfaFLarx3NPtOtZf+iAZn165d8sedMfnJhvR6/XFnbMi+D0RhYaHk5ub2Wx+LxaSxsbHXq7Ozs3vbxz72MfF6vbJt27Ze+x0+fFhycnLk/e9/f/e6z372swLID37wg15tv//97wsgn//857vXPfTQQwLImjVrJBwOd69/+umnBRCHwyGbNm3qXh8Oh6W8vFzOOeec7nXBYFDKysrkwgsvlGg02uuc3/72twWQv//9793rLr74Yqmqqkr6+Vx88cUCyOc+97l+26qrqwWQe++9t3vdq6++KoBccsklEgwGe7W3bVts2056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3dCPtQmNfu13isSE6kRQRFkO5PcJfIjFhZitWDNb9ROuqY4RqDSg/jdP0/jU73qt67k80SJ1880391qOxWLcf//9nHvuuf229cXv95OTkzOe3RsWJ06cAKCwsDClxxUROjo6yM7OHrrxBBCNRonH43i93snuyqTgHMJJ0dShRak8F7LcCpdD4XLoYIOjrdqnU5EjzM5XZKVJQEVPv5NAd4byHA9U5vZu2xWxt7ICcr3p0f9UYnxQaUbjU7/rJ1bpyrx581i3bh1bt27liiuuoLCwsNuM5vf7+fznP8/ZZ59NcXExHo+HRYsW8ZnPfIbOzs5ex/nHP/6BUoqHH36Yhx56iBUrVuDxeKiqquI///M/+5335Zdf5qqrrqK8vByv18usWbN461vfysaNGwFYt24dF198MQC33XYbSmmH8eHDhwHo6OjgnnvuYeHChXg8HsrLy7nllls4cuTIgP36wQ9+wPLly/F6vXzzm9/k8OHD3X6tX/7yl5x22mlkZGSwaNEiHnroIQCOHj3KTTfdRGFhITk5Odx88834/f5+7+f48eN88IMfZO7cubjdbiorK7nzzjtpaGjo1e6+++5DKcXOnTv5t3/7N2bPno3X6+1+3zMRj7N/3aMuLKWzTvhCsLcBDjQJgbD2QTksRZZbkekSTviF144K2+ts2oLdlbQnhWjC79SVZ6+27WSG8vmFvc3vIkIgDAuKFaU50/NWbkZQKeLwN/6LwBs7Un7csQhW9qmrmPepj6e4R705evQo69ev5x3veAfXX399t/gcO3aMn/3sZ9x44428973vxel08vzzz/Of//mfbN26NWlG7R//+MfU19dzxx13kJ+fzy9+8QvuvvtuZs+ezXvf+14A9u7dy2WXXUZ5eTkf//jHKSsr48SJE2zYsIE33niDc845h8997nOcf/75fPWrX+1lliwpKSEWi3HFFVewYcMGbrrpJu666y7279/Pj370I5577jk2b97M7Nmze/Xru9/9Ls3NzfzzP/8z5eXlzJkzp3vb73//e3784x/zoQ99iMLCQh544AFuv/123G43n/3sZ1m/fj1f/epXee2113jwwQfxer387Gc/6/X5nXvuuUQiEe644w4WLlzIgQMH+NGPfsTf//53Nm/eTF5eXq/+vO997yMjI4O77roLpRQVFRWpuZhTEKUUuV6hr0a5LFhYrIMp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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "model_series = collections.defaultdict(DataSeries)\n",
    "\n",
    "pdb_cache_dir = '/tmp'\n",
    "\n",
    "try:\n",
    "    if not 'force cache rebuild':\n",
    "        raise NameError()\n",
    "    get_model_errs()\n",
    "except NameError:\n",
    "    @functools.lru_cache\n",
    "    def get_model_errs(rowid=None):\n",
    "        if rowid is None:\n",
    "            return\n",
    "\n",
    "        for (fname, desc) in conn.execute(\n",
    "            'select fname, workflow_description from data where rowid = ?', (rowid,)):\n",
    "\n",
    "            desc = json.loads(desc)\n",
    "\n",
    "            model_type = [stage['type'] for stage in desc['stages'] if 'PDBInverseCoarseGrain' in stage['type']][0]\n",
    "\n",
    "            cache_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'] == 'PDBCache'][0]\n",
    "            desc['stages'][cache_index]['arguments']['cache_directory'] = pdb_cache_dir\n",
    "\n",
    "            train_index = [i for (i, stage) in enumerate(desc['stages']) if stage['type'].endswith('Train')][0]\n",
    "            steps = desc['stages'][train_index]['arguments']['generator_train_steps']\n",
    "            desc['stages'] = desc['stages'][:train_index]\n",
    "\n",
    "            workflow = flowws.Workflow.from_JSON(desc)\n",
    "            scope = workflow.run()\n",
    "\n",
    "            try:\n",
    "                with keras_gtar.Trajectory(fname, 'r') as traj:\n",
    "                    model = traj.load()\n",
    "            except Exception as e:\n",
    "                print('skipping: {}'.format(e))\n",
    "                return None, None, None\n",
    "\n",
    "            model_size = sum([np.prod(v.shape) for v in model.trainable_weights])\n",
    "\n",
    "            model_errs = []\n",
    "            for (x, y) in itertools.islice(scope['train_generator'], steps):\n",
    "                pred = model.predict_on_batch(x)\n",
    "                err = np.abs((pred - y))*scope['y_scale']\n",
    "                model_errs.append(err)\n",
    "\n",
    "            return model_type, model_size, model_errs\n",
    "\n",
    "for (fname, rowid) in conn.execute(\n",
    "    'select fname, ROWID from data where dataset = \"PDBCoarseGrained\"'):\n",
    "    (model_type, model_size, model_errs) = get_model_errs(rowid)\n",
    "\n",
    "    if model_type is not None:\n",
    "        model_series[model_type].add(model_size, np.mean(model_errs))\n",
    "\n",
    "colors = pp.rcParams['axes.prop_cycle'].by_key()['color']\n",
    "markers = itertools.cycle(['o-', 'X-'])\n",
    "label_map = dict(PDBInverseCoarseGrainTransformer='Transformer',\n",
    "                 PDBInverseCoarseGrain='Geometric Algebra')\n",
    "for color, marker, (model_type, ds) in zip(colors, markers, sorted(model_series.items())):\n",
    "    mu, sigma = ds.mean, ds.stderr\n",
    "    pp.fill_between(ds.x, mu - sigma, mu + sigma, color=color, alpha=.5)\n",
    "    pp.plot(ds.x, ds.mean, marker, color=color, label=label_map[model_type], markersize=8)\n",
    "pp.legend()\n",
    "pp.xlabel('Model Size')\n",
    "pp.gca().set_xscale('log')\n",
    "pp.gca().set_yscale('log')\n",
    "pp.ylabel('Training MAE/$\\\\AA$');\n",
    "pp.savefig('/tmp/cg_error_scaling.svg');"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## MD17 validation error"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n",
      "Unused arguments found for stage: ['save_model', 'hash_size', 'file_modifiers']\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MD17:aspirin\n",
      "[12.063223838806152, 32, 6, True, True]\n",
      "[12.56020450592041, 32, 6, True, True]\n",
      "[13.545938491821289, 32, 6, True, True]\n",
      "[13.83035659790039, 32, 6, True, True]\n",
      "[15.086081504821777, 32, 6, True, True]\n",
      "MD17:benzene\n",
      "[6.4762187004089355, 32, 6, True, True]\n",
      "[6.5197553634643555, 32, 6, True, True]\n",
      "[6.590536117553711, 32, 6, True, True]\n",
      "[6.7103095054626465, 32, 6, True, True]\n",
      "[6.741830348968506, 32, 6, True, True]\n",
      "MD17:ethanol\n",
      "[4.14215087890625, 32, 6, True, True]\n",
      "[4.231337547302246, 32, 6, True, True]\n",
      "[4.540034770965576, 32, 6, True, True]\n",
      "[4.676665306091309, 32, 6, True, True]\n",
      "[5.401256561279297, 32, 6, True, True]\n",
      "MD17:malonaldehyde\n",
      "[7.234607219696045, 32, 6, True, True]\n",
      "[7.901865005493164, 32, 6, True, True]\n",
      "[8.524330139160156, 32, 6, True, True]\n",
      "[8.665929794311523, 32, 6, True, True]\n",
      "[9.234832763671875, 32, 6, True, True]\n",
      "MD17:naphthalene\n",
      "[3.675591468811035, 32, 6, True, True]\n",
      "[3.9067883491516113, 32, 6, True, True]\n",
      "[4.200793266296387, 32, 6, True, True]\n",
      "[4.225848197937012, 32, 6, True, True]\n",
      "[4.2347540855407715, 32, 6, True, True]\n",
      "MD17:salicylic_acid\n",
      "[7.245472431182861, 32, 6, True, True]\n",
      "[7.37279748916626, 32, 10, True, True]\n",
      "[7.751132488250732, 32, 6, True, True]\n",
      "[8.287615776062012, 32, 6, True, True]\n",
      "[8.460135459899902, 32, 6, True, True]\n",
      "MD17:toluene\n",
      "[3.4551753997802734, 32, 6, True, True]\n",
      "[3.732964515686035, 32, 6, True, True]\n",
      "[3.7447140216827393, 32, 6, True, True]\n",
      "[3.869861364364624, 32, 6, True, True]\n",
      "[4.797181129455566, 32, 6, True, True]\n",
      "MD17:uracil\n",
      "[5.83696985244751, 32, 6, True, True]\n",
      "[5.903253078460693, 32, 6, True, True]\n",
      "[6.0041093826293945, 32, 6, True, True]\n",
      "[6.1087965965271, 32, 6, True, True]\n",
      "[6.275685787200928, 32, 6, True, True]\n"
     ]
    }
   ],
   "source": [
    "import itertools\n",
    "import json\n",
    "import flowws\n",
    "\n",
    "runs = []\n",
    "for (rowid, wflow) in conn.execute('select ROWID, workflow_description from data where dataset LIKE \"%MD17\"'):\n",
    "    for (molecules,) in conn.execute('select molecules from md17 where data_id = ?', (rowid,)):\n",
    "        pass\n",
    "    for (values,) in conn.execute('select arr_value from varyings where data_id = ? and name = ?', (rowid, 'val_mean_absolute_error')):\n",
    "        pass\n",
    "\n",
    "    w = flowws.Workflow.from_JSON(json.loads(wflow))\n",
    "    args = w.stages[2].arguments\n",
    "\n",
    "    dim = args['n_dim']\n",
    "    blocks = args['n_blocks']\n",
    "    residual = args['residual']\n",
    "    nonlin = args['block_nonlinearity']\n",
    "\n",
    "    opt = np.min(values)\n",
    "    if np.isfinite(opt):\n",
    "        runs.append([molecules, opt, dim, blocks, residual, nonlin, w])\n",
    "\n",
    "runs.sort()\n",
    "\n",
    "for (mol, bits) in itertools.groupby(runs, lambda x: x[0]):\n",
    "    print(mol)\n",
    "    for i, b in enumerate(bits):\n",
    "        print(b[1:6])\n",
    "        if i >= 4:\n",
    "            break\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training times\n",
    "\n",
    "These use approximate times-per-epoch to estimate how long training actually took, given the number of epochs used in training."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "49.583333333333336\n"
     ]
    }
   ],
   "source": [
    "seconds_per_epoch = 7\n",
    "epochs = []\n",
    "for (rowid,) in conn.execute(\n",
    "    'select rowid from data where dataset = \"PyriodicDataset\"'):\n",
    "    for (vals,) in conn.execute('select arr_value from varyings where name = \"loss\" and data_id = ?', (rowid,)):\n",
    "        epochs.append(len(vals))\n",
    "print(np.mean(epochs)*seconds_per_epoch/60)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MD17:aspirin 3120.36\n",
      "MD17:aspirin (52.01 hours)\n",
      "MD17:benzene 1401.66\n",
      "MD17:benzene (23.36 hours)\n",
      "MD17:ethanol 1254.87\n",
      "MD17:ethanol (20.91 hours)\n",
      "MD17:malonaldehyde 1087.10\n",
      "MD17:malonaldehyde (18.12 hours)\n",
      "MD17:naphthalene 5284.53\n",
      "MD17:naphthalene (88.08 hours)\n",
      "MD17:salicylic_acid 2359.50\n",
      "MD17:salicylic_acid (39.33 hours)\n",
      "MD17:toluene 2794.67\n",
      "MD17:toluene (46.58 hours)\n",
      "MD17:uracil 2020.65\n",
      "MD17:uracil (33.68 hours)\n"
     ]
    }
   ],
   "source": [
    "seconds_per_epoch = dict(\n",
    "    benzene=3,\n",
    "    uracil=3,\n",
    "    naphthalene=7,\n",
    "    aspirin=9,\n",
    "    salicylic_acid=5,\n",
    "    malonaldehyde=2,\n",
    "    ethanol=2,\n",
    "    toluene=4,\n",
    ")\n",
    "seconds_per_epoch['benzene,uracil,naphthalene,aspirin,salicylic_acid,malonaldehyde,ethanol,toluene'] = 71\n",
    "seconds_per_epoch = {'MD17:{}'.format(k): v for (k, v) in seconds_per_epoch.items()}\n",
    "epochs = collections.defaultdict(list)\n",
    "for (rowid,) in conn.execute(\n",
    "    'select rowid from data where dataset = \"MD17\"'):\n",
    "    for (molecules,) in conn.execute('select molecules from md17 where data_id = ?', (rowid,)):\n",
    "        pass\n",
    "    for (vals,) in conn.execute('select arr_value from varyings where name = \"loss\" and data_id = ?', (rowid,)):\n",
    "        epochs[molecules].append(len(vals))\n",
    "\n",
    "for (name, epoch_counts) in sorted(epochs.items()):\n",
    "    mu = np.mean(epoch_counts)*seconds_per_epoch[name]/60\n",
    "    print(name, '{:.02f}'.format(mu))\n",
    "    if ',' in name or mu > 60:\n",
    "        print(name, '({:.02f} hours)'.format(mu/60))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "9.221922222222222 hours\n"
     ]
    }
   ],
   "source": [
    "seconds_per_epoch = 47\n",
    "epochs = []\n",
    "for (rowid,) in conn.execute(\n",
    "    'select rowid from data where dataset = \"PDBCoarseGrained\" AND workflow_description NOT LIKE \"%PDBInverseCoarseGrainTransformer%\"'):\n",
    "    for (vals,) in conn.execute('select arr_value from varyings where name = \"loss\" and data_id = ?', (rowid,)):\n",
    "        epochs.append(len(vals))\n",
    "print(np.mean(epochs)*seconds_per_epoch/60/60, 'hours')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
