{
 "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 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": [
      "(55,)\n"
     ]
    }
   ],
   "source": [
    "dbases = []\n",
    "\n",
    "dbases.append('../data')\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'] == '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": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/matthew/env/default_20201216/lib/python3.8/site-packages/tensorflow_addons/utils/ensure_tf_install.py:54: UserWarning: Tensorflow Addons supports using Python ops for all Tensorflow versions above or equal to 2.2.0 and strictly below 2.4.0 (nightly versions are not supported). \n",
      " The versions of TensorFlow you are currently using is 2.4.1 and is not supported. \n",
      "Some things might work, some things might not.\n",
      "If you were to encounter a bug, do not file an issue.\n",
      "If you want to make sure you're using a tested and supported configuration, either change the TensorFlow version or the TensorFlow Addons's version. \n",
      "You can find the compatibility matrix in TensorFlow Addon's readme:\n",
      "https://github.com/tensorflow/addons\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 4s 3ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 2s 1ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 2s 1ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 2s 1ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "615/615 [==============================] - 2s 1ms/step\n",
      "0.9874465920651069 0.0017101677327771255\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": 5,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 16s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 14s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 14s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 14s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 14s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 6ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 7s 6ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 8s 6ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 7s 6ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 7s 6ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "2000/2000 [==============================] - 53s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "2000/2000 [==============================] - 52s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "2000/2000 [==============================] - 52s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "2000/2000 [==============================] - 52s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "2000/2000 [==============================] - 52s 22ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 10s 4ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 3ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 8s 4ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 8s 4ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 4ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 8s 4ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 8s 4ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 8s 4ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 8s 4ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 8s 4ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 13s 14ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 12s 14ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 14ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 14ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 14ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 11ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 11ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 11ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 11ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 11ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 11s 9ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 9ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 9ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 9ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 9ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 8s 6ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 9s 6ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 7s 6ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 7s 6ms/step\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "250/250 [==============================] - 7s 6ms/step\n",
      "MD17:aspirin 36.99101367690667 1.1042729532677409\n",
      "MD17:benzene 11.764758906964738 0.4953970557676517\n",
      "MD17:benzene,uracil,naphthalene,aspirin,salicylic_acid,malonaldehyde,ethanol,toluene 10.666215211184223 0.2364599184629445\n",
      "MD17:ethanol 21.437817424967996 0.47578129261973806\n",
      "MD17:malonaldehyde 30.57366003883384 1.117635110480079\n",
      "MD17:naphthalene 23.67018733657817 0.9669978837491647\n",
      "MD17:salicylic_acid 30.168514139848874 1.2489827187377458\n",
      "MD17:toluene 20.532730626625643 1.3310541408730818\n",
      "MD17:uracil 27.407266396493657 0.7781149949144871\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",
    "\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": "code",
   "execution_count": 6,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1 Physical GPUs, 1 Logical GPUs\n",
      "19 final records\n",
      "0 skipped records\n",
      "Max number of atoms in a residue: 14\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "19 final records\n",
      "0 skipped records\n",
      "Max number of atoms in a residue: 14\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "19 final records\n",
      "0 skipped records\n",
      "Max number of atoms in a residue: 14\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "19 final records\n",
      "0 skipped records\n",
      "Max number of atoms in a residue: 14\n",
      "1 Physical GPUs, 1 Logical GPUs\n",
      "19 final records\n",
      "0 skipped records\n",
      "Max number of atoms in a residue: 14\n",
      "0.12773171 0.0018086921592963216\n"
     ]
    }
   ],
   "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": 7,
   "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": "markdown",
   "metadata": {},
   "source": [
    "## MD17 validation error"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MD17:aspirin\n",
      "[32.964508056640625, 32, 6, True, True]\n",
      "[35.79458999633789, 32, 6, True, True]\n",
      "[37.25632095336914, 32, 6, True, True]\n",
      "[39.452945709228516, 32, 6, True, True]\n",
      "[39.75627899169922, 32, 6, True, True]\n",
      "MD17:benzene\n",
      "[10.182721138000488, 32, 6, True, True]\n",
      "[11.186267852783203, 32, 6, True, True]\n",
      "[11.416364669799805, 32, 6, True, True]\n",
      "[11.721916198730469, 32, 6, True, True]\n",
      "[13.775314331054688, 32, 6, True, True]\n",
      "MD17:benzene,uracil,naphthalene,aspirin,salicylic_acid,malonaldehyde,ethanol,toluene\n",
      "[9.896486282348633, 32, 6, True, True]\n",
      "[10.489762306213379, 32, 6, True, True]\n",
      "[10.687152862548828, 32, 6, True, True]\n",
      "[11.230234146118164, 32, 6, True, True]\n",
      "[11.292844772338867, 32, 6, True, True]\n",
      "MD17:ethanol\n",
      "[19.217145919799805, 32, 6, True, True]\n",
      "[20.62574577331543, 32, 6, True, True]\n",
      "[22.15045738220215, 32, 6, True, True]\n",
      "[22.185142517089844, 32, 6, True, True]\n",
      "[22.28136444091797, 32, 6, True, True]\n",
      "MD17:malonaldehyde\n",
      "[27.68175506591797, 32, 6, True, True]\n",
      "[28.531980514526367, 32, 6, True, True]\n",
      "[29.238136291503906, 32, 6, True, True]\n",
      "[32.286773681640625, 32, 6, True, True]\n",
      "[33.39577865600586, 32, 6, True, True]\n",
      "MD17:naphthalene\n",
      "[20.215312957763672, 32, 6, True, True]\n",
      "[22.803112030029297, 32, 6, True, True]\n",
      "[23.142925262451172, 32, 6, True, True]\n",
      "[24.473812103271484, 32, 6, True, True]\n",
      "[27.727848052978516, 32, 6, True, True]\n",
      "MD17:salicylic_acid\n",
      "[25.250476837158203, 32, 6, True, True]\n",
      "[29.978271484375, 32, 6, True, True]\n",
      "[30.733110427856445, 32, 6, True, True]\n",
      "[31.46633529663086, 32, 6, True, True]\n",
      "[33.941471099853516, 32, 6, True, True]\n",
      "MD17:toluene\n",
      "[17.826417922973633, 32, 6, True, True]\n",
      "[18.6573486328125, 32, 6, True, True]\n",
      "[19.00404930114746, 32, 6, True, True]\n",
      "[20.185976028442383, 32, 6, True, True]\n",
      "[25.715511322021484, 32, 6, True, True]\n",
      "MD17:uracil\n",
      "[25.167953491210938, 32, 6, True, True]\n",
      "[26.561229705810547, 32, 6, True, True]\n",
      "[26.78890037536621, 32, 6, True, True]\n",
      "[28.915441513061523, 32, 6, True, True]\n",
      "[29.16353988647461, 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": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "37.63666666666667\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": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MD17:aspirin 120.00\n",
      "MD17:benzene 40.00\n",
      "MD17:benzene,uracil,naphthalene,aspirin,salicylic_acid,malonaldehyde,ethanol,toluene 945.48\n",
      "MD17:benzene,uracil,naphthalene,aspirin,salicylic_acid,malonaldehyde,ethanol,toluene (15.76 hours)\n",
      "MD17:ethanol 26.04\n",
      "MD17:malonaldehyde 25.60\n",
      "MD17:naphthalene 92.68\n",
      "MD17:salicylic_acid 65.82\n",
      "MD17:toluene 53.33\n",
      "MD17:uracil 40.00\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:\n",
    "        print(name, '({:.02f} hours)'.format(mu/60))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "182.82\n"
     ]
    }
   ],
   "source": [
    "seconds_per_epoch = 18\n",
    "epochs = []\n",
    "for (rowid,) in conn.execute(\n",
    "    'select rowid from data where dataset = \"PDBCoarseGrained\"'):\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": 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.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
