updated jupyter notebooks for 1dot18dot0 release
diff --git a/community-artifacts/Deep-learning/.ipynb_checkpoints/Define-custom-functions-v1-checkpoint.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/Define-custom-functions-v1-checkpoint.ipynb
new file mode 100755
index 0000000..1e5c0f1
--- /dev/null
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/Define-custom-functions-v1-checkpoint.ipynb
@@ -0,0 +1,536 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Define custom functions\n",
+    "This function loads custom Python functions into a table for use by deep learning algorithms.\n",
+    "\n",
+    "Custom functions can be useful if, for example, you need loss functions or metrics that are not built into the standard libraries. The functions to be loaded must be in the form of serialized Python objects created using Dill, which extends Python's pickle module to the majority of the built-in Python types.\n",
+    "\n",
+    "Custom functions are also used to return top k categorical accuracy rate in the case that you want a different k value than the default from Keras. This module includes a helper function to create the custom function automatically for a specified k.\n",
+    "\n",
+    "This method was added in MADlib 1.18.0.\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#load_psycopg2\">1. Load object using psycopg2</a>\n",
+    "\n",
+    "<a href=\"#load_plpython\">2. Load object using a PL/Python function</a>\n",
+    "\n",
+    "<a href=\"#delete_object\">3. Delete object</a>\n",
+    "\n",
+    "<a href=\"#top_k\">4. Top k accuracy function</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 12,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "The sql extension is already loaded. To reload it, use:\n",
+      "  %reload_ext sql\n"
+     ]
+    }
+   ],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 13,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 14,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-84-g0256b81, cmake configuration time: Thu Mar  4 00:16:11 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-84-g0256b81, cmake configuration time: Thu Mar  4 00:16:11 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 14,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_psycopg2\"></a>\n",
+    "# 1. Load object using psycopg2\n",
+    "Psycopg is a PostgreSQL database adapter for the Python programming language. Note need to use the psycopg2.Binary() method to pass as bytes."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 15,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# import database connector psycopg2 and create connection cursor\n",
+    "import psycopg2 as p2\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
+    "cur = conn.cursor()\n",
+    "\n",
+    "# import Dill and define functions\n",
+    "import dill\n",
+    "\n",
+    "# custom loss\n",
+    "def squared_error(y_true, y_pred):\n",
+    "    import keras.backend as K \n",
+    "    return K.square(y_pred - y_true)\n",
+    "pb_squared_error=dill.dumps(squared_error)\n",
+    "\n",
+    "# custom metric\n",
+    "def rmse(y_true, y_pred):\n",
+    "    import keras.backend as K \n",
+    "    return K.sqrt(K.mean(K.square(y_pred - y_true), axis=-1))\n",
+    "pb_rmse=dill.dumps(rmse)\n",
+    "\n",
+    "# call load function\n",
+    "cur.execute(\"DROP TABLE IF EXISTS madlib.custom_function_table\")\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'squared_error', 'squared error')\", [p2.Binary(pb_squared_error)])\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'rmse', 'root mean square error')\", [p2.Binary(pb_rmse)])\n",
+    "conn.commit()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 16,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>squared_error</td>\n",
+       "        <td>squared error</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>rmse</td>\n",
+       "        <td>root mean square error</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'squared_error', u'squared error'),\n",
+       " (2, u'rmse', u'root mean square error')]"
+      ]
+     },
+     "execution_count": 16,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_plpython\"></a>\n",
+    "# 2. Load object using a PL/Python function"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 17,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 17,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "CREATE OR REPLACE FUNCTION custom_function_squared_error()\n",
+    "RETURNS BYTEA AS\n",
+    "$$\n",
+    "import dill\n",
+    "def squared_error(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.square(y_pred - y_true)\n",
+    "pb_squared_error=dill.dumps(squared_error)\n",
+    "return pb_squared_error\n",
+    "$$ language plpythonu;\n",
+    "CREATE OR REPLACE FUNCTION custom_function_rmse()\n",
+    "RETURNS BYTEA AS\n",
+    "$$\n",
+    "import dill\n",
+    "def rmse(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.sqrt(K.mean(K.square(y_pred - y_true), axis=-1))\n",
+    "pb_rmse=dill.dumps(rmse)\n",
+    "return pb_rmse\n",
+    "$$ language plpythonu;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 18,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>load_custom_function</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 18,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS madlib.custom_function_table;\n",
+    "SELECT madlib.load_custom_function('custom_function_table', \n",
+    "                                   custom_function_squared_error(), \n",
+    "                                   'squared_error', \n",
+    "                                   'squared error');\n",
+    "\n",
+    "SELECT madlib.load_custom_function('custom_function_table', \n",
+    "                                   custom_function_rmse(), \n",
+    "                                   'rmse', \n",
+    "                                   'root mean square error');"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 19,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>squared_error</td>\n",
+       "        <td>squared error</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>rmse</td>\n",
+       "        <td>root mean square error</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'squared_error', u'squared error'),\n",
+       " (2, u'rmse', u'root mean square error')]"
+      ]
+     },
+     "execution_count": 19,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"delete_object\"></a>\n",
+    "# 3. Delete object\n",
+    "Delete by id:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 20,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>rmse</td>\n",
+       "        <td>root mean square error</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2, u'rmse', u'root mean square error')]"
+      ]
+     },
+     "execution_count": 20,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.delete_custom_function( 'custom_function_table', 1);\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Delete by name:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 21,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>delete_custom_function</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 21,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.delete_custom_function( 'custom_function_table', 'rmse');"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Since this was the last object in the table, if you delete it then the table will also be dropped."
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"top_k\"></a>\n",
+    "# 4. Top k accuracy function\n",
+    "Load top 3 accuracy function followed by a top 10 accuracy function:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 22,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>top_3_accuracy</td>\n",
+       "        <td>returns top_3_accuracy</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>top_10_accuracy</td>\n",
+       "        <td>returns top_10_accuracy</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'top_3_accuracy', u'returns top_3_accuracy'),\n",
+       " (2, u'top_10_accuracy', u'returns top_10_accuracy')]"
+      ]
+     },
+     "execution_count": 22,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS madlib.custom_function_table;\n",
+    "\n",
+    "SELECT madlib.load_top_k_accuracy_function('custom_function_table',\n",
+    "                                           3);\n",
+    "\n",
+    "SELECT madlib.load_top_k_accuracy_function('custom_function_table',\n",
+    "                                           10);\n",
+    "\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/.ipynb_checkpoints/Define-model-configurations-v2-checkpoint.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/Define-model-configurations-v2-checkpoint.ipynb
new file mode 100755
index 0000000..fbe5ee5
--- /dev/null
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/Define-model-configurations-v2-checkpoint.ipynb
@@ -0,0 +1,2025 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Define model configurations\n",
+    "This module generates model configurations using grid search or random search.\n",
+    "\n",
+    "Once the configurations are defined, they can be used by the fit function in Train Model Configurations. By model configurations we mean both hyperparameters and model architectures. The output table from this module defines the combinations of model architectures, compile and fit parameters to be trained in parallel.\n",
+    "\n",
+    "This utility was added in MADlib 1.17.0.  Improvements were made in MADlib 1.18.0 including support for custom loss functions and custom metrics.\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#define_model_arch\">1. Define model architecture table</a>\n",
+    "\n",
+    "<a href=\"#load_model_arch\">2. Load model architecture</a>\n",
+    "\n",
+    "<a href=\"#generate_configs\">3. Generate model configurations</a>\n",
+    "\n",
+    "  - <a href=\"#grid_search\">3a. Grid search</a>\n",
+    "  \n",
+    "  - <a href=\"#random_search\">3b. Random search</a>\n",
+    "  \n",
+    "  - <a href=\"#incremental_load\">3c. Incremental loading</a>\n",
+    "  \n",
+    "<a href=\"#load_model_selection_manual\">4. Create model selection table manually</a>\n",
+    "\n",
+    "<a href=\"#custom\">5. Custom loss functions and custom metrics NOT COMPLETE</a>\n",
+    "\n",
+    "<a href=\"#load_model_selection\">6. Load model selection table [deprecated]</a>\n"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 37,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "The sql extension is already loaded. To reload it, use:\n",
+      "  %reload_ext sql\n"
+     ]
+    }
+   ],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 38,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP (PM demo machine) - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 39,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 39,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"define_model_arch\"></a>\n",
+    "# 1. Define model architecture table\n",
+    "The model selection loader works in conjunction with the model architecture table, so we first create a model architecture table with two different models.  See http://madlib.apache.org/docs/latest/group__grp__keras__model__arch.html for more details on the model architecture table.\n",
+    "\n",
+    "Import Keras libraries"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 40,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 1 hidden layer:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 41,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_4\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_14 (Dense)             (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_15 (Dense)             (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_16 (Dense)             (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 193\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model1 = Sequential()\n",
+    "model1.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model1.add(Dense(10, activation='relu'))\n",
+    "model1.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model1.summary();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 42,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_14\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_15\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_16\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_4\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 42,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model1.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 2, \"batch_input_shape\": [null, 3], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"new_dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'\n",
+    "        "
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 2 hidden layers:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 43,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_5\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_17 (Dense)             (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_18 (Dense)             (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_19 (Dense)             (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_20 (Dense)             (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 303\n",
+      "Trainable params: 303\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model2 = Sequential()\n",
+    "model2.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model2.summary();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 44,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_17\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_18\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_19\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_20\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_5\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 44,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model2.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_model_arch\"></a>\n",
+    "# 2. Load model architecture\n",
+    "\n",
+    "Load both into model architecture table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 45,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>model_weights</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>__internal_madlib_id__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>MLP with 1 hidden layer</td>\n",
+       "        <td>__madlib_temp_61202069_1614901986_7314581__</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>MLP with 2 hidden layers</td>\n",
+       "        <td>__madlib_temp_12006647_1614901987_43673839__</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'MLP with 1 hidden layer', u'__madlib_temp_61202069_1614901986_7314581__'),\n",
+       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1835 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'MLP with 2 hidden layers', u'__madlib_temp_12006647_1614901987_43673839__')]"
+      ]
+     },
+     "execution_count": 45,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS model_arch_library;\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Sophie',              -- Name\n",
+    "                               'MLP with 1 hidden layer'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Maria',               -- Name\n",
+    "                               'MLP with 2 hidden layers'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT * FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"generate_configs\"></a>\n",
+    "# 3. Generate model configurations\n",
+    "\n",
+    "<a id=\"grid_search\"></a>\n",
+    "## 3a. Grid search\n",
+    "\n",
+    "The output table for grid search contains the unique combinations of model architectures, compile and fit parameters."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 46,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "16 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128')]"
+      ]
+     },
+     "execution_count": 46,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [ {'optimizer': ['Adam', 'SGD'], 'lr': [0.001, 0.01]} ], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'grid'               -- search_type \n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Note that above uses the same learning rate for the two optimizers.  If you wanted to use different learning rates and different parameters for different optimizers (common):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 47,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "20 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 2, u\"optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 2, u\"optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 1, u\"optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 1, u\"optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 1, u\"optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 1, u\"optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (17, 2, u\"optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (18, 2, u\"optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (19, 2, u\"optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (20, 2, u\"optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128')]"
+      ]
+     },
+     "execution_count": 47,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [\n",
+    "                                                 {'optimizer': ['SGD']}, \n",
+    "                                                 {'optimizer': ['SGD'], 'lr': [0.0001, 0.001], 'momentum': [0.95]}, \n",
+    "                                                 {'optimizer': ['Adam'], 'lr': [0.01, 0.1], 'decay': [1e-4]}], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'grid'               -- search_type \n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"random_search\"></a>\n",
+    "## 3b. Random search\n",
+    "\n",
+    "The output table for random search contains the specified number of model architectures, compile and fit parameters, sampled from the specified distributions."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 48,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "20 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.0347167931002948,decay=4.746966178774611e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01062006045632861,decay=1.1876016717166215e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0006995070125407458,momentum=0.9844790514730665)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.07439975848075757,decay=1.7976337634506005e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.09030450672567254,decay=1.340890767690431e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01357387578284614,decay=2.3014993523846666e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.00010336714004241796,momentum=0.9711372680116186)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.00011116485234161093,momentum=0.9664752194346332)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0003071392825766392,momentum=0.9697893478568044)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.03540256307419597,decay=2.7490870549984347e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00026429087119428287,momentum=0.9702132562449013)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.04882317737663686,decay=8.006807036282709e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0005040379351745158,momentum=0.9863934944304705)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00037668508410008814,momentum=0.978821521218891)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.016426175651771575,decay=1.6439282808391488e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00046988338854109496,momentum=0.988290883937812)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0005402557401986037,momentum=0.9795021324622476)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.012596752275640428,decay=1.2801865417619381e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01055415187064375,decay=7.646989120220466e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00014021314734214438,momentum=0.9663397507032889)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 2, u\"optimizer='Adam(lr=0.0347167931002948,decay=4.746966178774611e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.01062006045632861,decay=1.1876016717166215e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.0006995070125407458,momentum=0.9844790514730665)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.07439975848075757,decay=1.7976337634506005e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (5, 2, u\"optimizer='Adam(lr=0.09030450672567254,decay=1.340890767690431e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01357387578284614,decay=2.3014993523846666e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (7, 2, u\"optimizer='SGD(lr=0.00010336714004241796,momentum=0.9711372680116186)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 2, u\"optimizer='SGD(lr=0.00011116485234161093,momentum=0.9664752194346332)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='SGD(lr=0.0003071392825766392,momentum=0.9697893478568044)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 1, u\"optimizer='Adam(lr=0.03540256307419597,decay=2.7490870549984347e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (11, 1, u\"optimizer='SGD(lr=0.00026429087119428287,momentum=0.9702132562449013)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 1, u\"optimizer='Adam(lr=0.04882317737663686,decay=8.006807036282709e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (13, 2, u\"optimizer='SGD(lr=0.0005040379351745158,momentum=0.9863934944304705)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (14, 1, u\"optimizer='SGD(lr=0.00037668508410008814,momentum=0.978821521218891)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (15, 1, u\"optimizer='Adam(lr=0.016426175651771575,decay=1.6439282808391488e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (16, 1, u\"optimizer='SGD(lr=0.00046988338854109496,momentum=0.988290883937812)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (17, 1, u\"optimizer='SGD(lr=0.0005402557401986037,momentum=0.9795021324622476)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (18, 2, u\"optimizer='Adam(lr=0.012596752275640428,decay=1.2801865417619381e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (19, 2, u\"optimizer='Adam(lr=0.01055415187064375,decay=7.646989120220466e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (20, 1, u\"optimizer='SGD(lr=0.00014021314734214438,momentum=0.9663397507032889)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64')]"
+      ]
+     },
+     "execution_count": 48,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [ \n",
+    "                                                 {'optimizer': ['SGD'], 'lr': [0.0001, 0.001, 'log'], 'momentum': [0.95, 0.99, 'log_near_one']}, \n",
+    "                                                 {'optimizer': ['Adam'], 'lr': [0.01, 0.1, 'log'], 'decay': [1e-6, 1e-4, 'log']}], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'random',            -- search_type\n",
+    "                                         20                   -- num_configs\n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"incremental_load\"></a>\n",
+    "# 3c.  Incremental loading for more complex combinations\n",
+    "\n",
+    "If it is easier to generate the model configurations incrementally rather than all at once, you can do that by not dropping the model selection table and associated summary table, in which case the new model configurations will be appended to the existing table.  Here we combine 2 of the previous examples in to a single output table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 49,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "16 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128')]"
+      ]
+     },
+     "execution_count": 49,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql \n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [ {'optimizer': ['Adam', 'SGD'], 'lr': [0.001, 0.01]} ], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'grid'               -- search_type \n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Now add to the existing table and note that mst_key continues where it left off:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 50,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "36 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00020031615564004395,momentum=0.9724038009180801)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.07529364006470769,decay=1.463102386655202e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.017537612203171578,decay=9.268965340542783e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.02436723652830891,decay=2.7036693659868636e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>21</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0009162225178908051,momentum=0.9636373679078051)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>22</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00020973934011486018,momentum=0.9810505351311615)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00011669504881554843,momentum=0.9563917160422619)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.019887949889421844,decay=1.3512689688436213e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>25</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.06844958467546351,decay=1.0949453143707621e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.0279719469411538,decay=3.116565475127251e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0005340915863494089,momentum=0.9846555995292319)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00037236518835129966,momentum=0.9750593509631483)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001703149580002491,momentum=0.9516827304557754)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0003205092574897573,momentum=0.9745610627224451)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.000638802198775629,momentum=0.9896674744988915)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0007145264848827797,momentum=0.9859303213231139)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.053811310244363884,decay=5.1052295876998844e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.0617217468673046,decay=2.0871014466512653e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>35</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.013012045649000626,decay=5.7173240691732966e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>36</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.000575290475361327,momentum=0.9883738353302843)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (17, 1, u\"optimizer='SGD(lr=0.00020031615564004395,momentum=0.9724038009180801)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (18, 2, u\"optimizer='Adam(lr=0.07529364006470769,decay=1.463102386655202e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (19, 1, u\"optimizer='Adam(lr=0.017537612203171578,decay=9.268965340542783e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (20, 1, u\"optimizer='Adam(lr=0.02436723652830891,decay=2.7036693659868636e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (21, 2, u\"optimizer='SGD(lr=0.0009162225178908051,momentum=0.9636373679078051)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (22, 1, u\"optimizer='SGD(lr=0.00020973934011486018,momentum=0.9810505351311615)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (23, 1, u\"optimizer='SGD(lr=0.00011669504881554843,momentum=0.9563917160422619)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (24, 1, u\"optimizer='Adam(lr=0.019887949889421844,decay=1.3512689688436213e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (25, 2, u\"optimizer='Adam(lr=0.06844958467546351,decay=1.0949453143707621e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (26, 1, u\"optimizer='Adam(lr=0.0279719469411538,decay=3.116565475127251e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (27, 1, u\"optimizer='SGD(lr=0.0005340915863494089,momentum=0.9846555995292319)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (28, 1, u\"optimizer='SGD(lr=0.00037236518835129966,momentum=0.9750593509631483)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (29, 1, u\"optimizer='SGD(lr=0.0001703149580002491,momentum=0.9516827304557754)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (30, 2, u\"optimizer='SGD(lr=0.0003205092574897573,momentum=0.9745610627224451)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (31, 2, u\"optimizer='SGD(lr=0.000638802198775629,momentum=0.9896674744988915)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (32, 2, u\"optimizer='SGD(lr=0.0007145264848827797,momentum=0.9859303213231139)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (33, 2, u\"optimizer='Adam(lr=0.053811310244363884,decay=5.1052295876998844e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (34, 1, u\"optimizer='Adam(lr=0.0617217468673046,decay=2.0871014466512653e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (35, 1, u\"optimizer='Adam(lr=0.013012045649000626,decay=5.7173240691732966e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (36, 1, u\"optimizer='SGD(lr=0.000575290475361327,momentum=0.9883738353302843)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64')]"
+      ]
+     },
+     "execution_count": 50,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [ \n",
+    "                                                 {'optimizer': ['SGD'], 'lr': [0.0001, 0.001, 'log'], 'momentum': [0.95, 0.99, 'log_near_one']}, \n",
+    "                                                 {'optimizer': ['Adam'], 'lr': [0.01, 0.1, 'log'], 'decay': [1e-6, 1e-4, 'log']}], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'random',            -- search_type\n",
+    "                                         20                   -- num_configs\n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_model_selection_manual\"></a>\n",
+    "# 4.  Create model selection table manually\n",
+    "\n",
+    "If you want more control over the content of the model selection table, you could use grid or random search to generate a large number of combinations, then SELECT a subset of rows for training.\n",
+    "\n",
+    "Alternatively, you could manually create the model selection table and the associated summary table.  Both must be created since they are needed by the multiple model fit module.\n",
+    "\n",
+    "For example, let's say we don't want all combinations but only want batch_size=4 for model_id=1 and batch_size=8 for model_id=2:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 51,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "6 rows affected.\n",
+      "6 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_arch_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (3, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (4, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (5, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (6, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1')]"
+      ]
+     },
+     "execution_count": 51,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table_manual;\n",
+    "\n",
+    "CREATE TABLE mst_table_manual(\n",
+    "    mst_key serial,\n",
+    "    model_arch_id integer,\n",
+    "    compile_params varchar,\n",
+    "    fit_params varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO mst_table_manual(model_arch_id, compile_params, fit_params) VALUES\n",
+    "(1, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$, 'batch_size=4,epochs=1'),\n",
+    "(1, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']$$, 'batch_size=4,epochs=1'),\n",
+    "(1, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$, 'batch_size=4,epochs=1'),\n",
+    "(2, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$, 'batch_size=8,epochs=1'),\n",
+    "(2, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']$$, 'batch_size=8,epochs=1'),\n",
+    "(2, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$, 'batch_size=8,epochs=1');\n",
+    "\n",
+    "SELECT * FROM mst_table_manual ORDER BY mst_key; "
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Create the summary table which must be named with the model selection output table appended by \"_summary\":"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 52,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_arch_table</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>model_arch_library</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'model_arch_library',)]"
+      ]
+     },
+     "execution_count": 52,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table_manual_summary;\n",
+    "\n",
+    "CREATE TABLE mst_table_manual_summary (\n",
+    "    model_arch_table varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO mst_table_manual_summary(model_arch_table) VALUES\n",
+    "('model_arch_library');\n",
+    "\n",
+    "SELECT * FROM mst_table_manual_summary; "
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"custom\"></a>\n",
+    "# 5. Custom loss functions and custom metrics\n",
+    "\n",
+    "Define custom functions using the utility \"Define Custom Functions\". Psycopg is a PostgreSQL database adapter for the Python programming language. Note need to use the psycopg2.Binary() method to pass as bytes."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 53,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# import database connector psycopg2 and create connection cursor\n",
+    "import psycopg2 as p2\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
+    "cur = conn.cursor()\n",
+    "\n",
+    "# import Dill and define functions\n",
+    "import dill\n",
+    "\n",
+    "# custom loss\n",
+    "def squared_error(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.square(y_pred - y_true)\n",
+    "pb_squared_error=dill.dumps(squared_error)\n",
+    "\n",
+    "# custom metric\n",
+    "def rmse(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.sqrt(K.mean(K.square(y_pred - y_true), axis=-1))\n",
+    "pb_rmse=dill.dumps(rmse)\n",
+    "\n",
+    "# call load function\n",
+    "cur.execute(\"DROP TABLE IF EXISTS madlib.custom_function_table\")\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'squared_error', 'squared error')\", [p2.Binary(pb_squared_error)])\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'rmse', 'root mean square error')\", [p2.Binary(pb_rmse)])\n",
+    "conn.commit()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 54,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "16 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 1, u\"optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 2, u\"optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 2, u\"optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 2, u\"optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 2, u\"optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128')]"
+      ]
+     },
+     "execution_count": 54,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['squared_error'],\n",
+    "                                             'optimizer_params_list': [ {'optimizer': ['Adam', 'SGD'], 'lr': [0.001, 0.01]} ],\n",
+    "                                             'metrics': ['rmse']}\n",
+    "                                         $$,                  -- compile_param_grid\n",
+    "                                         $$\n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10]\n",
+    "                                         }\n",
+    "                                         $$,                  -- fit_param_grid\n",
+    "                                         'grid',              -- search_type\n",
+    "                                         NULL,                -- num_configs\n",
+    "                                         NULL,                -- random_state\n",
+    "                                         'custom_function_table'  -- table with custom functions\n",
+    "                                         );\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_model_selection\"></a>\n",
+    "# 6.  Load model selection table [deprecated]\n",
+    "\n",
+    "#### This method is deprecated and replaced by generate_model_configs() method described above.\n",
+    "\n",
+    "Select the model(s) from the model architecture table that you want to run, along with the compile and fit parameters.  Unique combinations will be created for the set of model selection parameters."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 55,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (3, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (4, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (5, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (6, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (7, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (8, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (9, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (10, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (11, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (12, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1')]"
+      ]
+     },
+     "execution_count": 55,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.load_model_selection_table('model_arch_library', -- model architecture table\n",
+    "                                         'mst_table',          -- model selection table output\n",
+    "                                          ARRAY[1,2],              -- model ids from model architecture table\n",
+    "                                          ARRAY[                   -- compile params\n",
+    "                                              $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$,\n",
+    "                                              $$loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']$$,\n",
+    "                                              $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$\n",
+    "                                          ],\n",
+    "                                          ARRAY[                    -- fit params\n",
+    "                                              $$batch_size=4,epochs=1$$,\n",
+    "                                              $$batch_size=8,epochs=1$$\n",
+    "                                          ]\n",
+    "                                         );\n",
+    "                                  \n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Load-model-architecture-v2.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/Load-model-architecture-v2-checkpoint.ipynb
similarity index 68%
rename from community-artifacts/Deep-learning/Load-model-architecture-v2.ipynb
rename to community-artifacts/Deep-learning/.ipynb_checkpoints/Load-model-architecture-v2-checkpoint.ipynb
index 8aa3716..b823f09 100644
--- a/community-artifacts/Deep-learning/Load-model-architecture-v2.ipynb
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/Load-model-architecture-v2-checkpoint.ipynb
@@ -4,10 +4,16 @@
    "cell_type": "markdown",
    "metadata": {},
    "source": [
-    "# Load model architecture\n",
-    "This utility function loads model architectures and weights into a table for use by deep learning algorithms in Keras.  \n",
+    "# Define model architecture\n",
+    "This function loads model architectures and weights into a table for use by deep learning algorithms.\n",
     "\n",
-    "The model architecture loader was added in MADlib 1.16.\n",
+    "Model architecture is in JSON form and model weights are in the form of PostgreSQL binary data types (bytea). If the output table already exists, a new row is inserted into the table so it can act as a repository for multiple model architectures and weights.\n",
+    "\n",
+    "There is also a function to delete a model from the table.\n",
+    "\n",
+    "MADlib's deep learning methods are designed to use the TensorFlow package and its built in Keras functions. To ensure consistency, please use tensorflow.keras objects (models, layers, etc.) instead of importing Keras and using its objects.\n",
+    "\n",
+    "The model architecture loader was added in MADlib 1.16 and updated after that.\n",
     "\n",
     "## Table of contents\n",
     "\n",
@@ -25,17 +31,15 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 1,
+   "execution_count": 34,
    "metadata": {},
    "outputs": [
     {
-     "name": "stderr",
+     "name": "stdout",
      "output_type": "stream",
      "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
+      "The sql extension is already loaded. To reload it, use:\n",
+      "  %reload_ext sql\n"
      ]
     }
    ],
@@ -45,24 +49,10 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 5,
+   "execution_count": 35,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 5,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
     "# Greenplum Database 5.x on GCP - via tunnel\n",
     "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
@@ -72,7 +62,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
+   "execution_count": 36,
    "metadata": {},
    "outputs": [
     {
@@ -90,15 +80,15 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
-     "execution_count": 6,
+     "execution_count": 36,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -120,28 +110,13 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 37,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "Using TensorFlow backend.\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Couldn't import dot_parser, loading of dot files will not be possible.\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
-    "import keras\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense"
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
    ]
   },
   {
@@ -153,21 +128,22 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 8,
+   "execution_count": 38,
    "metadata": {},
    "outputs": [
     {
      "name": "stdout",
      "output_type": "stream",
      "text": [
+      "Model: \"sequential_3\"\n",
       "_________________________________________________________________\n",
       "Layer (type)                 Output Shape              Param #   \n",
       "=================================================================\n",
-      "dense_1 (Dense)              (None, 10)                50        \n",
+      "dense_9 (Dense)              (None, 10)                50        \n",
       "_________________________________________________________________\n",
-      "dense_2 (Dense)              (None, 10)                110       \n",
+      "dense_10 (Dense)             (None, 10)                110       \n",
       "_________________________________________________________________\n",
-      "dense_3 (Dense)              (None, 3)                 33        \n",
+      "dense_11 (Dense)             (None, 3)                 33        \n",
       "=================================================================\n",
       "Total params: 193\n",
       "Trainable params: 193\n",
@@ -182,21 +158,21 @@
     "model.add(Dense(10, activation='relu'))\n",
     "model.add(Dense(3, activation='softmax'))\n",
     "    \n",
-    "model.summary()"
+    "model.summary();"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 9,
+   "execution_count": 39,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_9\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_10\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_11\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_3\"}, \"backend\": \"tensorflow\"}'"
       ]
      },
-     "execution_count": 9,
+     "execution_count": 39,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -225,7 +201,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 10,
+   "execution_count": 40,
    "metadata": {},
    "outputs": [
     {
@@ -255,15 +231,15 @@
        "        <td>None</td>\n",
        "        <td>Sophie</td>\n",
        "        <td>A simple model</td>\n",
-       "        <td>__madlib_temp_19839392_1576692433_56744839__</td>\n",
+       "        <td>__madlib_temp_27065614_1614901189_16021319__</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_19839392_1576692433_56744839__')]"
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_27065614_1614901189_16021319__')]"
       ]
      },
-     "execution_count": 10,
+     "execution_count": 40,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -294,7 +270,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 11,
+   "execution_count": 41,
    "metadata": {},
    "outputs": [
     {
@@ -323,7 +299,7 @@
        "        <td>None</td>\n",
        "        <td>Maria</td>\n",
        "        <td>Also a simple model</td>\n",
-       "        <td>__madlib_temp_36064316_1576692433_8110861__</td>\n",
+       "        <td>__madlib_temp_87665369_1614901189_11144097__</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
@@ -331,16 +307,16 @@
        "        <td>None</td>\n",
        "        <td>Sophie</td>\n",
        "        <td>A simple model</td>\n",
-       "        <td>__madlib_temp_19839392_1576692433_56744839__</td>\n",
+       "        <td>__madlib_temp_27065614_1614901189_16021319__</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'Also a simple model', u'__madlib_temp_36064316_1576692433_8110861__'),\n",
-       " (1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_19839392_1576692433_56744839__')]"
+       "[(2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'Also a simple model', u'__madlib_temp_87665369_1614901189_11144097__'),\n",
+       " (1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_27065614_1614901189_16021319__')]"
       ]
      },
-     "execution_count": 11,
+     "execution_count": 41,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -376,7 +352,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 12,
+   "execution_count": 42,
    "metadata": {},
    "outputs": [
     {
@@ -384,7 +360,7 @@
      "output_type": "stream",
      "text": [
       "1 rows affected.\n",
-      "1 rows affected.\n"
+      "2 rows affected.\n"
      ]
     },
     {
@@ -392,18 +368,31 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>count</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "        <td>False</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1L,)]"
+       "[(1, u'Sophie', u'A simple model', False),\n",
+       " (2, u'Maria', u'Also a simple model', True)]"
       ]
      },
-     "execution_count": 12,
+     "execution_count": 42,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -416,7 +405,7 @@
     "WHERE model_arch_library.model_id = 2;\n",
     "\n",
     "-- Check weights loaded OK\n",
-    "SELECT COUNT(*) FROM model_arch_library WHERE model_weights IS NOT NULL;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   },
   {
@@ -430,7 +419,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 13,
+   "execution_count": 43,
    "metadata": {},
    "outputs": [
     {
@@ -438,7 +427,6 @@
      "output_type": "stream",
      "text": [
       "Done.\n",
-      "1 rows affected.\n",
       "1 rows affected.\n"
      ]
     },
@@ -447,18 +435,18 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>count</th>\n",
+       "        <th>load_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>2</td>\n",
+       "        <td></td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(2L,)]"
+       "[('',)]"
       ]
      },
-     "execution_count": 13,
+     "execution_count": 43,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -467,8 +455,8 @@
     "%%sql\n",
     "CREATE OR REPLACE FUNCTION load_weights() RETURNS VOID AS\n",
     "$$\n",
-    "from keras.layers import *\n",
-    "from keras import Sequential\n",
+    "from tensorflow.keras.layers import *\n",
+    "from tensorflow.keras import Sequential\n",
     "import numpy as np\n",
     "import plpy\n",
     "\n",
@@ -493,15 +481,12 @@
     "$$ language plpythonu;\n",
     "\n",
     "-- Call load function\n",
-    "SELECT load_weights();\n",
-    "\n",
-    "-- Check weights loaded OK\n",
-    "SELECT COUNT(*) FROM model_arch_library WHERE model_weights IS NOT NULL;"
+    "SELECT load_weights();"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 14,
+   "execution_count": 44,
    "metadata": {},
    "outputs": [
     {
@@ -518,33 +503,43 @@
        "    <tr>\n",
        "        <th>model_id</th>\n",
        "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
        "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "        <td>False</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>3</td>\n",
        "        <td>Ella</td>\n",
+       "        <td>Model x</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, u'Sophie'), (2, u'Maria'), (3, u'Ella')]"
+       "[(1, u'Sophie', u'A simple model', False),\n",
+       " (2, u'Maria', u'Also a simple model', True),\n",
+       " (3, u'Ella', u'Model x', True)]"
       ]
      },
-     "execution_count": 14,
+     "execution_count": 44,
      "metadata": {},
      "output_type": "execute_result"
     }
    ],
    "source": [
     "%%sql\n",
-    "SELECT model_id, name from model_arch_library ORDER BY model_id;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   },
   {
@@ -560,45 +555,16 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 15,
+   "execution_count": 45,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(2L,)]"
-      ]
-     },
-     "execution_count": 15,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
     "import psycopg2 as p2\n",
-    "#conn = p2.connect('postgresql://gpadmin@35.239.240.26:5432/madlib')\n",
-    "conn = p2.connect('postgresql://fmcquillan@localhost:5432/madlib')\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
     "cur = conn.cursor()\n",
     "\n",
-    "from keras.layers import *\n",
-    "from keras import Sequential\n",
+    "from tensorflow.keras.layers import *\n",
+    "from tensorflow.keras import Sequential\n",
     "import numpy as np\n",
     "\n",
     "# create model\n",
@@ -615,22 +581,19 @@
     "\n",
     "query = \"SELECT madlib.load_keras_model('model_arch_library', %s,%s,%s,%s)\"\n",
     "cur.execute(query,[model.to_json(), weights_bytea, \"Grace\", \"Model y\"])\n",
-    "conn.commit()\n",
-    "\n",
-    "# check weights loaded OK\n",
-    "%sql SELECT COUNT(*) FROM model_arch_library WHERE model_weights IS NOT NULL;"
+    "conn.commit()"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 16,
+   "execution_count": 46,
    "metadata": {},
    "outputs": [
     {
      "name": "stdout",
      "output_type": "stream",
      "text": [
-      "3 rows affected.\n"
+      "4 rows affected.\n"
      ]
     },
     {
@@ -640,33 +603,50 @@
        "    <tr>\n",
        "        <th>model_id</th>\n",
        "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
        "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "        <td>False</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>3</td>\n",
        "        <td>Ella</td>\n",
+       "        <td>Model x</td>\n",
+       "        <td>True</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>Grace</td>\n",
+       "        <td>Model y</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, u'Sophie'), (2, u'Maria'), (3, u'Ella')]"
+       "[(1, u'Sophie', u'A simple model', False),\n",
+       " (2, u'Maria', u'Also a simple model', True),\n",
+       " (3, u'Ella', u'Model x', True),\n",
+       " (4, u'Grace', u'Model y', True)]"
       ]
      },
-     "execution_count": 16,
+     "execution_count": 46,
      "metadata": {},
      "output_type": "execute_result"
     }
    ],
    "source": [
     "%%sql\n",
-    "SELECT model_id, name from model_arch_library ORDER BY model_id;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   },
   {
@@ -679,7 +659,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 17,
+   "execution_count": 47,
    "metadata": {},
    "outputs": [
     {
@@ -687,7 +667,7 @@
      "output_type": "stream",
      "text": [
       "1 rows affected.\n",
-      "2 rows affected.\n"
+      "3 rows affected.\n"
      ]
     },
     {
@@ -697,22 +677,36 @@
        "    <tr>\n",
        "        <th>model_id</th>\n",
        "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>3</td>\n",
        "        <td>Ella</td>\n",
+       "        <td>Model x</td>\n",
+       "        <td>True</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>Grace</td>\n",
+       "        <td>Model y</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(2, u'Maria'), (3, u'Ella')]"
+       "[(2, u'Maria', u'Also a simple model', True),\n",
+       " (3, u'Ella', u'Model x', True),\n",
+       " (4, u'Grace', u'Model y', True)]"
       ]
      },
-     "execution_count": 17,
+     "execution_count": 47,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -722,7 +716,7 @@
     "SELECT madlib.delete_keras_model('model_arch_library',   -- Output table\n",
     "                                  1                      -- Model id\n",
     "                                );\n",
-    "SELECT model_id, name from model_arch_library ORDER BY model_id;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   }
  ],
@@ -742,7 +736,7 @@
    "name": "python",
    "nbconvert_exporter": "python",
    "pygments_lexer": "ipython2",
-   "version": "2.7.10"
+   "version": "2.7.16"
   }
  },
  "nbformat": 4,
diff --git a/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-MLP-v2-checkpoint.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-MLP-v2-checkpoint.ipynb
new file mode 100644
index 0000000..cf99035
--- /dev/null
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-MLP-v2-checkpoint.ipynb
@@ -0,0 +1,5034 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Multilayer Perceptron Using Keras and MADlib\n",
+    "\n",
+    "E2E classification example using MADlib calling a Keras MLP.\n",
+    "\n",
+    "Deep learning works best on very large datasets, but that is not convenient for a quick introduction to the syntax.  So in this workbook we use the well known iris data set from https://archive.ics.uci.edu/ml/datasets/iris to help get you started.  It is similar to the example in user docs http://madlib.apache.org/docs/latest/index.html\n",
+    "\n",
+    "For more realistic examples with images please refer to the deep learning notebooks at\n",
+    "https://github.com/apache/madlib-site/tree/asf-site/community-artifacts\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#class\">Classification</a>\n",
+    "\n",
+    "* <a href=\"#create_input_data\">1. Create input data</a>\n",
+    "\n",
+    "* <a href=\"#pp\">2. Call preprocessor for deep learning</a>\n",
+    "\n",
+    "* <a href=\"#load\">3. Define and load model architecture</a>\n",
+    "\n",
+    "* <a href=\"#train\">4. Train</a>\n",
+    "\n",
+    "* <a href=\"#eval\">5. Evaluate</a>\n",
+    "\n",
+    "* <a href=\"#pred\">6. Predict</a>\n",
+    "\n",
+    "* <a href=\"#pred_byom\">7. Predict BYOM</a>\n",
+    "\n",
+    "<a href=\"#class2\">Classification with Other Parameters</a>\n",
+    "\n",
+    "* <a href=\"#val_dataset\">1. Validation dataset</a>\n",
+    "\n",
+    "* <a href=\"#pred_prob\">2. Predict probabilities</a>\n",
+    "\n",
+    "* <a href=\"#warm_start\">3. Warm start</a>\n",
+    "\n",
+    "<a href=\"#transfer_learn\">Transfer learning</a>\n",
+    "\n",
+    "* <a href=\"#load2\">1. Define and load model architecture with some layers frozen</a>\n",
+    "\n",
+    "* <a href=\"#train2\">2. Train transfer model</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 60,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "The sql extension is already loaded. To reload it, use:\n",
+      "  %reload_ext sql\n"
+     ]
+    }
+   ],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 61,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
+    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
+    "\n",
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 62,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 62,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class\"></a>\n",
+    "# Classification\n",
+    "\n",
+    "<a id=\"create_input_data\"></a>\n",
+    "# 1.  Create input data\n",
+    "\n",
+    "Load iris data set."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 63,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "150 rows affected.\n",
+      "150 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>attributes</th>\n",
+       "        <th>class_text</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>[Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>21</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>22</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>25</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>35</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>36</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>37</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>39</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>40</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>42</td>\n",
+       "        <td>[Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>44</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>45</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>46</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>47</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>48</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>[Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>50</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>[Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>53</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>58</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>59</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>62</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>65</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>69</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>79</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>86</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>87</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>88</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>91</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>95</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>97</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>[Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>105</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>106</td>\n",
+       "        <td>[Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>[Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>109</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>110</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>118</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>124</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>129</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>130</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>[Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>[Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>133</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>139</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>140</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>141</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>142</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (2, [Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (3, [Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (4, [Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (5, [Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (6, [Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (7, [Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (8, [Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (9, [Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (10, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (11, [Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (12, [Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (13, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (14, [Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (15, [Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (16, [Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (17, [Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (18, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (19, [Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (20, [Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (21, [Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (22, [Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (23, [Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (24, [Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')], u'Iris-setosa'),\n",
+       " (25, [Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (26, [Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (27, [Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (28, [Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (29, [Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (30, [Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (31, [Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (32, [Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (33, [Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (34, [Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (35, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (36, [Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (37, [Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (38, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (39, [Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (40, [Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (41, [Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (42, [Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (43, [Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (44, [Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')], u'Iris-setosa'),\n",
+       " (45, [Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (46, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (47, [Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (48, [Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (49, [Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (50, [Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (51, [Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (52, [Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (53, [Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (54, [Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (55, [Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (56, [Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (57, [Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (58, [Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (59, [Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (60, [Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (61, [Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (62, [Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (63, [Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (64, [Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (65, [Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (66, [Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (67, [Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (68, [Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (69, [Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (70, [Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (71, [Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')], u'Iris-versicolor'),\n",
+       " (72, [Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (73, [Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (74, [Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (75, [Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (76, [Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (77, [Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (78, [Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')], u'Iris-versicolor'),\n",
+       " (79, [Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (80, [Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (81, [Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (82, [Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (83, [Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (84, [Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (85, [Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (86, [Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (87, [Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (88, [Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (89, [Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (90, [Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (91, [Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (92, [Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (93, [Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (94, [Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (95, [Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (96, [Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (97, [Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (98, [Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (99, [Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (100, [Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (101, [Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (102, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (103, [Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (104, [Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (105, [Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (106, [Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (107, [Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')], u'Iris-virginica'),\n",
+       " (108, [Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (109, [Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (110, [Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (111, [Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (112, [Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (113, [Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (114, [Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (115, [Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (116, [Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (117, [Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (118, [Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (119, [Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (120, [Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (121, [Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (122, [Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (123, [Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (124, [Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (125, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (126, [Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (127, [Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (128, [Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (129, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (130, [Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')], u'Iris-virginica'),\n",
+       " (131, [Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (132, [Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (133, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (134, [Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (135, [Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')], u'Iris-virginica'),\n",
+       " (136, [Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (137, [Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (138, [Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (139, [Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (140, [Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (141, [Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (142, [Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (143, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (144, [Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (145, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (146, [Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (147, [Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (148, [Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (149, [Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (150, [Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')], u'Iris-virginica')]"
+      ]
+     },
+     "execution_count": 63,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql \n",
+    "DROP TABLE IF EXISTS iris_data;\n",
+    "\n",
+    "CREATE TABLE iris_data(\n",
+    "    id serial,\n",
+    "    attributes numeric[],\n",
+    "    class_text varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO iris_data(id, attributes, class_text) VALUES\n",
+    "(1,ARRAY[5.1,3.5,1.4,0.2],'Iris-setosa'),\n",
+    "(2,ARRAY[4.9,3.0,1.4,0.2],'Iris-setosa'),\n",
+    "(3,ARRAY[4.7,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(4,ARRAY[4.6,3.1,1.5,0.2],'Iris-setosa'),\n",
+    "(5,ARRAY[5.0,3.6,1.4,0.2],'Iris-setosa'),\n",
+    "(6,ARRAY[5.4,3.9,1.7,0.4],'Iris-setosa'),\n",
+    "(7,ARRAY[4.6,3.4,1.4,0.3],'Iris-setosa'),\n",
+    "(8,ARRAY[5.0,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(9,ARRAY[4.4,2.9,1.4,0.2],'Iris-setosa'),\n",
+    "(10,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(11,ARRAY[5.4,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(12,ARRAY[4.8,3.4,1.6,0.2],'Iris-setosa'),\n",
+    "(13,ARRAY[4.8,3.0,1.4,0.1],'Iris-setosa'),\n",
+    "(14,ARRAY[4.3,3.0,1.1,0.1],'Iris-setosa'),\n",
+    "(15,ARRAY[5.8,4.0,1.2,0.2],'Iris-setosa'),\n",
+    "(16,ARRAY[5.7,4.4,1.5,0.4],'Iris-setosa'),\n",
+    "(17,ARRAY[5.4,3.9,1.3,0.4],'Iris-setosa'),\n",
+    "(18,ARRAY[5.1,3.5,1.4,0.3],'Iris-setosa'),\n",
+    "(19,ARRAY[5.7,3.8,1.7,0.3],'Iris-setosa'),\n",
+    "(20,ARRAY[5.1,3.8,1.5,0.3],'Iris-setosa'),\n",
+    "(21,ARRAY[5.4,3.4,1.7,0.2],'Iris-setosa'),\n",
+    "(22,ARRAY[5.1,3.7,1.5,0.4],'Iris-setosa'),\n",
+    "(23,ARRAY[4.6,3.6,1.0,0.2],'Iris-setosa'),\n",
+    "(24,ARRAY[5.1,3.3,1.7,0.5],'Iris-setosa'),\n",
+    "(25,ARRAY[4.8,3.4,1.9,0.2],'Iris-setosa'),\n",
+    "(26,ARRAY[5.0,3.0,1.6,0.2],'Iris-setosa'),\n",
+    "(27,ARRAY[5.0,3.4,1.6,0.4],'Iris-setosa'),\n",
+    "(28,ARRAY[5.2,3.5,1.5,0.2],'Iris-setosa'),\n",
+    "(29,ARRAY[5.2,3.4,1.4,0.2],'Iris-setosa'),\n",
+    "(30,ARRAY[4.7,3.2,1.6,0.2],'Iris-setosa'),\n",
+    "(31,ARRAY[4.8,3.1,1.6,0.2],'Iris-setosa'),\n",
+    "(32,ARRAY[5.4,3.4,1.5,0.4],'Iris-setosa'),\n",
+    "(33,ARRAY[5.2,4.1,1.5,0.1],'Iris-setosa'),\n",
+    "(34,ARRAY[5.5,4.2,1.4,0.2],'Iris-setosa'),\n",
+    "(35,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(36,ARRAY[5.0,3.2,1.2,0.2],'Iris-setosa'),\n",
+    "(37,ARRAY[5.5,3.5,1.3,0.2],'Iris-setosa'),\n",
+    "(38,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(39,ARRAY[4.4,3.0,1.3,0.2],'Iris-setosa'),\n",
+    "(40,ARRAY[5.1,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(41,ARRAY[5.0,3.5,1.3,0.3],'Iris-setosa'),\n",
+    "(42,ARRAY[4.5,2.3,1.3,0.3],'Iris-setosa'),\n",
+    "(43,ARRAY[4.4,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(44,ARRAY[5.0,3.5,1.6,0.6],'Iris-setosa'),\n",
+    "(45,ARRAY[5.1,3.8,1.9,0.4],'Iris-setosa'),\n",
+    "(46,ARRAY[4.8,3.0,1.4,0.3],'Iris-setosa'),\n",
+    "(47,ARRAY[5.1,3.8,1.6,0.2],'Iris-setosa'),\n",
+    "(48,ARRAY[4.6,3.2,1.4,0.2],'Iris-setosa'),\n",
+    "(49,ARRAY[5.3,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(50,ARRAY[5.0,3.3,1.4,0.2],'Iris-setosa'),\n",
+    "(51,ARRAY[7.0,3.2,4.7,1.4],'Iris-versicolor'),\n",
+    "(52,ARRAY[6.4,3.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(53,ARRAY[6.9,3.1,4.9,1.5],'Iris-versicolor'),\n",
+    "(54,ARRAY[5.5,2.3,4.0,1.3],'Iris-versicolor'),\n",
+    "(55,ARRAY[6.5,2.8,4.6,1.5],'Iris-versicolor'),\n",
+    "(56,ARRAY[5.7,2.8,4.5,1.3],'Iris-versicolor'),\n",
+    "(57,ARRAY[6.3,3.3,4.7,1.6],'Iris-versicolor'),\n",
+    "(58,ARRAY[4.9,2.4,3.3,1.0],'Iris-versicolor'),\n",
+    "(59,ARRAY[6.6,2.9,4.6,1.3],'Iris-versicolor'),\n",
+    "(60,ARRAY[5.2,2.7,3.9,1.4],'Iris-versicolor'),\n",
+    "(61,ARRAY[5.0,2.0,3.5,1.0],'Iris-versicolor'),\n",
+    "(62,ARRAY[5.9,3.0,4.2,1.5],'Iris-versicolor'),\n",
+    "(63,ARRAY[6.0,2.2,4.0,1.0],'Iris-versicolor'),\n",
+    "(64,ARRAY[6.1,2.9,4.7,1.4],'Iris-versicolor'),\n",
+    "(65,ARRAY[5.6,2.9,3.6,1.3],'Iris-versicolor'),\n",
+    "(66,ARRAY[6.7,3.1,4.4,1.4],'Iris-versicolor'),\n",
+    "(67,ARRAY[5.6,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(68,ARRAY[5.8,2.7,4.1,1.0],'Iris-versicolor'),\n",
+    "(69,ARRAY[6.2,2.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(70,ARRAY[5.6,2.5,3.9,1.1],'Iris-versicolor'),\n",
+    "(71,ARRAY[5.9,3.2,4.8,1.8],'Iris-versicolor'),\n",
+    "(72,ARRAY[6.1,2.8,4.0,1.3],'Iris-versicolor'),\n",
+    "(73,ARRAY[6.3,2.5,4.9,1.5],'Iris-versicolor'),\n",
+    "(74,ARRAY[6.1,2.8,4.7,1.2],'Iris-versicolor'),\n",
+    "(75,ARRAY[6.4,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(76,ARRAY[6.6,3.0,4.4,1.4],'Iris-versicolor'),\n",
+    "(77,ARRAY[6.8,2.8,4.8,1.4],'Iris-versicolor'),\n",
+    "(78,ARRAY[6.7,3.0,5.0,1.7],'Iris-versicolor'),\n",
+    "(79,ARRAY[6.0,2.9,4.5,1.5],'Iris-versicolor'),\n",
+    "(80,ARRAY[5.7,2.6,3.5,1.0],'Iris-versicolor'),\n",
+    "(81,ARRAY[5.5,2.4,3.8,1.1],'Iris-versicolor'),\n",
+    "(82,ARRAY[5.5,2.4,3.7,1.0],'Iris-versicolor'),\n",
+    "(83,ARRAY[5.8,2.7,3.9,1.2],'Iris-versicolor'),\n",
+    "(84,ARRAY[6.0,2.7,5.1,1.6],'Iris-versicolor'),\n",
+    "(85,ARRAY[5.4,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(86,ARRAY[6.0,3.4,4.5,1.6],'Iris-versicolor'),\n",
+    "(87,ARRAY[6.7,3.1,4.7,1.5],'Iris-versicolor'),\n",
+    "(88,ARRAY[6.3,2.3,4.4,1.3],'Iris-versicolor'),\n",
+    "(89,ARRAY[5.6,3.0,4.1,1.3],'Iris-versicolor'),\n",
+    "(90,ARRAY[5.5,2.5,4.0,1.3],'Iris-versicolor'),\n",
+    "(91,ARRAY[5.5,2.6,4.4,1.2],'Iris-versicolor'),\n",
+    "(92,ARRAY[6.1,3.0,4.6,1.4],'Iris-versicolor'),\n",
+    "(93,ARRAY[5.8,2.6,4.0,1.2],'Iris-versicolor'),\n",
+    "(94,ARRAY[5.0,2.3,3.3,1.0],'Iris-versicolor'),\n",
+    "(95,ARRAY[5.6,2.7,4.2,1.3],'Iris-versicolor'),\n",
+    "(96,ARRAY[5.7,3.0,4.2,1.2],'Iris-versicolor'),\n",
+    "(97,ARRAY[5.7,2.9,4.2,1.3],'Iris-versicolor'),\n",
+    "(98,ARRAY[6.2,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(99,ARRAY[5.1,2.5,3.0,1.1],'Iris-versicolor'),\n",
+    "(100,ARRAY[5.7,2.8,4.1,1.3],'Iris-versicolor'),\n",
+    "(101,ARRAY[6.3,3.3,6.0,2.5],'Iris-virginica'),\n",
+    "(102,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(103,ARRAY[7.1,3.0,5.9,2.1],'Iris-virginica'),\n",
+    "(104,ARRAY[6.3,2.9,5.6,1.8],'Iris-virginica'),\n",
+    "(105,ARRAY[6.5,3.0,5.8,2.2],'Iris-virginica'),\n",
+    "(106,ARRAY[7.6,3.0,6.6,2.1],'Iris-virginica'),\n",
+    "(107,ARRAY[4.9,2.5,4.5,1.7],'Iris-virginica'),\n",
+    "(108,ARRAY[7.3,2.9,6.3,1.8],'Iris-virginica'),\n",
+    "(109,ARRAY[6.7,2.5,5.8,1.8],'Iris-virginica'),\n",
+    "(110,ARRAY[7.2,3.6,6.1,2.5],'Iris-virginica'),\n",
+    "(111,ARRAY[6.5,3.2,5.1,2.0],'Iris-virginica'),\n",
+    "(112,ARRAY[6.4,2.7,5.3,1.9],'Iris-virginica'),\n",
+    "(113,ARRAY[6.8,3.0,5.5,2.1],'Iris-virginica'),\n",
+    "(114,ARRAY[5.7,2.5,5.0,2.0],'Iris-virginica'),\n",
+    "(115,ARRAY[5.8,2.8,5.1,2.4],'Iris-virginica'),\n",
+    "(116,ARRAY[6.4,3.2,5.3,2.3],'Iris-virginica'),\n",
+    "(117,ARRAY[6.5,3.0,5.5,1.8],'Iris-virginica'),\n",
+    "(118,ARRAY[7.7,3.8,6.7,2.2],'Iris-virginica'),\n",
+    "(119,ARRAY[7.7,2.6,6.9,2.3],'Iris-virginica'),\n",
+    "(120,ARRAY[6.0,2.2,5.0,1.5],'Iris-virginica'),\n",
+    "(121,ARRAY[6.9,3.2,5.7,2.3],'Iris-virginica'),\n",
+    "(122,ARRAY[5.6,2.8,4.9,2.0],'Iris-virginica'),\n",
+    "(123,ARRAY[7.7,2.8,6.7,2.0],'Iris-virginica'),\n",
+    "(124,ARRAY[6.3,2.7,4.9,1.8],'Iris-virginica'),\n",
+    "(125,ARRAY[6.7,3.3,5.7,2.1],'Iris-virginica'),\n",
+    "(126,ARRAY[7.2,3.2,6.0,1.8],'Iris-virginica'),\n",
+    "(127,ARRAY[6.2,2.8,4.8,1.8],'Iris-virginica'),\n",
+    "(128,ARRAY[6.1,3.0,4.9,1.8],'Iris-virginica'),\n",
+    "(129,ARRAY[6.4,2.8,5.6,2.1],'Iris-virginica'),\n",
+    "(130,ARRAY[7.2,3.0,5.8,1.6],'Iris-virginica'),\n",
+    "(131,ARRAY[7.4,2.8,6.1,1.9],'Iris-virginica'),\n",
+    "(132,ARRAY[7.9,3.8,6.4,2.0],'Iris-virginica'),\n",
+    "(133,ARRAY[6.4,2.8,5.6,2.2],'Iris-virginica'),\n",
+    "(134,ARRAY[6.3,2.8,5.1,1.5],'Iris-virginica'),\n",
+    "(135,ARRAY[6.1,2.6,5.6,1.4],'Iris-virginica'),\n",
+    "(136,ARRAY[7.7,3.0,6.1,2.3],'Iris-virginica'),\n",
+    "(137,ARRAY[6.3,3.4,5.6,2.4],'Iris-virginica'),\n",
+    "(138,ARRAY[6.4,3.1,5.5,1.8],'Iris-virginica'),\n",
+    "(139,ARRAY[6.0,3.0,4.8,1.8],'Iris-virginica'),\n",
+    "(140,ARRAY[6.9,3.1,5.4,2.1],'Iris-virginica'),\n",
+    "(141,ARRAY[6.7,3.1,5.6,2.4],'Iris-virginica'),\n",
+    "(142,ARRAY[6.9,3.1,5.1,2.3],'Iris-virginica'),\n",
+    "(143,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(144,ARRAY[6.8,3.2,5.9,2.3],'Iris-virginica'),\n",
+    "(145,ARRAY[6.7,3.3,5.7,2.5],'Iris-virginica'),\n",
+    "(146,ARRAY[6.7,3.0,5.2,2.3],'Iris-virginica'),\n",
+    "(147,ARRAY[6.3,2.5,5.0,1.9],'Iris-virginica'),\n",
+    "(148,ARRAY[6.5,3.0,5.2,2.0],'Iris-virginica'),\n",
+    "(149,ARRAY[6.2,3.4,5.4,2.3],'Iris-virginica'),\n",
+    "(150,ARRAY[5.9,3.0,5.1,1.8],'Iris-virginica');\n",
+    "\n",
+    "SELECT * FROM iris_data ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Create a test/validation dataset from the training data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 64,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(120L,)]"
+      ]
+     },
+     "execution_count": 64,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train, iris_test;\n",
+    "\n",
+    "-- Set seed so results are reproducible\n",
+    "SELECT setseed(0);\n",
+    "\n",
+    "SELECT madlib.train_test_split('iris_data',     -- Source table\n",
+    "                               'iris',          -- Output table root name\n",
+    "                                0.8,            -- Train proportion\n",
+    "                                NULL,           -- Test proportion (0.2)\n",
+    "                                NULL,           -- Strata definition\n",
+    "                                NULL,           -- Output all columns\n",
+    "                                NULL,           -- Sample without replacement\n",
+    "                                TRUE            -- Separate output tables\n",
+    "                              );\n",
+    "\n",
+    "SELECT COUNT(*) FROM iris_train;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pp\"></a>\n",
+    "# 2. Call preprocessor for deep learning\n",
+    "Training dataset (uses training preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 65,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train</td>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>60</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train', u'iris_train_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 60, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 65,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train_packed, iris_train_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('iris_train',         -- Source table\n",
+    "                                       'iris_train_packed',  -- Output table\n",
+    "                                       'class_text',         -- Dependent variable\n",
+    "                                       'attributes'          -- Independent variable\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM iris_train_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Validation dataset (uses validation preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 66,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_test</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>15</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_test', u'iris_test_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 15, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 66,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_test_packed, iris_test_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('iris_test',          -- Source table\n",
+    "                                         'iris_test_packed',   -- Output table\n",
+    "                                         'class_text',         -- Dependent variable\n",
+    "                                         'attributes',         -- Independent variable\n",
+    "                                         'iris_train_packed'   -- From training preprocessor step\n",
+    "                                          ); \n",
+    "\n",
+    "SELECT * FROM iris_test_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load\"></a>\n",
+    "# 3. Define and load model architecture\n",
+    "Import Keras libraries"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 67,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 68,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_3\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_9 (Dense)              (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_10 (Dense)             (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_11 (Dense)             (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 193\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model_simple = Sequential()\n",
+    "model_simple.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model_simple.add(Dense(10, activation='relu'))\n",
+    "model_simple.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model_simple.summary()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 69,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_9\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_10\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_11\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_3\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 69,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model_simple.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 70,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Sophie', u'A simple model')]"
+      ]
+     },
+     "execution_count": 70,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS model_arch_library;\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Sophie',              -- Name\n",
+    "                               'A simple model'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT model_id, model_arch, name, description FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train\"></a>\n",
+    "# 4.  Train\n",
+    "Train the model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 71,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 71,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10                    -- num_iterations\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 72,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>1</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>10</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-05 00:32:32.677148</td>\n",
+       "        <td>2021-03-05 00:32:33.888866</td>\n",
+       "        <td>[1.21162915229797]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.891666650772</td>\n",
+       "        <td>0.609960496426</td>\n",
+       "        <td>[0.891666650772095]</td>\n",
+       "        <td>[0.609960496425629]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>[10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, None, None, 10, None, None, u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 5, 0, 32, 32, 677148), datetime.datetime(2021, 3, 5, 0, 32, 33, 888866), [1.21162915229797], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.891666650772095, 0.609960496425629, [0.891666650772095], [0.609960496425629], None, None, None, None, [10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 72,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"eval\"></a>\n",
+    "# 5. Evaluate\n",
+    "\n",
+    "Now run evaluate using model we built above:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 73,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>loss</th>\n",
+       "        <th>metric</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>0.565514206886</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(0.565514206886292, 0.899999976158142, [u'accuracy'], u'categorical_crossentropy')]"
+      ]
+     },
+     "execution_count": 73,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_validate;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_evaluate('iris_model',       -- model\n",
+    "                                   'iris_test_packed',  -- test table\n",
+    "                                   'iris_validate'      -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_validate;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred\"></a>\n",
+    "# 6. Predict\n",
+    "\n",
+    "Now predict using model we built.  We will use the validation data set for prediction as well, which is not usual but serves to show the syntax. The prediction is in the estimated_class_text column:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 74,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "90 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.4811115</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3254119</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.1934767</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.4824369</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.31281063</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.20475245</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.46726868</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.33416426</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.19856708</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.48340678</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.30329248</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.21330076</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.4921991</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.30441415</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.20338683</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.49637538</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.40710366</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.09652098</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.47494373</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.44864976</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.076406464</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.51390177</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.38778767</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.09831052</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.6093597</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.2456077</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.14503266</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.55952024</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3228162</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.11766361</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5187173</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.40794137</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.07334134</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.6168418</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.23869106</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.14446718</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.57109237</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3401038</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.088803805</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.53077435</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.32625726</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.14296837</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.53015316</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.32889763</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.14094917</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5004781</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3183768</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.18114516</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.49173826</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.39376155</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.11450017</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.63378763</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.33676574</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.029446673</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.52301323</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4507337</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.026253074</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8007931</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.17807306</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.021133851</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8288441</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.15422747</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.016928488</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.60298413</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3622377</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.034778137</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.79363465</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.18244599</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.023919372</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6008913</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.38202757</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.01708112</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.51136595</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.455122</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.033512004</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5061915</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4644996</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.029308934</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8607982</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.12758763</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.011614183</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7419237</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.23144698</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.026629237</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.81490934</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.16574617</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.019344518</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6649737</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.29889813</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.03612826</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(6, u'class_text', u'Iris-setosa', 0.4811115, 1),\n",
+       " (6, u'class_text', u'Iris-versicolor', 0.3254119, 2),\n",
+       " (6, u'class_text', u'Iris-virginica', 0.1934767, 3),\n",
+       " (9, u'class_text', u'Iris-setosa', 0.4824369, 1),\n",
+       " (9, u'class_text', u'Iris-versicolor', 0.31281063, 2),\n",
+       " (9, u'class_text', u'Iris-virginica', 0.20475245, 3),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.46726868, 1),\n",
+       " (31, u'class_text', u'Iris-versicolor', 0.33416426, 2),\n",
+       " (31, u'class_text', u'Iris-virginica', 0.19856708, 3),\n",
+       " (32, u'class_text', u'Iris-setosa', 0.48340678, 1),\n",
+       " (32, u'class_text', u'Iris-versicolor', 0.30329248, 2),\n",
+       " (32, u'class_text', u'Iris-virginica', 0.21330076, 3),\n",
+       " (41, u'class_text', u'Iris-setosa', 0.4921991, 1),\n",
+       " (41, u'class_text', u'Iris-versicolor', 0.30441415, 2),\n",
+       " (41, u'class_text', u'Iris-virginica', 0.20338683, 3),\n",
+       " (52, u'class_text', u'Iris-versicolor', 0.49637538, 1),\n",
+       " (52, u'class_text', u'Iris-virginica', 0.40710366, 2),\n",
+       " (52, u'class_text', u'Iris-setosa', 0.09652098, 3),\n",
+       " (57, u'class_text', u'Iris-virginica', 0.47494373, 1),\n",
+       " (57, u'class_text', u'Iris-versicolor', 0.44864976, 2),\n",
+       " (57, u'class_text', u'Iris-setosa', 0.076406464, 3),\n",
+       " (60, u'class_text', u'Iris-virginica', 0.51390177, 1),\n",
+       " (60, u'class_text', u'Iris-versicolor', 0.38778767, 2),\n",
+       " (60, u'class_text', u'Iris-setosa', 0.09831052, 3),\n",
+       " (63, u'class_text', u'Iris-versicolor', 0.6093597, 1),\n",
+       " (63, u'class_text', u'Iris-virginica', 0.2456077, 2),\n",
+       " (63, u'class_text', u'Iris-setosa', 0.14503266, 3),\n",
+       " (66, u'class_text', u'Iris-versicolor', 0.55952024, 1),\n",
+       " (66, u'class_text', u'Iris-virginica', 0.3228162, 2),\n",
+       " (66, u'class_text', u'Iris-setosa', 0.11766361, 3),\n",
+       " (67, u'class_text', u'Iris-virginica', 0.5187173, 1),\n",
+       " (67, u'class_text', u'Iris-versicolor', 0.40794137, 2),\n",
+       " (67, u'class_text', u'Iris-setosa', 0.07334134, 3),\n",
+       " (68, u'class_text', u'Iris-versicolor', 0.6168418, 1),\n",
+       " (68, u'class_text', u'Iris-virginica', 0.23869106, 2),\n",
+       " (68, u'class_text', u'Iris-setosa', 0.14446718, 3),\n",
+       " (77, u'class_text', u'Iris-versicolor', 0.57109237, 1),\n",
+       " (77, u'class_text', u'Iris-virginica', 0.3401038, 2),\n",
+       " (77, u'class_text', u'Iris-setosa', 0.088803805, 3),\n",
+       " (81, u'class_text', u'Iris-versicolor', 0.53077435, 1),\n",
+       " (81, u'class_text', u'Iris-virginica', 0.32625726, 2),\n",
+       " (81, u'class_text', u'Iris-setosa', 0.14296837, 3),\n",
+       " (83, u'class_text', u'Iris-versicolor', 0.53015316, 1),\n",
+       " (83, u'class_text', u'Iris-virginica', 0.32889763, 2),\n",
+       " (83, u'class_text', u'Iris-setosa', 0.14094917, 3),\n",
+       " (94, u'class_text', u'Iris-versicolor', 0.5004781, 1),\n",
+       " (94, u'class_text', u'Iris-virginica', 0.3183768, 2),\n",
+       " (94, u'class_text', u'Iris-setosa', 0.18114516, 3),\n",
+       " (100, u'class_text', u'Iris-versicolor', 0.49173826, 1),\n",
+       " (100, u'class_text', u'Iris-virginica', 0.39376155, 2),\n",
+       " (100, u'class_text', u'Iris-setosa', 0.11450017, 3),\n",
+       " (104, u'class_text', u'Iris-virginica', 0.63378763, 1),\n",
+       " (104, u'class_text', u'Iris-versicolor', 0.33676574, 2),\n",
+       " (104, u'class_text', u'Iris-setosa', 0.029446673, 3),\n",
+       " (108, u'class_text', u'Iris-virginica', 0.52301323, 1),\n",
+       " (108, u'class_text', u'Iris-versicolor', 0.4507337, 2),\n",
+       " (108, u'class_text', u'Iris-setosa', 0.026253074, 3),\n",
+       " (114, u'class_text', u'Iris-virginica', 0.8007931, 1),\n",
+       " (114, u'class_text', u'Iris-versicolor', 0.17807306, 2),\n",
+       " (114, u'class_text', u'Iris-setosa', 0.021133851, 3),\n",
+       " (116, u'class_text', u'Iris-virginica', 0.8288441, 1),\n",
+       " (116, u'class_text', u'Iris-versicolor', 0.15422747, 2),\n",
+       " (116, u'class_text', u'Iris-setosa', 0.016928488, 3),\n",
+       " (117, u'class_text', u'Iris-virginica', 0.60298413, 1),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.3622377, 2),\n",
+       " (117, u'class_text', u'Iris-setosa', 0.034778137, 3),\n",
+       " (122, u'class_text', u'Iris-virginica', 0.79363465, 1),\n",
+       " (122, u'class_text', u'Iris-versicolor', 0.18244599, 2),\n",
+       " (122, u'class_text', u'Iris-setosa', 0.023919372, 3),\n",
+       " (123, u'class_text', u'Iris-virginica', 0.6008913, 1),\n",
+       " (123, u'class_text', u'Iris-versicolor', 0.38202757, 2),\n",
+       " (123, u'class_text', u'Iris-setosa', 0.01708112, 3),\n",
+       " (126, u'class_text', u'Iris-virginica', 0.51136595, 1),\n",
+       " (126, u'class_text', u'Iris-versicolor', 0.455122, 2),\n",
+       " (126, u'class_text', u'Iris-setosa', 0.033512004, 3),\n",
+       " (132, u'class_text', u'Iris-virginica', 0.5061915, 1),\n",
+       " (132, u'class_text', u'Iris-versicolor', 0.4644996, 2),\n",
+       " (132, u'class_text', u'Iris-setosa', 0.029308934, 3),\n",
+       " (137, u'class_text', u'Iris-virginica', 0.8607982, 1),\n",
+       " (137, u'class_text', u'Iris-versicolor', 0.12758763, 2),\n",
+       " (137, u'class_text', u'Iris-setosa', 0.011614183, 3),\n",
+       " (143, u'class_text', u'Iris-virginica', 0.7419237, 1),\n",
+       " (143, u'class_text', u'Iris-versicolor', 0.23144698, 2),\n",
+       " (143, u'class_text', u'Iris-setosa', 0.026629237, 3),\n",
+       " (146, u'class_text', u'Iris-virginica', 0.81490934, 1),\n",
+       " (146, u'class_text', u'Iris-versicolor', 0.16574617, 2),\n",
+       " (146, u'class_text', u'Iris-setosa', 0.019344518, 3),\n",
+       " (150, u'class_text', u'Iris-virginica', 0.6649737, 1),\n",
+       " (150, u'class_text', u'Iris-versicolor', 0.29889813, 2),\n",
+       " (150, u'class_text', u'Iris-setosa', 0.03612826, 3)]"
+      ]
+     },
+     "execution_count": 74,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_model', -- model\n",
+    "                                   'iris_test',  -- test_table\n",
+    "                                   'id',  -- id column\n",
+    "                                   'attributes', -- independent var\n",
+    "                                   'iris_predict'  -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id, rank;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 75,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3L,)]"
+      ]
+     },
+     "execution_count": 75,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict JOIN iris_test USING (id)\n",
+    "WHERE iris_predict.class_value != iris_test.class_text AND iris_predict.rank = 1;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 76,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90.00</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('90.00'),)]"
+      ]
+     },
+     "execution_count": 76,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict.class_value as estimated\n",
+    "     from iris_predict inner join iris_test\n",
+    "     on iris_test.id=iris_predict.id where iris_predict.rank = 1) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred_byom\"></a>\n",
+    "# 7. Predict BYOM\n",
+    "The predict BYOM function allows you to do inference on models that have not been trained on MADlib, but rather imported from elsewhere.  \n",
+    "\n",
+    "We will use the validation dataset for prediction as well, which is not usual but serves to show the syntax.\n",
+    "\n",
+    "See load_keras_model()\n",
+    "http://madlib.apache.org/docs/latest/group__grp__keras__model__arch.html\n",
+    "for details on how to load the model architecture and weights.  In this example we will use weights we already have:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 77,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 77,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "UPDATE model_arch_library \n",
+    "SET model_weights = iris_model.model_weights \n",
+    "FROM iris_model \n",
+    "WHERE model_arch_library.model_id = 1;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Now train using a model from the model architecture table directly without referencing the model table from the MADlib training.  \n",
+    "\n",
+    "Note that if you specify the class values parameter as we do below, it must reflect how the dependent variable was 1-hot encoded for training.  In this example the 'training_preprocessor_dl()' in Step 2 above encoded in the order {'Iris-setosa', 'Iris-versicolor', 'Iris-virginica'} so this is the order we pass in the parameter.  If we accidently picked another order that did not match the 1-hot encoding, the predictions would be wrong."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 78,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "30 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.4811115</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.4824369</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.46726868</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.48340678</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.4921991</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.49637538</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.47494373</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.51390177</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.6093597</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.55952024</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5187173</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.6168418</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.57109237</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.53077435</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.53015316</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5004781</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.49173826</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.63378763</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.52301323</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8007931</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8288441</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.60298413</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.79363465</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6008913</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.51136595</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5061915</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8607982</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7419237</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.81490934</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6649737</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(6, u'dependent_var', u'Iris-setosa', 0.4811115),\n",
+       " (9, u'dependent_var', u'Iris-setosa', 0.4824369),\n",
+       " (31, u'dependent_var', u'Iris-setosa', 0.46726868),\n",
+       " (32, u'dependent_var', u'Iris-setosa', 0.48340678),\n",
+       " (41, u'dependent_var', u'Iris-setosa', 0.4921991),\n",
+       " (52, u'dependent_var', u'Iris-versicolor', 0.49637538),\n",
+       " (57, u'dependent_var', u'Iris-virginica', 0.47494373),\n",
+       " (60, u'dependent_var', u'Iris-virginica', 0.51390177),\n",
+       " (63, u'dependent_var', u'Iris-versicolor', 0.6093597),\n",
+       " (66, u'dependent_var', u'Iris-versicolor', 0.55952024),\n",
+       " (67, u'dependent_var', u'Iris-virginica', 0.5187173),\n",
+       " (68, u'dependent_var', u'Iris-versicolor', 0.6168418),\n",
+       " (77, u'dependent_var', u'Iris-versicolor', 0.57109237),\n",
+       " (81, u'dependent_var', u'Iris-versicolor', 0.53077435),\n",
+       " (83, u'dependent_var', u'Iris-versicolor', 0.53015316),\n",
+       " (94, u'dependent_var', u'Iris-versicolor', 0.5004781),\n",
+       " (100, u'dependent_var', u'Iris-versicolor', 0.49173826),\n",
+       " (104, u'dependent_var', u'Iris-virginica', 0.63378763),\n",
+       " (108, u'dependent_var', u'Iris-virginica', 0.52301323),\n",
+       " (114, u'dependent_var', u'Iris-virginica', 0.8007931),\n",
+       " (116, u'dependent_var', u'Iris-virginica', 0.8288441),\n",
+       " (117, u'dependent_var', u'Iris-virginica', 0.60298413),\n",
+       " (122, u'dependent_var', u'Iris-virginica', 0.79363465),\n",
+       " (123, u'dependent_var', u'Iris-virginica', 0.6008913),\n",
+       " (126, u'dependent_var', u'Iris-virginica', 0.51136595),\n",
+       " (132, u'dependent_var', u'Iris-virginica', 0.5061915),\n",
+       " (137, u'dependent_var', u'Iris-virginica', 0.8607982),\n",
+       " (143, u'dependent_var', u'Iris-virginica', 0.7419237),\n",
+       " (146, u'dependent_var', u'Iris-virginica', 0.81490934),\n",
+       " (150, u'dependent_var', u'Iris-virginica', 0.6649737)]"
+      ]
+     },
+     "execution_count": 78,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict_byom;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict_byom('model_arch_library',  -- model arch table\n",
+    "                                         1,                    -- model arch id\n",
+    "                                        'iris_test',           -- test_table\n",
+    "                                        'id',                  -- id column\n",
+    "                                        'attributes',          -- independent var\n",
+    "                                        'iris_predict_byom',   -- output table\n",
+    "                                        'response',            -- prediction type\n",
+    "                                         FALSE,                -- use GPUs\n",
+    "                                         ARRAY[ARRAY['Iris-setosa', 'Iris-versicolor', 'Iris-virginica']], -- class values\n",
+    "                                         1.0                   -- normalizing const\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict_byom ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 79,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3L,)]"
+      ]
+     },
+     "execution_count": 79,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict_byom JOIN iris_test USING (id)\n",
+    "WHERE iris_predict_byom.class_value != iris_test.class_text;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 80,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90.00</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('90.00'),)]"
+      ]
+     },
+     "execution_count": 80,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict_byom.class_value as estimated\n",
+    "     from iris_predict_byom inner join iris_test\n",
+    "     on iris_test.id=iris_predict_byom.id) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class2\"></a>\n",
+    "# Classification with Other Parameters\n",
+    "\n",
+    "<a id=\"val_dataset\"></a>\n",
+    "# 1.  Validation dataset\n",
+    "Now use a validation dataset and compute metrics every 2nd iteration using the 'metrics_compute_frequency' parameter.  This can help reduce run time if you do not need metrics computed at every iteration."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 81,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 81,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10,                   -- num_iterations\n",
+    "                                FALSE,                -- use GPUs\n",
+    "                                'iris_test_packed',   -- validation dataset\n",
+    "                                2,                    -- metrics compute frequency\n",
+    "                                FALSE,                -- warm start\n",
+    "                               'Sophie L.',           -- name\n",
+    "                               'Simple MLP for iris dataset'  -- description\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 82,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>1</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Simple MLP for iris dataset</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-05 00:32:44.058709</td>\n",
+       "        <td>2021-03-05 00:32:45.314395</td>\n",
+       "        <td>[0.694608211517334, 0.840541124343872, 0.978843212127686, 1.11710405349731, 1.25560808181763]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.241201594472</td>\n",
+       "        <td>[0.941666662693024, 0.941666662693024, 0.949999988079071, 0.949999988079071, 0.949999988079071]</td>\n",
+       "        <td>[0.488521963357925, 0.39494463801384, 0.326846390962601, 0.278280317783356, 0.241201594471931]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.26036465168</td>\n",
+       "        <td>[0.966666638851166, 0.966666638851166, 0.966666638851166, 1.0, 0.966666638851166]</td>\n",
+       "        <td>[0.47617694735527, 0.398945957422256, 0.344237744808197, 0.293299406766891, 0.260364651679993]</td>\n",
+       "        <td>[2, 4, 6, 8, 10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', None, 2, u'Sophie L.', u'Simple MLP for iris dataset', u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 5, 0, 32, 44, 58709), datetime.datetime(2021, 3, 5, 0, 32, 45, 314395), [0.694608211517334, 0.840541124343872, 0.978843212127686, 1.11710405349731, 1.25560808181763], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.241201594471931, [0.941666662693024, 0.941666662693024, 0.949999988079071, 0.949999988079071, 0.949999988079071], [0.488521963357925, 0.39494463801384, 0.326846390962601, 0.278280317783356, 0.241201594471931], 0.966666638851166, 0.260364651679993, [0.966666638851166, 0.966666638851166, 0.966666638851166, 1.0, 0.966666638851166], [0.47617694735527, 0.398945957422256, 0.344237744808197, 0.293299406766891, 0.260364651679993], [2, 4, 6, 8, 10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 82,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Accuracy by iteration"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 83,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np\n",
+    "import sys\n",
+    "import os\n",
+    "from matplotlib import pyplot as plt\n",
+    "\n",
+    "# get accuracy and iteration number\n",
+    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
+    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
+    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
+    "\n",
+    "# get number of points\n",
+    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
+    "num_points = num_points_proxy[0]\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "iters = np.array(iters_proxy).reshape(num_points)\n",
+    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
+    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by iteration')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Loss by iteration"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 84,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get loss\n",
+    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
+    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
+    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation loss by iteration')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Loss')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Accuracy by time"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 85,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get time\n",
+    "time_proxy = %sql SELECT metrics_elapsed_time FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "time = np.array(time_proxy).reshape(num_points)/60.0\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by time')\n",
+    "plt.xlabel('Time (min)')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(time, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(time, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Time to achieve a given accuracy"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 86,
+   "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": [
+    "#plot\n",
+    "plt.title('Iris time by validation accuracy')\n",
+    "plt.xlabel('Accuracy')\n",
+    "plt.ylabel('Time (min)')\n",
+    "plt.grid(True)\n",
+    "plt.plot(train_accuracy, time, 'g.-', label='Train')\n",
+    "plt.plot(test_accuracy, time, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred_prob\"></a>\n",
+    "# 2. Predict probabilities\n",
+    "Predict with probabilities for each class:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 87,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "90 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.891271</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.10080952</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.007919549</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8695044</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.11757523</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.012920381</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8633581</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.12582295</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.010819025</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.88681984</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.102223285</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.010956874</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8932031</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.096981615</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.009815287</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.76211596</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.19126216</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.04662187</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.69715446</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.27219558</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.030649954</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.60125184</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.34705988</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.051688295</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8020071</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.13346818</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.06452477</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.81577915</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.12264396</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.06157696</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.60419196</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.36887947</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.026928646</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8244004</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.090247154</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.08535246</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7504721</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.22607413</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.023453651</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7521254</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.16548002</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.08239466</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.77373123</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.13472785</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.091540955</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7143379</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.15561377</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.13004832</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.73750204</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.19979021</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.06270776</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.75223774</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.24626015</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0015020997</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7122927</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.28699583</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0007114962</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.92487866</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.07457555</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00054574676</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9063354</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.092982225</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0006823736</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6758052</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.32160306</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0025916724</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.88375044</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.11495294</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0012965987</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8630747</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.1367832</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00014210763</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5698242</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4279695</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0022062436</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.52581084</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.4718109</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0023783143</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.93258834</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.06707534</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00033634563</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.85896367</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.13981287</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0012235282</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.90900683</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.090333104</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0006600239</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7043904</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.2916941</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0039154952</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(6, u'class_text', u'Iris-setosa', 0.891271, 1),\n",
+       " (6, u'class_text', u'Iris-versicolor', 0.10080952, 2),\n",
+       " (6, u'class_text', u'Iris-virginica', 0.007919549, 3),\n",
+       " (9, u'class_text', u'Iris-setosa', 0.8695044, 1),\n",
+       " (9, u'class_text', u'Iris-versicolor', 0.11757523, 2),\n",
+       " (9, u'class_text', u'Iris-virginica', 0.012920381, 3),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.8633581, 1),\n",
+       " (31, u'class_text', u'Iris-versicolor', 0.12582295, 2),\n",
+       " (31, u'class_text', u'Iris-virginica', 0.010819025, 3),\n",
+       " (32, u'class_text', u'Iris-setosa', 0.88681984, 1),\n",
+       " (32, u'class_text', u'Iris-versicolor', 0.102223285, 2),\n",
+       " (32, u'class_text', u'Iris-virginica', 0.010956874, 3),\n",
+       " (41, u'class_text', u'Iris-setosa', 0.8932031, 1),\n",
+       " (41, u'class_text', u'Iris-versicolor', 0.096981615, 2),\n",
+       " (41, u'class_text', u'Iris-virginica', 0.009815287, 3),\n",
+       " (52, u'class_text', u'Iris-versicolor', 0.76211596, 1),\n",
+       " (52, u'class_text', u'Iris-virginica', 0.19126216, 2),\n",
+       " (52, u'class_text', u'Iris-setosa', 0.04662187, 3),\n",
+       " (57, u'class_text', u'Iris-versicolor', 0.69715446, 1),\n",
+       " (57, u'class_text', u'Iris-virginica', 0.27219558, 2),\n",
+       " (57, u'class_text', u'Iris-setosa', 0.030649954, 3),\n",
+       " (60, u'class_text', u'Iris-versicolor', 0.60125184, 1),\n",
+       " (60, u'class_text', u'Iris-virginica', 0.34705988, 2),\n",
+       " (60, u'class_text', u'Iris-setosa', 0.051688295, 3),\n",
+       " (63, u'class_text', u'Iris-versicolor', 0.8020071, 1),\n",
+       " (63, u'class_text', u'Iris-virginica', 0.13346818, 2),\n",
+       " (63, u'class_text', u'Iris-setosa', 0.06452477, 3),\n",
+       " (66, u'class_text', u'Iris-versicolor', 0.81577915, 1),\n",
+       " (66, u'class_text', u'Iris-virginica', 0.12264396, 2),\n",
+       " (66, u'class_text', u'Iris-setosa', 0.06157696, 3),\n",
+       " (67, u'class_text', u'Iris-versicolor', 0.60419196, 1),\n",
+       " (67, u'class_text', u'Iris-virginica', 0.36887947, 2),\n",
+       " (67, u'class_text', u'Iris-setosa', 0.026928646, 3),\n",
+       " (68, u'class_text', u'Iris-versicolor', 0.8244004, 1),\n",
+       " (68, u'class_text', u'Iris-setosa', 0.090247154, 2),\n",
+       " (68, u'class_text', u'Iris-virginica', 0.08535246, 3),\n",
+       " (77, u'class_text', u'Iris-versicolor', 0.7504721, 1),\n",
+       " (77, u'class_text', u'Iris-virginica', 0.22607413, 2),\n",
+       " (77, u'class_text', u'Iris-setosa', 0.023453651, 3),\n",
+       " (81, u'class_text', u'Iris-versicolor', 0.7521254, 1),\n",
+       " (81, u'class_text', u'Iris-virginica', 0.16548002, 2),\n",
+       " (81, u'class_text', u'Iris-setosa', 0.08239466, 3),\n",
+       " (83, u'class_text', u'Iris-versicolor', 0.77373123, 1),\n",
+       " (83, u'class_text', u'Iris-virginica', 0.13472785, 2),\n",
+       " (83, u'class_text', u'Iris-setosa', 0.091540955, 3),\n",
+       " (94, u'class_text', u'Iris-versicolor', 0.7143379, 1),\n",
+       " (94, u'class_text', u'Iris-setosa', 0.15561377, 2),\n",
+       " (94, u'class_text', u'Iris-virginica', 0.13004832, 3),\n",
+       " (100, u'class_text', u'Iris-versicolor', 0.73750204, 1),\n",
+       " (100, u'class_text', u'Iris-virginica', 0.19979021, 2),\n",
+       " (100, u'class_text', u'Iris-setosa', 0.06270776, 3),\n",
+       " (104, u'class_text', u'Iris-virginica', 0.75223774, 1),\n",
+       " (104, u'class_text', u'Iris-versicolor', 0.24626015, 2),\n",
+       " (104, u'class_text', u'Iris-setosa', 0.0015020997, 3),\n",
+       " (108, u'class_text', u'Iris-virginica', 0.7122927, 1),\n",
+       " (108, u'class_text', u'Iris-versicolor', 0.28699583, 2),\n",
+       " (108, u'class_text', u'Iris-setosa', 0.0007114962, 3),\n",
+       " (114, u'class_text', u'Iris-virginica', 0.92487866, 1),\n",
+       " (114, u'class_text', u'Iris-versicolor', 0.07457555, 2),\n",
+       " (114, u'class_text', u'Iris-setosa', 0.00054574676, 3),\n",
+       " (116, u'class_text', u'Iris-virginica', 0.9063354, 1),\n",
+       " (116, u'class_text', u'Iris-versicolor', 0.092982225, 2),\n",
+       " (116, u'class_text', u'Iris-setosa', 0.0006823736, 3),\n",
+       " (117, u'class_text', u'Iris-virginica', 0.6758052, 1),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.32160306, 2),\n",
+       " (117, u'class_text', u'Iris-setosa', 0.0025916724, 3),\n",
+       " (122, u'class_text', u'Iris-virginica', 0.88375044, 1),\n",
+       " (122, u'class_text', u'Iris-versicolor', 0.11495294, 2),\n",
+       " (122, u'class_text', u'Iris-setosa', 0.0012965987, 3),\n",
+       " (123, u'class_text', u'Iris-virginica', 0.8630747, 1),\n",
+       " (123, u'class_text', u'Iris-versicolor', 0.1367832, 2),\n",
+       " (123, u'class_text', u'Iris-setosa', 0.00014210763, 3),\n",
+       " (126, u'class_text', u'Iris-virginica', 0.5698242, 1),\n",
+       " (126, u'class_text', u'Iris-versicolor', 0.4279695, 2),\n",
+       " (126, u'class_text', u'Iris-setosa', 0.0022062436, 3),\n",
+       " (132, u'class_text', u'Iris-versicolor', 0.52581084, 1),\n",
+       " (132, u'class_text', u'Iris-virginica', 0.4718109, 2),\n",
+       " (132, u'class_text', u'Iris-setosa', 0.0023783143, 3),\n",
+       " (137, u'class_text', u'Iris-virginica', 0.93258834, 1),\n",
+       " (137, u'class_text', u'Iris-versicolor', 0.06707534, 2),\n",
+       " (137, u'class_text', u'Iris-setosa', 0.00033634563, 3),\n",
+       " (143, u'class_text', u'Iris-virginica', 0.85896367, 1),\n",
+       " (143, u'class_text', u'Iris-versicolor', 0.13981287, 2),\n",
+       " (143, u'class_text', u'Iris-setosa', 0.0012235282, 3),\n",
+       " (146, u'class_text', u'Iris-virginica', 0.90900683, 1),\n",
+       " (146, u'class_text', u'Iris-versicolor', 0.090333104, 2),\n",
+       " (146, u'class_text', u'Iris-setosa', 0.0006600239, 3),\n",
+       " (150, u'class_text', u'Iris-virginica', 0.7043904, 1),\n",
+       " (150, u'class_text', u'Iris-versicolor', 0.2916941, 2),\n",
+       " (150, u'class_text', u'Iris-setosa', 0.0039154952, 3)]"
+      ]
+     },
+     "execution_count": 87,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_model',      -- model\n",
+    "                                   'iris_test',       -- test_table\n",
+    "                                   'id',              -- id column\n",
+    "                                   'attributes',      -- independent var\n",
+    "                                   'iris_predict',    -- output table\n",
+    "                                   'prob'             -- response type\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id, rank;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"warm_start\"></a>\n",
+    "# 3. Warm start\n",
+    "Next, use the warm_start parameter to continue learning, using the coefficients from the run above. Note that we don't drop the model table or model summary table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 88,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 88,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10,                   -- num_iterations\n",
+    "                                FALSE,                -- use GPUs\n",
+    "                                'iris_test_packed',   -- validation dataset\n",
+    "                                2,                    -- metrics compute frequency\n",
+    "                                TRUE,                 -- warm start\n",
+    "                               'Sophie L.',           -- name \n",
+    "                               'Simple MLP for iris dataset'  -- description\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "In the summary table and plots below note that the loss and accuracy values pick up from where the previous run left off:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 89,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>1</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Simple MLP for iris dataset</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-05 00:32:52.007110</td>\n",
+       "        <td>2021-03-05 00:32:53.468497</td>\n",
+       "        <td>[0.856245040893555, 1.01584315299988, 1.16454410552979, 1.31038999557495, 1.4613139629364]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.148840203881</td>\n",
+       "        <td>[0.958333313465118, 0.949999988079071, 0.958333313465118, 0.949999988079071, 0.949999988079071]</td>\n",
+       "        <td>[0.212763890624046, 0.190055623650551, 0.173214688897133, 0.159584209322929, 0.148840203881264]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.15544141829</td>\n",
+       "        <td>[0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.239033177495003, 0.209584251046181, 0.192669615149498, 0.169393673539162, 0.155441418290138]</td>\n",
+       "        <td>[2, 4, 6, 8, 10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', None, 2, u'Sophie L.', u'Simple MLP for iris dataset', u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 5, 0, 32, 52, 7110), datetime.datetime(2021, 3, 5, 0, 32, 53, 468497), [0.856245040893555, 1.01584315299988, 1.16454410552979, 1.31038999557495, 1.4613139629364], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.148840203881264, [0.958333313465118, 0.949999988079071, 0.958333313465118, 0.949999988079071, 0.949999988079071], [0.212763890624046, 0.190055623650551, 0.173214688897133, 0.159584209322929, 0.148840203881264], 0.966666638851166, 0.155441418290138, [0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166], [0.239033177495003, 0.209584251046181, 0.192669615149498, 0.169393673539162, 0.155441418290138], [2, 4, 6, 8, 10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 89,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 90,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np\n",
+    "import sys\n",
+    "import os\n",
+    "from matplotlib import pyplot as plt\n",
+    "\n",
+    "# get accuracy and iteration number\n",
+    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
+    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
+    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
+    "\n",
+    "# get number of points\n",
+    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
+    "num_points = num_points_proxy[0]\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "iters = np.array(iters_proxy).reshape(num_points)\n",
+    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
+    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by iteration - warm start')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 91,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get loss\n",
+    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
+    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
+    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation loss by iteration - warm start')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Loss')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"transfer_learn\"></a>\n",
+    "# Transfer learning\n",
+    "\n",
+    "<a id=\"load2\"></a>\n",
+    "# 1. Define and load model architecture with some layers frozen\n",
+    "Here we want to start with initial weights from a pre-trained model rather than training from scratch.  We also want to use a model architecture with the earlier feature layer(s) frozen to save on training time.  The example below is somewhat contrived but gives you the idea of the steps.\n",
+    "\n",
+    "First define a model architecture with the 1st hidden layer frozen:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 92,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_4\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_12 (Dense)             (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_13 (Dense)             (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_14 (Dense)             (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 143\n",
+      "Non-trainable params: 50\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model_transfer = Sequential()\n",
+    "model_transfer.add(Dense(10, activation='relu', input_shape=(4,), trainable=False))\n",
+    "model_transfer.add(Dense(10, activation='relu'))\n",
+    "model_transfer.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model_transfer.summary()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 93,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_12\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": false, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_13\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_14\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_4\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 93,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model_transfer.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load transfer model into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 94,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': False, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>A transfer model</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Sophie', u'A simple model'),\n",
+       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1341 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Maria', u'A transfer model')]"
+      ]
+     },
+     "execution_count": 94,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,                      \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": false, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Maria',               -- Name\n",
+    "                               'A transfer model'     -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT model_id, model_arch, name, description FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train2\"></a>\n",
+    "# 2. Train transfer model\n",
+    "\n",
+    "Fetch the weights from a previous MADlib run.  (Normally these would be downloaded from a source that trained the same model architecture on a related dataset.)"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 95,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 95,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "UPDATE model_arch_library \n",
+    "SET model_weights = iris_model.model_weights \n",
+    "FROM iris_model \n",
+    "WHERE model_arch_library.model_id = 2;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Now train the model using the transfer model and the pre-trained weights:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 96,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 96,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                2,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10,                   -- num_iterations\n",
+    "                                FALSE,                -- use GPUs\n",
+    "                                'iris_test_packed',   -- validation dataset\n",
+    "                                2                     -- metrics compute frequency\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 97,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>2</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-05 00:32:57.328345</td>\n",
+       "        <td>2021-03-05 00:32:58.501111</td>\n",
+       "        <td>[0.633425951004028, 0.770482063293457, 0.901815891265869, 1.0358898639679, 1.17268896102905]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.120888665318</td>\n",
+       "        <td>[0.949999988079071, 0.949999988079071, 0.949999988079071, 0.949999988079071, 0.949999988079071]</td>\n",
+       "        <td>[0.141591802239418, 0.135162934660912, 0.129805147647858, 0.125084936618805, 0.120888665318489]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.117368154228</td>\n",
+       "        <td>[0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.151054725050926, 0.138582393527031, 0.133960351347923, 0.12214257568121, 0.117368154227734]</td>\n",
+       "        <td>[2, 4, 6, 8, 10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 2, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', None, 2, None, None, u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 5, 0, 32, 57, 328345), datetime.datetime(2021, 3, 5, 0, 32, 58, 501111), [0.633425951004028, 0.770482063293457, 0.901815891265869, 1.0358898639679, 1.17268896102905], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.120888665318489, [0.949999988079071, 0.949999988079071, 0.949999988079071, 0.949999988079071, 0.949999988079071], [0.141591802239418, 0.135162934660912, 0.129805147647858, 0.125084936618805, 0.120888665318489], 0.966666638851166, 0.117368154227734, [0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166], [0.151054725050926, 0.138582393527031, 0.133960351347923, 0.12214257568121, 0.117368154227734], [2, 4, 6, 8, 10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 97,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Note loss picks up from where the last training left off:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 98,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np\n",
+    "import sys\n",
+    "import os\n",
+    "from matplotlib import pyplot as plt\n",
+    "\n",
+    "# get accuracy and iteration number\n",
+    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
+    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
+    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
+    "\n",
+    "# get number of points\n",
+    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
+    "num_points = num_points_proxy[0]\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "iters = np.array(iters_proxy).reshape(num_points)\n",
+    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
+    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by iteration - transfer learn')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 99,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get loss\n",
+    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
+    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
+    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation loss by iteration - transfer learn')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Loss')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/MADlib-Keras-cifar10-cnn-v3.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-cifar10-cnn-v3-checkpoint.ipynb
similarity index 98%
copy from community-artifacts/Deep-learning/MADlib-Keras-cifar10-cnn-v3.ipynb
copy to community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-cifar10-cnn-v3-checkpoint.ipynb
index 987ff4b..f7053ef 100644
--- a/community-artifacts/Deep-learning/MADlib-Keras-cifar10-cnn-v3.ipynb
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-cifar10-cnn-v3-checkpoint.ipynb
@@ -59,46 +59,21 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 2,
+   "execution_count": 1,
    "metadata": {
     "scrolled": true
    },
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
     "%load_ext sql"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 4,
+   "execution_count": 2,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 4,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
     "# Greenplum Database 5.x on GCP - via tunnel\n",
     "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
@@ -108,7 +83,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 5,
+   "execution_count": 3,
    "metadata": {},
    "outputs": [
     {
@@ -126,15 +101,15 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
-     "execution_count": 5,
+     "execution_count": 3,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -155,32 +130,17 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
+   "execution_count": 5,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "Using TensorFlow backend.\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Couldn't import dot_parser, loading of dot files will not be possible.\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
     "from __future__ import print_function\n",
-    "import keras\n",
-    "from keras.datasets import cifar10\n",
-    "from keras.preprocessing.image import ImageDataGenerator\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense, Dropout, Activation, Flatten\n",
-    "from keras.layers import Conv2D, MaxPooling2D\n",
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.datasets import cifar10\n",
+    "from tensorflow.keras.preprocessing.image import ImageDataGenerator\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense, Dropout, Activation, Flatten\n",
+    "from tensorflow.keras.layers import Conv2D, MaxPooling2D\n",
     "import os\n",
     "\n",
     "batch_size = 32\n",
@@ -197,7 +157,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 6,
    "metadata": {},
    "outputs": [],
    "source": [
diff --git a/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-cifar10-inference-v1-checkpoint.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-cifar10-inference-v1-checkpoint.ipynb
new file mode 100644
index 0000000..eb3daa2
--- /dev/null
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-cifar10-inference-v1-checkpoint.ipynb
@@ -0,0 +1,718 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Inference for CIFAR-10 dataset using predict BYOM\n",
+    "The predict BYOM function allows you to do inference using models that have not been trained with MADlib, but rather imported or created elsewhere. It was added in MADlib 1.17.\n",
+    "\n",
+    "In this workbook we train a model in Python using\n",
+    "https://keras.io/examples/cifar10_cnn/\n",
+    "and run inference on the validation set.\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#setup\">1. Setup</a>\n",
+    "\n",
+    "<a href=\"#train_model\">2. Train model in Python</a>\n",
+    "\n",
+    "<a href=\"#load_model\">3. Load model into table</a>\n",
+    "\n",
+    "<a href=\"#load_images\">4. Get validation data set and load into table</a>\n",
+    "\n",
+    "<a href=\"#inference\">5. Inference</a>"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"setup\"></a>\n",
+    "# 1. Setup"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train_model\"></a>\n",
+    "# 2. Train model in Python\n",
+    "\n",
+    "Train a model in Python using https://keras.io/examples/cifar10_cnn/\n",
+    "\n",
+    "Define model"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "x_train shape: (50000, 32, 32, 3)\n",
+      "50000 train samples\n",
+      "10000 test samples\n",
+      "WARNING:tensorflow:From /Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/tensorflow/python/ops/init_ops.py:1251: calling __init__ (from tensorflow.python.ops.init_ops) with dtype is deprecated and will be removed in a future version.\n",
+      "Instructions for updating:\n",
+      "Call initializer instance with the dtype argument instead of passing it to the constructor\n"
+     ]
+    }
+   ],
+   "source": [
+    "from __future__ import print_function\n",
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.datasets import cifar10\n",
+    "from tensorflow.keras.preprocessing.image import ImageDataGenerator\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense, Dropout, Activation, Flatten\n",
+    "from tensorflow.keras.layers import Conv2D, MaxPooling2D\n",
+    "import os\n",
+    "\n",
+    "batch_size = 32\n",
+    "num_classes = 10\n",
+    "epochs = 2\n",
+    "data_augmentation = True\n",
+    "num_predictions = 20\n",
+    "#save_dir = os.path.join(os.getcwd(), 'saved_models')\n",
+    "#model_name = 'keras_cifar10_trained_model.h5'\n",
+    "\n",
+    "# The data, split between train and test sets:\n",
+    "(x_train, y_train), (x_test, y_test) = cifar10.load_data()\n",
+    "print('x_train shape:', x_train.shape)\n",
+    "print(x_train.shape[0], 'train samples')\n",
+    "print(x_test.shape[0], 'test samples')\n",
+    "\n",
+    "# Convert class vectors to binary class matrices.\n",
+    "y_train = keras.utils.to_categorical(y_train, num_classes)\n",
+    "y_test = keras.utils.to_categorical(y_test, num_classes)\n",
+    "\n",
+    "model = Sequential()\n",
+    "model.add(Conv2D(32, (3, 3), padding='same',\n",
+    "                 input_shape=x_train.shape[1:]))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(Conv2D(32, (3, 3)))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(MaxPooling2D(pool_size=(2, 2)))\n",
+    "model.add(Dropout(0.25))\n",
+    "\n",
+    "model.add(Conv2D(64, (3, 3), padding='same'))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(Conv2D(64, (3, 3)))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(MaxPooling2D(pool_size=(2, 2)))\n",
+    "model.add(Dropout(0.25))\n",
+    "\n",
+    "model.add(Flatten())\n",
+    "model.add(Dense(512))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(Dropout(0.5))\n",
+    "model.add(Dense(num_classes))\n",
+    "model.add(Activation('softmax'))\n",
+    "\n",
+    "# initiate RMSprop optimizer\n",
+    "opt = keras.optimizers.RMSprop(lr=0.0001, decay=1e-6)\n",
+    "\n",
+    "# Let's train the model using RMSprop\n",
+    "model.compile(loss='categorical_crossentropy',\n",
+    "              optimizer=opt,\n",
+    "              metrics=['accuracy']);"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"conv2d\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 32, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"batch_input_shape\": [null, 32, 32, 3], \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation\"}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"conv2d_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"valid\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 32, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_1\"}}, {\"class_name\": \"MaxPooling2D\", \"config\": {\"name\": \"max_pooling2d\", \"dtype\": \"float32\", \"trainable\": true, \"data_format\": \"channels_last\", \"pool_size\": [2, 2], \"padding\": \"valid\", \"strides\": [2, 2]}}, {\"class_name\": \"Dropout\", \"config\": {\"name\": \"dropout\", \"dtype\": \"float32\", \"trainable\": true, \"rate\": 0.25, \"seed\": null, \"noise_shape\": null}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"conv2d_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 64, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_2\"}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"conv2d_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"valid\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 64, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_3\"}}, {\"class_name\": \"MaxPooling2D\", \"config\": {\"name\": \"max_pooling2d_1\", \"dtype\": \"float32\", \"trainable\": true, \"data_format\": \"channels_last\", \"pool_size\": [2, 2], \"padding\": \"valid\", \"strides\": [2, 2]}}, {\"class_name\": \"Dropout\", \"config\": {\"name\": \"dropout_1\", \"dtype\": \"float32\", \"trainable\": true, \"rate\": 0.25, \"seed\": null, \"noise_shape\": null}}, {\"class_name\": \"Flatten\", \"config\": {\"dtype\": \"float32\", \"trainable\": true, \"name\": \"flatten\", \"data_format\": \"channels_last\"}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 512, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_4\"}}, {\"class_name\": \"Dropout\", \"config\": {\"name\": \"dropout_2\", \"dtype\": \"float32\", \"trainable\": true, \"rate\": 0.5, \"seed\": null, \"noise_shape\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"name\": \"activation_5\"}}], \"name\": \"sequential\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model.to_json()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Using real-time data augmentation.\n",
+      "Epoch 1/2\n",
+      "1563/1563 [==============================] - 107s 69ms/step - loss: 1.8637 - acc: 0.3142 - val_loss: 1.6037 - val_acc: 0.4154\n",
+      "Epoch 2/2\n",
+      "1563/1563 [==============================] - 116s 74ms/step - loss: 1.5880 - acc: 0.4174 - val_loss: 1.4362 - val_acc: 0.4754\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "<tensorflow.python.keras.callbacks.History at 0x14dfc98d0>"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "10000/10000 [==============================] - 7s 698us/sample - loss: 1.4365 - acc: 0.4754\n",
+      "Test loss: 1.4364811393737793\n",
+      "Test accuracy: 0.4754\n"
+     ]
+    }
+   ],
+   "source": [
+    "x_train = x_train.astype('float32')\n",
+    "x_test = x_test.astype('float32')\n",
+    "x_train /= 255\n",
+    "x_test /= 255\n",
+    "\n",
+    "if not data_augmentation:\n",
+    "    print('Not using data augmentation.')\n",
+    "    model.fit(x_train, y_train,\n",
+    "              batch_size=batch_size,\n",
+    "              epochs=epochs,\n",
+    "              validation_data=(x_test, y_test),\n",
+    "              shuffle=True)\n",
+    "else:\n",
+    "    print('Using real-time data augmentation.')\n",
+    "    # This will do preprocessing and realtime data augmentation:\n",
+    "    datagen = ImageDataGenerator(\n",
+    "        featurewise_center=False,  # set input mean to 0 over the dataset\n",
+    "        samplewise_center=False,  # set each sample mean to 0\n",
+    "        featurewise_std_normalization=False,  # divide inputs by std of the dataset\n",
+    "        samplewise_std_normalization=False,  # divide each input by its std\n",
+    "        zca_whitening=False,  # apply ZCA whitening\n",
+    "        zca_epsilon=1e-06,  # epsilon for ZCA whitening\n",
+    "        rotation_range=0,  # randomly rotate images in the range (degrees, 0 to 180)\n",
+    "        # randomly shift images horizontally (fraction of total width)\n",
+    "        width_shift_range=0.1,\n",
+    "        # randomly shift images vertically (fraction of total height)\n",
+    "        height_shift_range=0.1,\n",
+    "        shear_range=0.,  # set range for random shear\n",
+    "        zoom_range=0.,  # set range for random zoom\n",
+    "        channel_shift_range=0.,  # set range for random channel shifts\n",
+    "        # set mode for filling points outside the input boundaries\n",
+    "        fill_mode='nearest',\n",
+    "        cval=0.,  # value used for fill_mode = \"constant\"\n",
+    "        horizontal_flip=True,  # randomly flip images\n",
+    "        vertical_flip=False,  # randomly flip images\n",
+    "        # set rescaling factor (applied before any other transformation)\n",
+    "        rescale=None,\n",
+    "        # set function that will be applied on each input\n",
+    "        preprocessing_function=None,\n",
+    "        # image data format, either \"channels_first\" or \"channels_last\"\n",
+    "        data_format=None,\n",
+    "        # fraction of images reserved for validation (strictly between 0 and 1)\n",
+    "        validation_split=0.0)\n",
+    "\n",
+    "    # Compute quantities required for feature-wise normalization\n",
+    "    # (std, mean, and principal components if ZCA whitening is applied).\n",
+    "    datagen.fit(x_train)\n",
+    "\n",
+    "    # Fit the model on the batches generated by datagen.flow().\n",
+    "    model.fit_generator(datagen.flow(x_train, y_train,\n",
+    "                                     batch_size=batch_size),\n",
+    "                        epochs=epochs,\n",
+    "                        validation_data=(x_test, y_test),\n",
+    "                        workers=1)\n",
+    "\n",
+    "# Save model and weights\n",
+    "#if not os.path.isdir(save_dir):\n",
+    "#    os.makedirs(save_dir)\n",
+    "#model_path = os.path.join(save_dir, model_name)\n",
+    "#model.save(model_path)\n",
+    "#print('Saved trained model at %s ' % model_path)\n",
+    "\n",
+    "# Score trained model.\n",
+    "scores = model.evaluate(x_test, y_test, verbose=1)\n",
+    "print('Test loss:', scores[0])\n",
+    "print('Test accuracy:', scores[1])"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_model\"></a>\n",
+    "# 3.  Load model into table\n",
+    "\n",
+    "Load the model architecture and weights into the model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>CIFAR10 model</td>\n",
+       "        <td>CNN model with weights trained on CIFAR10.</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'CIFAR10 model', u'CNN model with weights trained on CIFAR10.')]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "import psycopg2 as p2\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
+    "#conn = p2.connect('postgresql://fmcquillan@localhost:5432/madlib')\n",
+    "cur = conn.cursor()\n",
+    "\n",
+    "from keras.layers import *\n",
+    "from keras import Sequential\n",
+    "import numpy as np\n",
+    "\n",
+    "# get weights, flatten and serialize\n",
+    "weights = model.get_weights()\n",
+    "weights_flat = [w.flatten() for w in weights]\n",
+    "weights1d =  np.concatenate(weights_flat).ravel()\n",
+    "weights_bytea = p2.Binary(weights1d.tostring())\n",
+    "\n",
+    "%sql DROP TABLE IF EXISTS model_arch_library_cifar10;\n",
+    "query = \"SELECT madlib.load_keras_model('model_arch_library_cifar10', %s,%s,%s,%s)\"\n",
+    "cur.execute(query,[model.to_json(), weights_bytea, \"CIFAR10 model\", \"CNN model with weights trained on CIFAR10.\"])\n",
+    "conn.commit()\n",
+    "\n",
+    "# check weights loaded OK\n",
+    "%sql SELECT model_id, name, description FROM model_arch_library_cifar10;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_images\"></a>\n",
+    "# 4. Get validation data set and load into table\n",
+    "\n",
+    "First set up image loader using the script called <em>madlib_image_loader.py</em> located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "import sys\n",
+    "import os\n",
+    "madlib_site_dir = '/Users/fmcquillan/Documents/Product/MADlib/Demos/data'\n",
+    "sys.path.append(madlib_site_dir)\n",
+    "\n",
+    "# Import image loader module\n",
+    "from madlib_image_loader import ImageLoader, DbCredentials\n",
+    "\n",
+    "# Specify database credentials, for connecting to db\n",
+    "#db_creds = DbCredentials(user='fmcquillan',\n",
+    "#                         host='localhost',\n",
+    "#                         port='5432',\n",
+    "#                         password='')\n",
+    "\n",
+    "# Specify database credentials, for connecting to db\n",
+    "db_creds = DbCredentials(user='gpadmin', \n",
+    "                         db_name='madlib',\n",
+    "                         host='localhost',\n",
+    "                         port='8000',\n",
+    "                         password='')\n",
+    "\n",
+    "# Initialize ImageLoader (increase num_workers to run faster)\n",
+    "iloader = ImageLoader(num_workers=5, db_creds=db_creds)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Next load CIFAR-10 data from Keras consisting of 50,000 32x32 color training images, labeled over 10 categories, and 10,000 test images."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "MainProcess: Connected to madlib db.\n",
+      "Executing: CREATE TABLE cifar_10_test_data (id SERIAL, x REAL[], y TEXT)\n",
+      "CREATE TABLE\n",
+      "Created table cifar_10_test_data in madlib db\n",
+      "Spawning 5 workers...\n",
+      "Initializing PoolWorker-1 [pid 95042]\n",
+      "PoolWorker-1: Created temporary directory /tmp/madlib_dTZhEGBDFE\n",
+      "Initializing PoolWorker-2 [pid 95043]\n",
+      "PoolWorker-2: Created temporary directory /tmp/madlib_ctWjbhcjwz\n",
+      "Initializing PoolWorker-3 [pid 95044]\n",
+      "PoolWorker-3: Created temporary directory /tmp/madlib_nx9VuMScrX\n",
+      "Initializing PoolWorker-4 [pid 95045]\n",
+      "PoolWorker-4: Created temporary directory /tmp/madlib_thkphNCw4r\n",
+      "Initializing PoolWorker-5 [pid 95046]\n",
+      "PoolWorker-5: Created temporary directory /tmp/madlib_037luEXgEL\n",
+      "PoolWorker-2: Connected to madlib db.\n",
+      "PoolWorker-3: Connected to madlib db.\n",
+      "PoolWorker-1: Connected to madlib db.\n",
+      "PoolWorker-5: Connected to madlib db.\n",
+      "PoolWorker-4: Connected to madlib db.\n",
+      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_nx9VuMScrX/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_dTZhEGBDFE/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_ctWjbhcjwz/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_037luEXgEL/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_thkphNCw4r/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-3: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_nx9VuMScrX/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-1: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_dTZhEGBDFE/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-2: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-4: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-5: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_ctWjbhcjwz/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_thkphNCw4r/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_037luEXgEL/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-3: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-1: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-2: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-4: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-5: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-1: Removed temporary directory /tmp/madlib_dTZhEGBDFE\n",
+      "PoolWorker-2: Removed temporary directory /tmp/madlib_ctWjbhcjwz\n",
+      "PoolWorker-3: Removed temporary directory /tmp/madlib_nx9VuMScrX\n",
+      "PoolWorker-5: Removed temporary directory /tmp/madlib_037luEXgEL\n",
+      "PoolWorker-4: Removed temporary directory /tmp/madlib_thkphNCw4r\n",
+      "Done!  Loaded 10000 images in 108.267487049s\n",
+      "5 workers terminated.\n"
+     ]
+    }
+   ],
+   "source": [
+    "from keras.datasets import cifar10\n",
+    "\n",
+    "# Load dataset into np array\n",
+    "(x_train, y_train), (x_test, y_test) = cifar10.load_data()\n",
+    "\n",
+    "%sql DROP TABLE IF EXISTS cifar_10_test_data;\n",
+    "\n",
+    "# Save images to temporary directories and load into database\n",
+    "#iloader.load_dataset_from_np(x_train, y_train, 'cifar_10_train_data', append=False)\n",
+    "iloader.load_dataset_from_np(x_test, y_test, 'cifar_10_test_data', append=False)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"inference\"></a>\n",
+    "# 5. Inference"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n"
+     ]
+    },
+    {
+     "ename": "InternalError",
+     "evalue": "(psycopg2.errors.InternalError_) plpy.Error: Unable to get number of classes from model architecture. (plpython.c:5038)\nCONTEXT:  Traceback (most recent call last):\n  PL/Python function \"madlib_keras_predict_byom\", line 23, in <module>\n    madlib_keras_predict.PredictBYOM(**globals())\n  PL/Python function \"madlib_keras_predict_byom\", line 42, in wrapper\n  PL/Python function \"madlib_keras_predict_byom\", line 314, in __init__\n  PL/Python function \"madlib_keras_predict_byom\", line 326, in validate_and_set_defaults\n  PL/Python function \"madlib_keras_predict_byom\", line 207, in set_default_class_values\n  PL/Python function \"madlib_keras_predict_byom\", line 75, in get_num_classes\nPL/Python function \"madlib_keras_predict_byom\"\n\n[SQL: SELECT madlib.madlib_keras_predict_byom('model_arch_library_cifar10',  -- model arch table\n                                         1,                            -- model arch id\n                                        'cifar_10_test_data',          -- test_table\n                                        'id',                          -- id column\n                                        'x',                           -- independent var\n                                        'cifar10_predict_byom',        -- output table\n                                        'response',                    -- prediction type\n                                         FALSE,                        -- use gpus\n                                         NULL,                         -- class values\n                                         255.0                         -- normalizing const\n                                   );]\n(Background on this error at: http://sqlalche.me/e/2j85)",
+     "output_type": "error",
+     "traceback": [
+      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
+      "\u001b[0;31mInternalError\u001b[0m                             Traceback (most recent call last)",
+      "\u001b[0;32m<ipython-input-11-d7da0ccca3f1>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m()\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0mget_ipython\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrun_cell_magic\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34mu'sql'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34mu''\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34mu\"DROP TABLE IF EXISTS cifar10_predict_byom;\\n\\nSELECT madlib.madlib_keras_predict_byom('model_arch_library_cifar10',  -- model arch table\\n                                         1,                            -- model arch id\\n                                        'cifar_10_test_data',          -- test_table\\n                                        'id',                          -- id column\\n                                        'x',                           -- independent var\\n                                        'cifar10_predict_byom',        -- output table\\n                                        'response',                    -- prediction type\\n                                         FALSE,                        -- use gpus\\n                                         NULL,                         -- class values\\n                                         255.0                         -- normalizing const\\n                                   );\\nSELECT * FROM cifar10_predict_byom ORDER BY id LIMIT 10;\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/IPython/core/interactiveshell.pyc\u001b[0m in \u001b[0;36mrun_cell_magic\u001b[0;34m(self, magic_name, line, cell)\u001b[0m\n\u001b[1;32m   2115\u001b[0m             \u001b[0mmagic_arg_s\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mvar_expand\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mline\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstack_depth\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2116\u001b[0m             \u001b[0;32mwith\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbuiltin_trap\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2117\u001b[0;31m                 \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mfn\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mmagic_arg_s\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcell\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2118\u001b[0m             \u001b[0;32mreturn\u001b[0m \u001b[0mresult\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2119\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m</Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/decorator.pyc:decorator-gen-124>\u001b[0m in \u001b[0;36mexecute\u001b[0;34m(self, line, cell, local_ns)\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/IPython/core/magic.pyc\u001b[0m in \u001b[0;36m<lambda>\u001b[0;34m(f, *a, **k)\u001b[0m\n\u001b[1;32m    186\u001b[0m     \u001b[0;31m# but it's overkill for just that one bit of state.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    187\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mmagic_deco\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0marg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 188\u001b[0;31m         \u001b[0mcall\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mlambda\u001b[0m \u001b[0mf\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mf\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    189\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    190\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mcallable\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0marg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m</Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/decorator.pyc:decorator-gen-123>\u001b[0m in \u001b[0;36mexecute\u001b[0;34m(self, line, cell, local_ns)\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/IPython/core/magic.pyc\u001b[0m in \u001b[0;36m<lambda>\u001b[0;34m(f, *a, **k)\u001b[0m\n\u001b[1;32m    186\u001b[0m     \u001b[0;31m# but it's overkill for just that one bit of state.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    187\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mmagic_deco\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0marg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 188\u001b[0;31m         \u001b[0mcall\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mlambda\u001b[0m \u001b[0mf\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mf\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    189\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    190\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mcallable\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0marg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sql/magic.pyc\u001b[0m in \u001b[0;36mexecute\u001b[0;34m(self, line, cell, local_ns)\u001b[0m\n\u001b[1;32m    135\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    136\u001b[0m         \u001b[0;32mtry\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 137\u001b[0;31m             \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0msql\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrun\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrun\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mconn\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparsed\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;34m'sql'\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0muser_ns\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    138\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    139\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0mresult\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mNone\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0misinstance\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mresult\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstr\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcolumn_local_vars\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sql/run.pyc\u001b[0m in \u001b[0;36mrun\u001b[0;34m(conn, sql, config, user_namespace)\u001b[0m\n\u001b[1;32m    361\u001b[0m             \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    362\u001b[0m                 \u001b[0mtxt\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0msqlalchemy\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msql\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtext\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mstatement\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 363\u001b[0;31m                 \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mconn\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msession\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mexecute\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtxt\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0muser_namespace\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    364\u001b[0m             \u001b[0m_commit\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mconn\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mconn\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mconfig\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mconfig\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    365\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0mresult\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0mconfig\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfeedback\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36mexecute\u001b[0;34m(self, object_, *multiparams, **params)\u001b[0m\n\u001b[1;32m    980\u001b[0m             \u001b[0;32mraise\u001b[0m \u001b[0mexc\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mObjectNotExecutableError\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mobject_\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    981\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 982\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mmeth\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmultiparams\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparams\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    983\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    984\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0m_execute_function\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mfunc\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmultiparams\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparams\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/sql/elements.pyc\u001b[0m in \u001b[0;36m_execute_on_connection\u001b[0;34m(self, connection, multiparams, params)\u001b[0m\n\u001b[1;32m    285\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0m_execute_on_connection\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mconnection\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmultiparams\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparams\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    286\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msupports_execution\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 287\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mconnection\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_execute_clauseelement\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmultiparams\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparams\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    288\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    289\u001b[0m             \u001b[0;32mraise\u001b[0m \u001b[0mexc\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mObjectNotExecutableError\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36m_execute_clauseelement\u001b[0;34m(self, elem, multiparams, params)\u001b[0m\n\u001b[1;32m   1099\u001b[0m             \u001b[0mdistilled_params\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1100\u001b[0m             \u001b[0mcompiled_sql\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1101\u001b[0;31m             \u001b[0mdistilled_params\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1102\u001b[0m         )\n\u001b[1;32m   1103\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_has_events\u001b[0m \u001b[0;32mor\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mengine\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_has_events\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36m_execute_context\u001b[0;34m(self, dialect, constructor, statement, parameters, *args)\u001b[0m\n\u001b[1;32m   1248\u001b[0m         \u001b[0;32mexcept\u001b[0m \u001b[0mBaseException\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1249\u001b[0m             self._handle_dbapi_exception(\n\u001b[0;32m-> 1250\u001b[0;31m                 \u001b[0me\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparameters\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcursor\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcontext\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1251\u001b[0m             )\n\u001b[1;32m   1252\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36m_handle_dbapi_exception\u001b[0;34m(self, e, statement, parameters, cursor, context)\u001b[0m\n\u001b[1;32m   1474\u001b[0m                 \u001b[0mutil\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mraise_from_cause\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mnewraise\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexc_info\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1475\u001b[0m             \u001b[0;32melif\u001b[0m \u001b[0mshould_wrap\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1476\u001b[0;31m                 \u001b[0mutil\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mraise_from_cause\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0msqlalchemy_exception\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexc_info\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1477\u001b[0m             \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1478\u001b[0m                 \u001b[0mutil\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mreraise\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0mexc_info\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/util/compat.pyc\u001b[0m in \u001b[0;36mraise_from_cause\u001b[0;34m(exception, exc_info)\u001b[0m\n\u001b[1;32m    396\u001b[0m     \u001b[0mexc_type\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexc_value\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexc_tb\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mexc_info\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    397\u001b[0m     \u001b[0mcause\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mexc_value\u001b[0m \u001b[0;32mif\u001b[0m \u001b[0mexc_value\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mexception\u001b[0m \u001b[0;32melse\u001b[0m \u001b[0mNone\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 398\u001b[0;31m     \u001b[0mreraise\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtype\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mexception\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexception\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtb\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mexc_tb\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcause\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mcause\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    399\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    400\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36m_execute_context\u001b[0;34m(self, dialect, constructor, statement, parameters, *args)\u001b[0m\n\u001b[1;32m   1244\u001b[0m                 \u001b[0;32mif\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mevt_handled\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1245\u001b[0m                     self.dialect.do_execute(\n\u001b[0;32m-> 1246\u001b[0;31m                         \u001b[0mcursor\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparameters\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcontext\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1247\u001b[0m                     )\n\u001b[1;32m   1248\u001b[0m         \u001b[0;32mexcept\u001b[0m \u001b[0mBaseException\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/default.pyc\u001b[0m in \u001b[0;36mdo_execute\u001b[0;34m(self, cursor, statement, parameters, context)\u001b[0m\n\u001b[1;32m    579\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    580\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mdo_execute\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcursor\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparameters\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcontext\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mNone\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 581\u001b[0;31m         \u001b[0mcursor\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mexecute\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparameters\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    582\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    583\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mdo_execute_no_params\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcursor\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcontext\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mNone\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;31mInternalError\u001b[0m: (psycopg2.errors.InternalError_) plpy.Error: Unable to get number of classes from model architecture. (plpython.c:5038)\nCONTEXT:  Traceback (most recent call last):\n  PL/Python function \"madlib_keras_predict_byom\", line 23, in <module>\n    madlib_keras_predict.PredictBYOM(**globals())\n  PL/Python function \"madlib_keras_predict_byom\", line 42, in wrapper\n  PL/Python function \"madlib_keras_predict_byom\", line 314, in __init__\n  PL/Python function \"madlib_keras_predict_byom\", line 326, in validate_and_set_defaults\n  PL/Python function \"madlib_keras_predict_byom\", line 207, in set_default_class_values\n  PL/Python function \"madlib_keras_predict_byom\", line 75, in get_num_classes\nPL/Python function \"madlib_keras_predict_byom\"\n\n[SQL: SELECT madlib.madlib_keras_predict_byom('model_arch_library_cifar10',  -- model arch table\n                                         1,                            -- model arch id\n                                        'cifar_10_test_data',          -- test_table\n                                        'id',                          -- id column\n                                        'x',                           -- independent var\n                                        'cifar10_predict_byom',        -- output table\n                                        'response',                    -- prediction type\n                                         FALSE,                        -- use gpus\n                                         NULL,                         -- class values\n                                         255.0                         -- normalizing const\n                                   );]\n(Background on this error at: http://sqlalche.me/e/2j85)"
+     ]
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS cifar10_predict_byom;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict_byom('model_arch_library_cifar10',  -- model arch table\n",
+    "                                         1,                            -- model arch id\n",
+    "                                        'cifar_10_test_data',          -- test_table\n",
+    "                                        'id',                          -- id column\n",
+    "                                        'x',                           -- independent var\n",
+    "                                        'cifar10_predict_byom',        -- output table\n",
+    "                                        'response',                    -- prediction type\n",
+    "                                         FALSE,                        -- use gpus\n",
+    "                                         NULL,                         -- class values\n",
+    "                                         255.0                         -- normalizing const\n",
+    "                                   );\n",
+    "SELECT * FROM cifar10_predict_byom ORDER BY id LIMIT 10;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Number of missclassifications:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2551</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2551L,)]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM cifar10_predict_byom JOIN cifar_10_test_data USING (id)\n",
+    "WHERE cifar10_predict_byom.estimated_dependent_var != cifar_10_test_data.y;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Predict accuracy. From https://keras.io/examples/cifar10_cnn/ accuracy claim is 75% on validation set after 25 epochs.  From run above test accuracy: 0.7449.  MADlib predict BYOM accuracy matches:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74.49</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('74.49'),)]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100.0/10000.0, 2) as test_accuracy_percent from\n",
+    "    (select cifar_10_test_data.y as actual, cifar10_predict_byom.estimated_dependent_var as estimated\n",
+    "     from cifar10_predict_byom inner join cifar_10_test_data\n",
+    "     on cifar_10_test_data.id=cifar10_predict_byom.id) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 2
+}
diff --git a/community-artifacts/Deep-learning/MADlib-Keras-imagenet-inference-v1.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-imagenet-inference-v1-checkpoint.ipynb
similarity index 89%
copy from community-artifacts/Deep-learning/MADlib-Keras-imagenet-inference-v1.ipynb
copy to community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-imagenet-inference-v1-checkpoint.ipynb
index 968ea88..d3aa1f5 100644
--- a/community-artifacts/Deep-learning/MADlib-Keras-imagenet-inference-v1.ipynb
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-imagenet-inference-v1-checkpoint.ipynb
@@ -40,17 +40,15 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 1,
+   "execution_count": 7,
    "metadata": {},
    "outputs": [
     {
-     "name": "stderr",
+     "name": "stdout",
      "output_type": "stream",
      "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
+      "The sql extension is already loaded. To reload it, use:\n",
+      "  %reload_ext sql\n"
      ]
     }
    ],
@@ -60,24 +58,10 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 2,
+   "execution_count": 8,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
     "# Greenplum Database 5.x on GCP - via tunnel\n",
     "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
@@ -87,7 +71,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 3,
+   "execution_count": 9,
    "metadata": {},
    "outputs": [
     {
@@ -105,15 +89,15 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-10-g205bdba, cmake configuration time: Mon Aug 26 16:15:40 UTC 2019, build type: release, build system: Linux-3.10.0-957.21.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-10-g205bdba, cmake configuration time: Mon Aug 26 16:15:40 UTC 2019, build type: release, build system: Linux-3.10.0-957.21.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
-     "execution_count": 3,
+     "execution_count": 9,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -135,7 +119,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 19,
+   "execution_count": 10,
    "metadata": {},
    "outputs": [
     {
@@ -151,7 +135,7 @@
        "[]"
       ]
      },
-     "execution_count": 19,
+     "execution_count": 10,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -176,7 +160,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 20,
+   "execution_count": 11,
    "metadata": {},
    "outputs": [
     {
@@ -208,7 +192,7 @@
        "[(1, u'VGG16', u'VGG16 model with weights pre-trained on ImageNet.')]"
       ]
      },
-     "execution_count": 20,
+     "execution_count": 11,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -261,7 +245,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 21,
+   "execution_count": 12,
    "metadata": {},
    "outputs": [
     {
@@ -283,23 +267,23 @@
        "        <th>description</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>VGG16</td>\n",
-       "        <td>VGG16 model with weights pre-trained on ImageNet.</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>ResNet50</td>\n",
        "        <td>ResNet50 model with weights pre-trained on ImageNet.</td>\n",
        "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>VGG16</td>\n",
+       "        <td>VGG16 model with weights pre-trained on ImageNet.</td>\n",
+       "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, u'VGG16', u'VGG16 model with weights pre-trained on ImageNet.'),\n",
-       " (2, u'ResNet50', u'ResNet50 model with weights pre-trained on ImageNet.')]"
+       "[(2, u'ResNet50', u'ResNet50 model with weights pre-trained on ImageNet.'),\n",
+       " (1, u'VGG16', u'VGG16 model with weights pre-trained on ImageNet.')]"
       ]
      },
-     "execution_count": 21,
+     "execution_count": 12,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -4678,7 +4662,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 3,
+   "execution_count": 11,
    "metadata": {},
    "outputs": [
     {
@@ -4704,7 +4688,7 @@
        "[(17863L,)]"
       ]
      },
-     "execution_count": 3,
+     "execution_count": 11,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -4724,7 +4708,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 3,
+   "execution_count": 12,
    "metadata": {},
    "outputs": [
     {
@@ -4750,7 +4734,7 @@
        "[(Decimal('64.27'),)]"
       ]
      },
-     "execution_count": 3,
+     "execution_count": 12,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -4773,7 +4757,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 13,
    "metadata": {},
    "outputs": [
     {
@@ -4792,29 +4776,41 @@
        "<table>\n",
        "    <tr>\n",
        "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
        "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
-       "        <td>[2.4545777e-06, 5.2808724e-07, 4.793985e-07, 3.0648587e-06, 1.7589542e-05, 7.423473e-06, 2.8404587e-05, 7.702516e-07, 2.9076324e-07, 4.9191232e-05, 7.6386027e-07, 1.6786764e-07, 2.8007811e-08, 1.10271536e-07, 6.7959e-08, 4.662994e-07, 1.4375345e-07, 1.0798308e-07, 3.0669946e-07, 2.0534213e-07, 7.262093e-07, 1.9336496e-06, 5.923924e-06, 2.124527e-06, 1.517234e-06, 9.527852e-05, 0.00018055105, 8.729679e-06, 1.3466038e-05, 4.9883015e-06, 0.00016520561, 8.931284e-07, 0.00010447865, 0.008126934, 0.0041637723, 0.00013488345, 0.0004189078, 1.1425285e-05, 0.0005037103, 0.0009210269, 1.1044925e-06, 0.00024138592, 9.045082e-05, 9.4167415e-05, 0.0013647893, 0.0005210496, 2.0178764e-05, 5.4065484e-05, 0.0009393721, 0.05819438, 0.027708823, 5.9237965e-07, 0.00053300196, 0.002545086, 0.037394878, 6.184113e-05, 0.000853419, 0.0009619954, 0.1408455, 0.00020672889, 0.014558754, 0.00151045, 0.055647947, 0.011832289, 0.0007671514, 0.5346089, 0.006554945, 0.041236367, 0.039578155, 1.4561356e-05, 1.6464638e-07, 3.165858e-05, 3.935167e-07, 1.0918371e-06, 5.218124e-07, 2.7307672e-07, 1.0199102e-05, 3.5511948e-06, 2.86876e-07, 4.0874023e-05, 2.6729484e-07, 1.0093462e-06, 4.629379e-06, 2.1517774e-06, 1.2261435e-05, 4.338481e-06, 9.929987e-07, 1.7831232e-07, 2.4536823e-07, 1.5457067e-07, 1.2165957e-07, 3.4199334e-07, 5.923435e-07, 2.5983104e-06, 1.6504788e-07, 7.1641075e-08, 5.894999e-07, 1.5662519e-06, 5.8751893e-06, 3.89696e-06, 4.1094467e-05, 4.955301e-06, 6.5522227e-06, 1.9406043e-05, 1.6132468e-05, 3.7659976e-07, 6.3438647e-06, 4.0902314e-06, 1.0292862e-06, 6.6630196e-06, 5.040724e-06, 8.1575585e-05, 7.456448e-05, 2.8282989e-06, 3.618538e-05, 3.8675685e-07, 7.651428e-05, 1.0225481e-06, 1.266903e-05, 4.5824258e-05, 7.288306e-05, 3.3498336e-06, 2.992845e-06, 1.8091552e-06, 1.875435e-05, 6.7368404e-05, 4.9985447e-06, 3.3620515e-07, 4.305059e-06, 1.7280902e-06, 7.6235165e-07, 5.0209233e-06, 7.799939e-06, 1.937166e-06, 9.611823e-07, 1.614425e-06, 1.8713204e-07, 2.9885357e-06, 4.8278425e-06, 2.0012436e-05, 7.2182515e-06, 4.359301e-06, 1.574922e-05, 2.238073e-06, 1.7358202e-06, 1.3882881e-06, 2.0444086e-06, 2.4902007e-05, 8.8671567e-07, 9.5118907e-07, 5.4420452e-05, 4.4048585e-07, 1.533093e-07, 2.2423917e-07, 1.2845265e-07, 5.4538754e-07, 6.9552794e-07, 3.5553444e-07, 4.2179462e-07, 1.1077553e-06, 1.831457e-06, 1.143593e-06, 5.030103e-07, 9.824084e-07, 6.9452412e-06, 1.0099182e-06, 3.9539648e-07, 8.06763e-08, 4.0702827e-07, 5.923435e-07, 5.3197714e-06, 1.4797711e-06, 2.7858605e-06, 1.0203154e-06, 1.270072e-06, 1.7932915e-06, 1.1223898e-06, 1.32488985e-05, 2.3145172e-07, 2.5678237e-06, 6.031348e-07, 7.921732e-07, 2.581308e-06, 3.6643971e-06, 7.416457e-07, 9.522029e-07, 7.175475e-07, 1.1451864e-06, 2.0917491e-06, 9.24253e-06, 3.8683064e-07, 1.0389535e-05, 7.724614e-07, 7.472543e-07, 5.071702e-07, 1.6446974e-06, 1.1704658e-06, 3.0614149e-06, 1.409079e-06, 8.466847e-07, 1.8017014e-06, 3.507826e-07, 2.1284595e-06, 8.698909e-07, 1.224563e-06, 1.7106437e-06, 5.696064e-06, 4.514067e-07, 1.1763666e-06, 3.3413244e-06, 3.4128946e-06, 5.175482e-07, 3.714022e-06, 2.2927e-07, 2.316875e-06, 3.377438e-07, 2.3052058e-07, 2.61643e-06, 6.701335e-07, 8.474925e-07, 2.7116255e-07, 6.2081126e-06, 1.5719688e-07, 1.4582722e-06, 6.9947583e-07, 1.865066e-06, 3.862484e-06, 2.6300042e-06, 3.1585807e-07, 5.6471627e-06, 2.8117404e-06, 3.3101567e-06, 4.781126e-06, 5.332083e-06, 4.262747e-06, 3.4367472e-06, 1.5752081e-06, 9.933036e-07, 4.4076637e-06, 3.7497364e-06, 1.7161192e-06, 2.5797578e-06, 1.5539875e-06, 1.019196e-06, 1.471617e-06, 2.4699254e-06, 2.445008e-06, 7.9564916e-07, 9.740301e-07, 3.635142e-06, 1.5163473e-06, 1.1429203e-06, 3.060701e-07, 2.6062207e-07, 1.1011417e-06, 1.8995628e-07, 1.8400178e-06, 2.40029e-07, 1.1530976e-06, 4.4667553e-07, 1.499923e-07, 1.0562828e-06, 2.805113e-07, 3.573091e-07, 4.562204e-06, 9.904574e-07, 4.4645708e-07, 8.6095275e-07, 3.799917e-07, 5.7280236e-06, 1.5454673e-06, 8.292788e-07, 1.09758175e-05, 1.1667296e-05, 9.930498e-07, 8.749859e-06, 5.806174e-05, 1.8263291e-06, 2.6493578e-06, 2.9941327e-06, 1.7455774e-05, 1.29098025e-05, 5.3980243e-06, 6.8297766e-08, 2.1631668e-07, 6.335388e-06, 1.089367e-06, 2.0598577e-06, 4.8624817e-05, 1.12490325e-05, 9.968333e-06, 1.8108658e-06, 3.7059235e-06, 4.596375e-05, 1.0402488e-06, 4.026173e-06, 1.0984137e-06, 2.3183823e-06, 0.00011660683, 8.382035e-06, 1.2012907e-07, 2.1115221e-07, 1.6709426e-07, 2.803805e-07, 9.1352774e-08, 8.547764e-06, 8.3231095e-07, 7.207249e-07, 4.964948e-07, 4.7260255e-07, 1.924433e-06, 5.233134e-07, 3.1484038e-07, 8.789158e-08, 3.3203023e-06, 1.5410322e-07, 7.599468e-07, 6.5865486e-08, 2.2502013e-06, 1.9028707e-06, 2.0829236e-07, 1.9414686e-07, 1.737569e-08, 4.187208e-08, 2.133833e-08, 5.217957e-08, 7.756315e-08, 0.00028948556, 3.2254661e-06, 0.00045745773, 3.6141873e-06, 3.6879846e-06, 4.3864155e-08, 1.1546421e-07, 6.098287e-06, 4.7297253e-06, 7.4544114e-06, 2.3579938e-05, 1.7185849e-07, 8.238342e-09, 6.6577663e-06, 7.1307113e-07, 6.6831553e-06, 4.188079e-06, 1.1908708e-05, 2.2311808e-06, 2.3669925e-05, 5.633239e-07, 2.5464924e-06, 3.673358e-06, 9.228042e-06, 3.7669442e-06, 2.5243994e-06, 3.923309e-06, 1.5492952e-06, 5.415315e-07, 1.2985793e-05, 7.727771e-06, 1.1638047e-05, 6.415278e-06, 0.000117049596, 1.844425e-05, 3.0543357e-05, 7.667886e-05, 4.6682158e-06, 5.8063925e-08, 3.4565298e-07, 1.3907674e-06, 3.9034413e-07, 2.3856628e-06, 1.3961561e-06, 1.0203523e-06, 9.6493195e-06, 1.916817e-06, 5.992382e-06, 1.3963438e-06, 1.6447373e-07, 1.1997806e-05, 3.522568e-06, 7.992761e-07, 2.7826318e-06, 2.9383807e-06, 1.2469625e-06, 6.400709e-06, 8.6687646e-07, 1.3604218e-05, 3.653841e-06, 3.270486e-07, 2.1591251e-07, 7.4862564e-06, 0.0022334387, 2.6425723e-06, 8.8534085e-07, 3.9918265e-07, 1.4015102e-05, 6.251236e-05, 3.7113346e-06, 4.424491e-05, 5.257468e-08, 2.3197601e-06, 8.55421e-09, 6.37598e-08, 1.7909628e-07, 8.214551e-08, 3.2871288e-08, 6.140439e-08, 1.6078678e-07, 1.8680272e-07, 5.777686e-06, 3.2538915e-07, 8.600066e-07, 2.7519786e-08, 1.1244575e-06, 4.4513232e-07, 1.3912985e-07, 2.2559327e-07, 7.5065955e-08, 2.1787962e-07, 6.5873184e-07, 1.6194141e-07, 4.2486533e-07, 9.449337e-08, 4.586137e-07, 6.8615904e-08, 3.4233057e-08, 2.3015714e-08, 8.529465e-08, 1.1009139e-06, 2.085397e-06, 1.1364241e-06, 5.969457e-08, 5.854545e-08, 1.2437502e-07, 2.8979143e-06, 3.7183227e-07, 6.1114713e-07, 1.8259102e-07, 5.007582e-06, 2.0794462e-07, 7.3661596e-08, 1.2289431e-06, 4.696466e-07, 8.236131e-08, 3.7003272e-07, 9.213571e-07, 6.144086e-07, 2.8531753e-07, 1.6841451e-06, 3.648355e-07, 3.489435e-07, 2.1981393e-07, 3.489891e-07, 6.616541e-08, 1.7107e-07, 5.8820905e-08, 2.657458e-06, 6.7280183e-07, 7.9155576e-08, 3.4055512e-07, 5.2415576e-08, 1.9475921e-05, 2.0859183e-08, 5.2399713e-07, 3.550162e-06, 6.954769e-07, 1.5623053e-07, 4.141466e-08, 7.0278865e-08, 1.1458085e-07, 1.1404557e-07, 3.500317e-07, 5.067037e-07, 2.6709915e-07, 6.325267e-07, 2.1115885e-07, 6.933021e-07, 4.481367e-08, 7.1330064e-08, 4.751686e-07, 6.4066296e-07, 2.3656271e-07, 1.1212038e-07, 5.1450616e-08, 4.898539e-07, 3.9345363e-07, 7.469027e-08, 2.8215636e-08, 1.4027972e-06, 1.5045851e-05, 5.45923e-07, 3.3606793e-07, 6.793033e-07, 3.2025406e-07, 5.567628e-08, 5.144404e-08, 4.7856094e-08, 8.7845734e-08, 2.304809e-08, 3.320628e-08, 1.531284e-07, 8.455002e-07, 4.030141e-08, 5.826353e-07, 2.0933905e-07, 2.1854659e-07, 1.8154402e-07, 8.5039017e-07, 9.850174e-07, 4.691499e-08, 2.9529717e-08, 5.921111e-06, 1.4848057e-07, 5.5719124e-07, 3.0771523e-07, 7.578906e-07, 1.2629686e-06, 4.9141647e-08, 7.4479624e-07, 1.2221409e-06, 1.6379033e-07, 3.4507277e-08, 3.4415274e-07, 1.2394273e-07, 2.48148e-06, 3.051856e-08, 7.7588567e-07, 3.814518e-08, 1.1678488e-08, 3.970682e-07, 3.907356e-07, 6.9163754e-08, 3.2372859e-07, 4.107152e-08, 1.32908e-07, 7.114948e-08, 1.3805939e-07, 4.362909e-07, 1.7327786e-06, 1.0840065e-07, 4.287437e-06, 1.3406026e-06, 2.7652288e-07, 5.110169e-07, 2.6476766e-07, 3.4128325e-06, 7.380665e-08, 2.5400274e-07, 6.9870474e-08, 5.537498e-08, 4.607658e-07, 1.6728343e-08, 1.2180828e-07, 3.1195963e-08, 3.9396355e-07, 2.7166365e-07, 1.8899802e-07, 1.434787e-07, 5.057714e-07, 1.0786199e-06, 9.368018e-08, 4.7229683e-08, 8.0803005e-07, 5.0072924e-07, 2.4026627e-07, 1.9182908e-08, 8.188763e-08, 1.274422e-07, 2.8407675e-07, 2.4712435e-07, 1.4823198e-07, 5.0134037e-07, 7.983131e-07, 1.8063096e-07, 7.675681e-06, 4.8255936e-07, 8.072598e-07, 1.6763661e-07, 1.07333484e-07, 6.337291e-08, 1.5299548e-08, 1.586752e-07, 2.2856735e-08, 1.3107238e-07, 3.1036965e-08, 4.9698727e-08, 5.08264e-07, 1.4144146e-06, 1.758415e-06, 2.6504657e-07, 9.582741e-08, 5.6582618e-08, 2.8249872e-07, 1.8841897e-08, 8.451918e-08, 1.1465327e-06, 6.435631e-07, 7.6025344e-07, 5.33686e-08, 1.4108464e-07, 4.900749e-07, 1.1835917e-05, 1.6986917e-08, 5.604835e-08, 3.662114e-07, 6.067581e-08, 2.396404e-07, 4.516629e-07, 4.7011735e-08, 3.1379034e-07, 2.5050101e-06, 3.2410483e-07, 1.0516861e-06, 1.4487472e-07, 3.085546e-08, 4.3942286e-08, 4.497369e-07, 4.5126813e-05, 5.931131e-08, 3.7003736e-07, 2.668337e-08, 8.2547594e-08, 7.860159e-07, 1.1476485e-06, 3.1438208e-06, 7.495808e-08, 2.8493167e-07, 1.9044575e-06, 1.75579e-07, 2.7747865e-07, 1.8837882e-07, 3.0251408e-07, 5.71522e-07, 3.9703924e-08, 2.5046242e-07, 7.552639e-08, 4.5776423e-07, 9.707653e-08, 1.4215377e-07, 1.4616903e-06, 1.1546075e-06, 2.3138962e-06, 1.7347833e-07, 4.9520267e-08, 3.0973612e-07, 1.7835579e-06, 3.6174768e-08, 6.2489794e-06, 2.0208286e-07, 7.262567e-08, 3.0372778e-06, 1.3853102e-07, 2.5439329e-07, 3.3290007e-07, 3.0902535e-07, 5.590861e-07, 1.06721984e-07, 6.7565156e-07, 2.1733736e-07, 3.322144e-07, 2.0757233e-07, 7.719017e-07, 8.4283094e-08, 1.0376899e-07, 1.9896996e-07, 2.2272106e-07, 7.127427e-07, 9.009337e-07, 3.2640187e-08, 2.0750068e-07, 4.8350348e-08, 7.0408504e-07, 5.6114324e-05, 7.5532836e-07, 8.887479e-08, 7.085125e-07, 1.1560844e-07, 3.8795744e-07, 3.976793e-06, 4.6256346e-07, 5.975669e-07, 3.0750516e-07, 7.345843e-08, 2.4090642e-07, 2.1353691e-07, 3.3497807e-07, 1.00545954e-07, 4.805937e-07, 8.9473104e-09, 1.5028067e-07, 2.299759e-08, 2.2111642e-07, 8.5874444e-07, 1.4600192e-07, 2.5260093e-07, 4.1094324e-08, 3.6426795e-06, 6.740626e-07, 1.0925928e-07, 1.2064076e-07, 2.6033217e-07, 2.3739936e-07, 4.02779e-06, 6.986927e-08, 4.927631e-07, 2.0377253e-07, 4.859285e-08, 6.001323e-07, 4.019476e-07, 6.5550914e-08, 3.255296e-08, 4.502512e-08, 4.5910076e-07, 1.9682864e-07, 1.5400917e-07, 2.1057043e-07, 3.4854838e-07, 2.8710346e-07, 7.8076866e-08, 3.7246292e-07, 8.849859e-08, 2.2556638e-07, 8.8687415e-08, 1.4012883e-07, 2.5905608e-08, 4.2669419e-07, 3.5269767e-07, 9.342652e-08, 1.16506385e-07, 1.8241785e-07, 2.592098e-08, 2.0602221e-07, 5.343529e-07, 4.3310727e-08, 3.1328245e-06, 3.7680095e-07, 1.6470683e-07, 1.3352549e-08, 4.713009e-07, 4.3408545e-07, 3.0829065e-07, 4.148033e-07, 3.5043252e-07, 4.1328406e-08, 1.039676e-07, 2.0944228e-07, 3.6792827e-08, 1.9913675e-06, 7.9480465e-08, 3.3763465e-07, 3.8642725e-07, 3.1329128e-07, 1.0457145e-06, 5.3874476e-07, 5.3555846e-07, 2.299623e-08, 2.9225612e-07, 1.2956369e-06, 5.7133235e-07, 1.7188988e-06, 1.951809e-07, 3.692544e-07, 3.600926e-07, 5.115637e-08, 2.873042e-07, 3.5520603e-07, 4.8817757e-08, 3.2432975e-08, 2.3311536e-07, 2.5904757e-07, 5.7130944e-07, 5.5044434e-06, 5.5146495e-07, 3.4003781e-07, 1.2421808e-07, 1.0033226e-05, 1.1899007e-06, 1.1852086e-06, 2.8190502e-07, 3.9343806e-08, 6.526603e-08, 2.8353597e-07, 3.716835e-08, 7.739528e-08, 1.2510313e-06, 1.1097918e-06, 1.4137441e-08, 5.196871e-08, 1.2651475e-07, 4.3768156e-07, 5.16784e-08, 1.8941446e-07, 3.4604793e-06, 2.1872393e-06, 1.603818e-06, 2.8407675e-07, 7.211409e-07, 4.7171773e-07, 7.0506945e-07, 2.37784e-08, 2.6139796e-07, 1.0857099e-08, 4.7737427e-05, 1.7715913e-06, 2.4736508e-07, 5.6247893e-07, 1.7013748e-06, 5.3935753e-07, 6.814604e-08, 8.114711e-07, 2.0709443e-07, 2.1679611e-08, 5.5730574e-08, 1.0388673e-07, 2.0995584e-07, 3.1774303e-07, 1.2546808e-07, 1.5703132e-07, 9.148216e-08, 1.1042813e-07, 1.16445406e-07, 2.0244965e-08, 1.4764365e-07, 9.9541595e-08, 2.9003402e-07, 5.2911406e-07, 3.6716312e-06, 1.8401056e-06, 2.6981215e-07, 2.3241279e-07, 4.6320945e-08, 6.2793663e-07, 4.0248636e-08, 7.398446e-08, 7.438862e-07, 8.786995e-08, 2.891773e-07, 7.3661136e-07, 6.0076565e-07, 2.798309e-06, 2.8826727e-07, 2.20146e-07, 1.3246057e-07, 4.587592e-06, 1.1943971e-06, 4.5323137e-07, 6.210182e-07, 7.987308e-08, 1.1633191e-06, 2.6830708e-08, 2.667916e-07, 9.157102e-08, 2.2695318e-07, 2.539181e-06, 2.993917e-07, 1.3205372e-08, 5.2766785e-07, 4.7152092e-08, 4.660542e-08, 4.905271e-07, 1.408878e-07, 4.298302e-08, 6.095104e-07, 1.6381455e-07, 6.7567737e-07, 1.3618406e-07, 2.9217654e-08, 6.0456813e-07, 3.481089e-07, 2.199539e-06, 3.1379585e-08, 4.0919085e-06, 1.3511705e-07, 1.7438792e-06, 5.164891e-07, 6.14058e-08, 1.692717e-07, 6.0094214e-07, 4.616263e-08, 1.7136415e-08, 1.8789655e-07, 1.664482e-05, 1.109561e-08, 2.0408984e-06, 5.144324e-07, 2.2324717e-07, 7.9635775e-08, 2.3219015e-07, 1.4640878e-07, 1.3904564e-07, 6.791909e-08, 5.3922327e-07, 1.7450128e-07, 2.2044287e-07, 8.2480156e-08, 7.791018e-08, 3.608704e-08, 1.4282757e-06, 4.4595026e-07, 6.1917463e-06, 1.2274731e-06, 6.541434e-08, 5.510292e-07, 6.2777127e-07, 9.852104e-08, 9.2759166e-08, 1.0801202e-07, 5.1773508e-08, 7.03263e-07, 5.48382e-08, 9.163828e-08, 1.8281544e-07, 2.2309499e-07, 1.5063874e-06, 2.9042156e-05, 8.7185555e-08, 3.4663535e-07, 2.3967124e-07, 6.1186444e-07, 2.6795408e-07, 1.7159874e-07, 2.5335256e-08, 1.0267665e-06, 1.2672062e-07, 1.19088995e-07, 9.9702916e-08, 6.4066995e-08, 1.2536097e-08, 9.928059e-08, 7.231101e-08, 2.8704923e-07, 6.1002176e-08, 2.986067e-07, 1.2906129e-06, 5.925127e-08, 1.8925866e-07, 9.22468e-08, 1.4960415e-07, 1.06911564e-07, 4.2838074e-08, 3.6908082e-07, 1.1071727e-07, 3.8998283e-07, 1.568034e-07, 3.122595e-07, 8.165866e-08, 1.6952727e-07, 1.9333584e-08, 1.3010535e-07, 3.7023185e-07, 3.297176e-08, 6.786729e-08, 2.6440307e-07, 4.7650342e-08, 2.6313728e-07, 1.0631267e-06, 7.3527224e-07, 1.0764516e-06, 7.899638e-08, 4.3475617e-07, 8.123929e-08, 9.337006e-08, 5.917117e-07, 8.175583e-08, 2.3680897e-07, 1.224908e-07, 1.1772941e-07, 7.237772e-08, 1.7921716e-08, 2.1354629e-07, 1.7915333e-07, 4.6141595e-06, 1.3726662e-06, 4.41399e-06, 1.4019968e-05, 5.0816493e-06, 3.158545e-05, 1.0377246e-05, 0.00013469988, 0.00012541312, 8.943293e-07, 3.5171972e-06, 4.143929e-07, 5.997899e-08, 3.7551363e-06, 2.5072768e-07, 4.117451e-08, 1.9050432e-07, 3.9115857e-06, 2.6173373e-07, 2.9716453e-08, 1.8971113e-07, 2.3804073e-07, 5.234724e-08, 1.01332226e-07, 4.7450158e-07, 1.5748033e-07, 6.902779e-07, 8.266525e-07, 3.983048e-06, 1.5279669e-07]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>58</td>\n",
+       "        <td>0.14084539</td>\n",
        "        <td>2</td>\n",
-       "        <td>[1.5398843e-09, 2.3107729e-08, 7.636333e-08, 2.2277517e-07, 1.7589974e-07, 5.438902e-08, 9.2527934e-08, 4.70354e-08, 9.018498e-09, 1.8840204e-08, 2.2168626e-07, 5.591516e-09, 3.5177464e-08, 4.6703323e-07, 4.6959094e-09, 4.3828187e-08, 1.6602337e-08, 2.8010957e-08, 1.4341178e-07, 8.532091e-08, 5.8298728e-08, 8.00204e-09, 3.6414684e-07, 3.851048e-08, 2.0718534e-08, 2.6480208e-08, 3.3010747e-09, 4.47217e-08, 2.8583589e-08, 6.513382e-10, 1.9336234e-08, 5.272263e-10, 2.404317e-09, 1.12124745e-07, 7.488657e-07, 6.5691674e-09, 4.623265e-09, 1.6595015e-09, 1.9597215e-08, 1.6531517e-07, 1.6838877e-08, 8.48814e-09, 2.9826325e-08, 4.543374e-09, 2.3641775e-08, 4.0592163e-08, 6.912111e-09, 3.3884773e-09, 1.042833e-08, 7.361764e-09, 4.2130747e-08, 1.6772022e-07, 1.4776109e-09, 5.525266e-09, 4.0264826e-08, 1.278795e-09, 1.4832867e-07, 4.2117554e-09, 1.3611935e-07, 3.0520868e-09, 6.4132884e-08, 1.4925305e-08, 5.7810297e-08, 6.520122e-09, 2.525428e-09, 2.0901005e-07, 3.504386e-08, 2.659847e-07, 1.3491556e-08, 8.361713e-08, 7.3479924e-08, 1.3090515e-08, 4.685739e-08, 1.2734573e-07, 1.3352158e-07, 9.6114135e-08, 3.8392837e-08, 1.2469758e-07, 5.4070338e-08, 1.2920589e-07, 3.2127724e-07, 1.414701e-05, 3.858608e-07, 1.1320818e-08, 1.5107693e-08, 3.15158e-08, 3.0302337e-07, 3.696147e-09, 2.5316362e-08, 1.6199753e-08, 5.834453e-09, 6.6545938e-09, 7.8418395e-08, 2.9896134e-09, 9.421624e-09, 2.3791766e-09, 2.7823246e-09, 1.8130814e-09, 4.316991e-09, 1.6660833e-08, 1.0815284e-08, 1.2120864e-08, 7.508208e-09, 2.4426359e-09, 7.537742e-08, 2.1063304e-09, 3.549428e-09, 4.384449e-08, 7.2657627e-09, 3.1592776e-08, 2.4405764e-10, 1.6091322e-08, 1.8533458e-06, 2.4308275e-08, 3.7633074e-09, 1.6087233e-09, 2.277431e-08, 5.8490266e-09, 4.0977223e-08, 3.8039954e-07, 1.3494928e-08, 1.781187e-07, 6.407897e-08, 5.389317e-07, 8.723511e-08, 7.2745024e-08, 2.6316968e-08, 6.543285e-08, 7.232058e-08, 4.530396e-08, 1.2370523e-07, 1.0784221e-08, 4.778658e-08, 8.2433225e-09, 1.152689e-07, 1.4541708e-08, 4.6979163e-09, 1.9269741e-08, 6.921267e-09, 2.708216e-07, 5.5321866e-08, 1.6940814e-07, 2.0328073e-08, 2.275105e-07, 7.400611e-08, 4.040531e-06, 5.9436662e-08, 3.5894548e-08, 5.200469e-08, 1.9099544e-09, 2.3085869e-07, 6.950096e-08, 9.922675e-09, 8.168692e-08, 7.0257973e-09, 8.1655614e-08, 1.0280881e-06, 3.198189e-08, 1.8011564e-08, 1.1670971e-06, 4.824007e-08, 4.8371714e-07, 2.3133298e-07, 9.200232e-07, 2.2442546e-07, 4.7174932e-08, 1.1280227e-07, 2.7898892e-07, 8.8787215e-07, 9.2161355e-07, 2.3470897e-05, 1.7707903e-06, 6.7424407e-06, 6.0599414e-06, 2.8161035e-06, 9.434914e-09, 1.0188343e-06, 5.5756454e-06, 3.1946593e-06, 8.136942e-07, 1.1003609e-07, 4.576751e-07, 3.3645716e-07, 2.740537e-07, 5.4528084e-07, 5.777307e-07, 2.9695158e-07, 7.171372e-08, 2.6257868e-07, 1.6330781e-06, 9.9922396e-08, 6.695125e-07, 8.737203e-07, 8.1306325e-08, 5.515203e-08, 6.864878e-07, 7.5751146e-08, 1.1799731e-06, 1.4831099e-07, 2.772437e-07, 1.7763999e-07, 2.098404e-08, 1.940811e-07, 2.6700523e-06, 1.0012253e-07, 6.0266035e-07, 4.6589884e-07, 6.262852e-07, 3.776922e-06, 7.038604e-06, 2.2661327e-06, 1.1063678e-06, 6.118894e-07, 2.5951251e-06, 2.1583949e-06, 2.2275087e-06, 5.9048972e-08, 9.353487e-07, 8.8109886e-07, 1.6752422e-07, 4.7095458e-08, 1.1280829e-07, 6.4681115e-07, 1.4628699e-06, 1.5081841e-06, 6.317585e-07, 1.1813028e-06, 1.900376e-07, 2.4231375e-07, 2.6553482e-07, 4.894391e-07, 2.6518939e-07, 1.5399695e-07, 8.8166956e-07, 6.533642e-07, 4.457954e-07, 1.8340435e-06, 1.855212e-07, 4.893554e-06, 2.3288433e-06, 2.3774019e-06, 5.201625e-06, 7.6095256e-07, 2.3237685e-07, 1.4151934e-07, 8.257309e-08, 2.1411304e-06, 2.2388014e-05, 1.3785373e-05, 5.831647e-06, 1.3944214e-05, 2.2903537e-06, 4.6498332e-08, 4.6375638e-07, 6.242959e-07, 2.3912223e-06, 1.4558304e-06, 9.471192e-07, 5.5140313e-06, 9.166841e-08, 1.2893312e-07, 3.9749816e-07, 4.1223927e-08, 3.76367e-07, 1.4717683e-06, 9.8032444e-08, 3.420696e-07, 1.1348545e-06, 1.233211e-07, 6.2267026e-07, 1.1291587e-06, 6.029464e-08, 4.1942053e-06, 3.126346e-07, 2.5256027e-08, 3.411787e-08, 1.6709652e-08, 1.0040704e-06, 2.5781976e-08, 9.700623e-07, 1.6046155e-07, 2.0606766e-08, 2.4433362e-08, 4.4592547e-09, 1.1156811e-08, 1.8641177e-08, 1.579223e-07, 2.0955612e-07, 2.9641933e-09, 4.2151328e-08, 8.4548657e-10, 1.2786364e-09, 5.8412592e-08, 7.3222175e-09, 3.4796508e-07, 6.028762e-08, 2.8341196e-06, 2.5817788e-09, 1.6543543e-08, 5.1597104e-09, 6.515199e-09, 2.4591106e-08, 1.10343e-08, 2.3506432e-08, 2.5722677e-09, 9.193776e-09, 7.1656943e-09, 6.2450964e-09, 4.8909826e-08, 2.9583264e-08, 1.0357064e-07, 7.1164735e-08, 3.6447915e-08, 3.097284e-07, 4.2578133e-08, 1.8602746e-08, 4.852659e-09, 4.6380855e-09, 8.3475065e-08, 7.4244093e-07, 2.558261e-08, 3.2909315e-07, 7.988803e-09, 2.3116634e-08, 3.85773e-08, 1.5388956e-08, 7.1112463e-09, 1.4074487e-07, 2.9982766e-07, 1.564242e-08, 1.0789998e-07, 1.299596e-07, 1.7822981e-09, 2.1977655e-09, 3.8496357e-09, 1.3551873e-09, 5.3080055e-07, 5.1470135e-08, 8.4123347e-10, 7.115103e-08, 1.0199323e-07, 8.621156e-08, 3.209869e-07, 1.0228442e-08, 3.5695646e-09, 1.6592921e-06, 5.754166e-08, 4.5394786e-06, 6.6713767e-07, 8.642847e-06, 8.179076e-06, 8.936807e-08, 7.0527335e-08, 9.755931e-08, 1.8077302e-06, 2.6266648e-06, 1.8360647e-06, 1.18492515e-07, 1.0602841e-08, 2.8182551e-08, 8.855424e-09, 3.9218904e-08, 9.4692405e-09, 1.8187903e-08, 6.0960176e-10, 2.8660767e-09, 2.7049174e-09, 2.9135108e-09, 2.0202557e-09, 3.7000687e-09, 1.3453749e-09, 7.608665e-08, 1.4827498e-08, 1.8008379e-07, 3.214444e-08, 3.3369876e-09, 1.1487082e-08, 1.0620547e-09, 1.0122127e-08, 2.7668134e-09, 1.38777265e-08, 5.459645e-09, 9.779465e-09, 1.4373716e-08, 1.4275604e-08, 3.1167882e-08, 1.6579142e-08, 1.6735798e-09, 1.7655769e-09, 2.4233465e-08, 7.761856e-08, 8.390994e-07, 8.398905e-09, 1.0605148e-08, 3.585211e-07, 5.846406e-07, 2.773881e-07, 9.707507e-09, 1.7990827e-08, 7.335421e-07, 5.715663e-08, 1.6425967e-06, 2.6860542e-07, 1.945019e-07, 4.4494698e-08, 2.3012471e-07, 1.0821924e-07, 2.522256e-07, 5.9017445e-08, 4.4775834e-07, 2.4033554e-06, 5.852322e-08, 3.763297e-07, 1.2767528e-05, 3.413865e-07, 2.4368111e-08, 1.2132007e-07, 2.718393e-07, 4.8032852e-08, 4.3237666e-08, 3.820392e-07, 2.2197873e-06, 3.518223e-08, 6.4883015e-09, 1.7396843e-08, 1.4981359e-05, 9.289176e-08, 5.7502575e-07, 2.5261088e-06, 7.218663e-08, 1.6187707e-08, 2.5262677e-08, 1.4110595e-06, 2.7774984e-06, 5.1647742e-08, 1.7309242e-07, 6.657365e-08, 1.1532635e-06, 2.5521492e-08, 1.2916955e-07, 1.5800223e-06, 5.573535e-08, 6.0709255e-08, 2.8454885e-08, 2.6240545e-07, 2.2590486e-06, 6.349486e-08, 2.5054753e-06, 1.0332632e-06, 7.5764866e-08, 3.0934818e-05, 1.0285893e-06, 4.9956363e-08, 3.806668e-08, 1.12557945e-08, 4.212406e-09, 9.062275e-06, 2.1683526e-07, 7.8610384e-08, 3.9823655e-07, 1.2365699e-05, 2.6495582e-07, 7.8097855e-06, 1.8755473e-07, 4.612106e-09, 1.1201466e-07, 1.1676837e-08, 3.508029e-09, 9.1218254e-08, 7.643896e-08, 1.5362502e-08, 9.381413e-07, 2.6301302e-07, 6.57294e-09, 7.180635e-07, 3.8983737e-07, 1.554282e-08, 8.0090764e-08, 8.545088e-08, 1.569008e-08, 5.355029e-08, 2.2874363e-07, 3.753023e-08, 3.225784e-07, 1.6325798e-06, 3.007842e-09, 3.2840123e-07, 3.130718e-08, 5.363401e-08, 9.742059e-08, 3.938637e-08, 1.09314815e-05, 2.7909737e-08, 1.9837347e-07, 1.0354215e-06, 5.9552523e-08, 1.6291214e-07, 2.0342968e-06, 2.5287791e-08, 1.59485e-07, 9.795091e-07, 3.1564498e-06, 1.7559394e-08, 1.6565266e-08, 2.620732e-08, 4.9752394e-08, 1.6515578e-08, 1.9081748e-08, 7.422872e-09, 3.3019376e-09, 1.7074969e-07, 1.0950456e-08, 8.516746e-09, 1.1135265e-06, 1.636078e-07, 1.3219143e-06, 3.5221102e-08, 4.5833468e-07, 1.7809934e-07, 1.052475e-07, 2.4009454e-08, 4.471895e-09, 3.8112885e-06, 4.0083058e-05, 1.1635215e-07, 5.101486e-07, 3.6760483e-08, 1.9992733e-08, 9.947765e-09, 3.0854395e-08, 1.118227e-08, 4.7650698e-08, 8.499971e-08, 2.4238595e-08, 1.4947384e-08, 3.6674674e-09, 6.124752e-08, 0.00038468122, 1.8884897e-07, 2.5177487e-08, 2.9864592e-07, 1.9823315e-07, 1.1138956e-07, 3.4749004e-08, 4.807366e-07, 1.1343858e-08, 1.24109e-07, 3.708061e-08, 1.3761401e-07, 7.507371e-07, 8.474337e-10, 3.2129975e-08, 1.6138574e-07, 4.289849e-08, 4.1769333e-08, 1.524812e-07, 4.4421853e-08, 3.3816161e-06, 1.883187e-06, 8.6855124e-07, 3.4589572e-08, 1.10152946e-07, 2.6198538e-07, 1.5651674e-08, 2.0048878e-07, 1.0421765e-07, 4.6393686e-07, 7.667737e-09, 3.3911301e-06, 5.303072e-07, 6.010618e-07, 9.7195084e-08, 4.687357e-08, 1.2324756e-08, 3.2322134e-07, 7.846987e-08, 2.5114367e-08, 7.899244e-08, 2.5696616e-07, 1.01712125e-07, 5.3800395e-08, 4.404471e-09, 7.0128237e-09, 4.1187302e-08, 2.6257606e-08, 7.664451e-08, 3.842652e-07, 6.999354e-07, 5.2077125e-09, 1.8243146e-08, 8.38386e-09, 1.0335663e-07, 6.9752186e-09, 5.2016794e-08, 8.570752e-07, 1.0925754e-08, 3.7993168e-06, 2.1593397e-08, 8.062468e-08, 1.6209674e-08, 9.4739295e-07, 3.8672835e-07, 2.2282576e-07, 4.2491993e-07, 4.789252e-08, 5.8676232e-08, 5.8623435e-08, 1.3501982e-08, 2.2437293e-06, 1.702915e-06, 9.943773e-07, 3.649515e-08, 1.6232762e-07, 3.140646e-08, 1.11932536e-07, 3.590689e-07, 1.3766297e-07, 5.5616647e-07, 5.2407668e-08, 2.748883e-08, 9.189626e-08, 2.5330624e-07, 6.336275e-08, 4.12668e-08, 6.258344e-09, 1.3503933e-07, 1.733054e-08, 1.9494284e-08, 1.6185105e-07, 1.3785838e-08, 6.109794e-08, 1.8560323e-08, 1.6609394e-07, 2.476262e-08, 2.6778312e-07, 4.351084e-08, 3.084329e-07, 7.817878e-06, 1.8975626e-06, 2.0993969e-06, 2.6453353e-08, 2.2118513e-07, 8.731006e-07, 1.6439921e-07, 2.3696006e-07, 1.787288e-06, 4.8213653e-05, 8.37574e-09, 6.81651e-09, 2.8432564e-06, 2.704404e-07, 4.226339e-08, 9.407418e-06, 9.6293405e-08, 3.725218e-07, 1.2643505e-06, 9.604207e-07, 7.47385e-06, 3.566186e-07, 9.538723e-09, 1.6844483e-06, 7.3710655e-09, 5.2484666e-09, 8.958981e-07, 7.355466e-08, 3.6778154e-08, 6.467584e-08, 1.0149527e-07, 1.1401684e-07, 2.0010434e-08, 2.130884e-08, 3.8332364e-06, 0.0007202051, 8.997154e-10, 3.1479754e-08, 8.319596e-08, 2.7916498e-08, 1.4024812e-06, 8.000587e-08, 6.105344e-08, 1.6910072e-08, 2.397463e-08, 2.5698992e-07, 8.2904137e-07, 2.7703791e-08, 2.4609814e-09, 8.351855e-10, 5.907249e-09, 2.3557195e-08, 5.1269083e-08, 3.1052042e-07, 1.9230353e-07, 1.3536194e-07, 5.51488e-07, 1.9878282e-08, 7.9125144e-08, 8.0264836e-05, 2.3167297e-07, 9.360291e-08, 3.483506e-07, 1.8976078e-06, 9.991526e-06, 7.9052325e-07, 2.5110057e-06, 1.2902883e-05, 1.06492195e-07, 1.8583303e-07, 5.769002e-08, 3.6822023e-08, 1.5638037e-08, 1.6243138e-07, 1.0981206e-07, 1.0313105e-09, 2.2036936e-08, 1.6936184e-08, 5.209431e-07, 1.066118e-05, 3.8097987e-08, 6.077448e-08, 8.576721e-09, 2.7099158e-09, 6.393335e-07, 6.024069e-09, 8.597185e-07, 2.4906703e-07, 1.7457806e-08, 9.4648165e-09, 2.0813505e-07, 1.3174564e-06, 4.096353e-09, 4.0667996e-06, 8.2589827e-07, 3.04197e-08, 2.339224e-05, 3.6062332e-08, 1.1635309e-06, 4.037927e-08, 6.19253e-08, 3.770228e-08, 6.161791e-08, 1.5994248e-06, 6.009784e-08, 8.941896e-08, 1.1777902e-07, 2.4193057e-06, 9.981634e-09, 3.2275516e-06, 5.3249114e-08, 1.5546792e-08, 6.834216e-07, 3.9778675e-07, 4.532384e-08, 6.9061976e-08, 1.1572694e-08, 5.0055938e-08, 6.2338563e-07, 4.9324466e-08, 5.308013e-06, 3.0234658e-06, 1.6129864e-07, 7.341118e-07, 4.529463e-08, 4.721031e-08, 2.7191814e-07, 9.214177e-05, 3.4954944e-08, 5.892889e-09, 6.9181067e-07, 3.0443786e-08, 9.5250334e-07, 6.399774e-08, 1.8667293e-08, 4.4535962e-08, 7.128982e-09, 6.305315e-08, 1.3619449e-06, 8.321957e-07, 6.1739435e-08, 7.1391058e-09, 3.2976266e-07, 4.1574063e-07, 9.305059e-09, 1.4348497e-07, 1.12088614e-07, 1.717341e-07, 9.155937e-10, 8.49131e-08, 2.274656e-07, 1.31738735e-08, 6.4238044e-08, 2.5300033e-09, 4.735305e-07, 0.0016679094, 4.3191326e-07, 1.3948901e-08, 0.6965568, 0.00032454787, 3.5461414e-06, 3.661261e-09, 1.7207987e-07, 9.268941e-09, 2.3562737e-07, 0.0002525007, 8.555244e-05, 1.0181198e-09, 6.2870043e-07, 1.4091459e-06, 1.3009806e-06, 2.7303082e-07, 1.4669624e-09, 8.454194e-09, 1.8131272e-08, 2.4682942e-07, 1.1727215e-06, 1.1580163e-07, 4.3070994e-08, 1.063989e-07, 1.1100011e-08, 9.354254e-07, 1.5596758e-07, 1.1565727e-07, 1.0455736e-08, 1.5751957e-06, 7.423669e-08, 2.7763226e-06, 6.379263e-06, 8.358341e-08, 1.1284831e-07, 8.936478e-09, 2.7141138e-07, 1.1885099e-06, 3.5644707e-08, 4.211571e-05, 8.188094e-07, 4.756142e-07, 4.3325363e-07, 3.3440047e-06, 3.8567996e-06, 2.567217e-07, 2.7330253e-07, 5.7382164e-07, 1.7412797e-06, 1.52012235e-05, 3.6163758e-06, 4.464074e-08, 1.5914673e-06, 2.5722297e-08, 5.8252516e-07, 5.265234e-08, 8.234005e-09, 6.593523e-08, 7.2670247e-07, 6.157772e-08, 1.9472304e-06, 1.6877687e-08, 5.4864344e-08, 1.2159968e-07, 8.8625045e-09, 3.85879e-08, 1.18826236e-07, 5.029152e-09, 7.548831e-09, 4.8796755e-06, 3.368214e-06, 3.206473e-07, 2.0191665e-08, 7.773727e-07, 8.836705e-07, 2.781339e-08, 8.5779556e-07, 6.021054e-07, 5.5811405e-08, 0.00017171177, 1.3041937e-07, 1.03372e-07, 9.105134e-07, 1.9368134e-07, 3.4401072e-08, 6.993172e-10, 1.4681606e-05, 2.6645605e-06, 5.3564694e-08, 1.8430373e-07, 3.574854e-08, 4.3808715e-08, 1.1790208e-07, 2.903986e-08, 1.7527881e-07, 9.018778e-08, 5.909041e-07, 1.3226456e-06, 9.100276e-10, 9.4933505e-08, 1.7600302e-08, 1.237418e-07, 2.4791518e-06, 1.6265204e-08, 5.6816635e-08, 3.0899054e-07, 2.259528e-07, 1.2160329e-07, 1.7874806e-08, 1.1018447e-07, 8.6053426e-08, 1.6846097e-09, 1.3612078e-09, 4.0714268e-09, 3.1117708e-07, 1.8836668e-06, 2.4274023e-09, 1.6711957e-07, 1.2489967e-07, 5.7767593e-06, 0.00023884398, 4.883025e-07, 0.00018770229, 6.8687876e-07, 3.7221625e-08, 1.396456e-07, 9.718689e-07, 9.019608e-07, 3.1364755e-07, 1.7217104e-08, 4.8390746e-08, 7.264543e-09, 7.605233e-10, 1.654826e-10, 6.4277764e-08, 1.371566e-07, 9.566331e-08, 6.1089866e-09, 1.1865948e-08, 9.200144e-09, 2.860801e-09, 3.0422314e-08, 1.2621048e-08, 5.286336e-08, 6.646337e-09, 1.1934101e-09, 1.2263828e-09, 1.2506999e-09, 1.0561177e-09, 1.5004105e-09, 4.2410333e-09, 1.4444183e-08, 2.3039358e-10, 7.3044473e-09, 7.0201445e-09, 4.5222894e-08, 6.5248993e-09, 3.0367296e-08, 1.61907e-08, 2.812724e-08, 7.16595e-10, 2.4030102e-09, 1.4037422e-09, 2.8542007e-09, 6.8244588e-09, 7.394124e-07, 8.272793e-09, 9.1326186e-08, 3.4254477e-08, 3.938575e-09, 1.2116357e-08, 2.4269255e-09, 2.9024413e-08, 1.6277409e-07, 4.0866994e-09, 4.3306834e-08, 7.559003e-09, 0.29820144, 6.001986e-07, 4.3510576e-05, 6.434583e-08, 4.367675e-05, 6.187613e-06, 5.1900697e-06, 4.6378314e-06, 8.057671e-06, 5.7155186e-05, 1.4133632e-05, 1.1964084e-07, 1.1058091e-07, 4.345451e-07, 3.3152878e-07, 8.424061e-09, 8.0378465e-10, 4.2880637e-09, 4.369939e-08, 6.1343945e-07, 2.029015e-08, 3.3226485e-08, 7.760297e-09, 3.8018118e-09, 3.5106462e-09, 1.7808583e-08, 1.4098259e-08, 3.8490683e-10, 3.2085192e-09, 3.8055336e-07]</td>\n",
        "    </tr>\n",
        "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>49</td>\n",
+       "        <td>0.05819455</td>\n",
        "        <td>3</td>\n",
-       "        <td>[1.6689529e-07, 2.2206162e-07, 2.820838e-07, 9.411482e-08, 9.196814e-08, 6.835522e-08, 1.4712062e-07, 9.119155e-06, 6.7199594e-06, 1.9530382e-06, 2.4907507e-07, 6.704823e-07, 1.3618245e-07, 3.1312987e-07, 1.2618708e-07, 8.8433814e-07, 1.3651416e-07, 1.0374823e-06, 9.697097e-07, 3.1258924e-07, 1.154241e-07, 2.7880057e-07, 7.071258e-07, 1.1773246e-06, 2.7102183e-07, 8.7961396e-08, 5.8193404e-08, 2.5870187e-07, 3.1527037e-08, 2.715556e-07, 1.865584e-07, 3.0306205e-07, 1.0248689e-07, 2.2406402e-07, 2.4086475e-07, 8.182266e-08, 3.4661014e-07, 2.965118e-07, 1.560052e-07, 2.3101217e-07, 5.0157405e-07, 1.6106436e-07, 1.7701154e-07, 7.382814e-07, 4.017762e-07, 8.348868e-08, 4.505219e-07, 3.507288e-08, 1.9216787e-07, 2.1005312e-07, 1.7100543e-06, 1.2874328e-07, 1.7951021e-07, 1.1528065e-07, 3.0493317e-07, 8.817166e-07, 7.344486e-07, 8.2197954e-07, 4.4888705e-07, 6.063366e-07, 9.589398e-07, 3.6966567e-06, 1.7984804e-06, 3.5922747e-07, 4.3303217e-07, 1.4106764e-07, 3.4898633e-07, 2.1168786e-07, 2.1905065e-07, 7.523292e-08, 1.2124268e-07, 1.6527275e-07, 6.333906e-08, 5.592894e-08, 9.339264e-08, 3.3048057e-08, 2.849944e-07, 5.0003926e-08, 3.2996897e-07, 1.093866e-06, 4.7521535e-08, 6.467596e-08, 6.746868e-07, 7.208058e-07, 1.3252958e-06, 4.4775916e-07, 1.3629414e-06, 2.6028246e-07, 1.2213138e-07, 2.378058e-07, 4.0746293e-08, 4.4012336e-07, 1.5162593e-07, 1.3352037e-06, 1.0124106e-06, 1.385228e-07, 2.0494727e-07, 1.7858578e-05, 1.1512349e-06, 8.851452e-06, 2.1538575e-07, 6.265657e-08, 5.9420086e-07, 2.688158e-07, 6.4805367e-06, 5.182306e-07, 2.372954e-07, 2.2233793e-07, 3.245263e-07, 5.0860205e-08, 1.017551e-07, 8.684927e-08, 3.1345842e-06, 2.1760503e-07, 2.1266662e-07, 5.1490655e-08, 5.8833677e-08, 7.994336e-07, 1.437624e-06, 1.8312319e-07, 4.1118618e-08, 6.079955e-07, 7.317862e-07, 1.2565258e-07, 8.145708e-07, 7.439697e-07, 4.4777835e-07, 8.953791e-08, 2.082923e-07, 3.7194494e-07, 6.495325e-08, 1.15024555e-07, 4.6843113e-07, 1.9805561e-06, 1.128121e-06, 3.502328e-08, 7.75236e-07, 1.0817615e-06, 2.0329721e-06, 3.4939458e-07, 1.3838962e-06, 5.9606386e-08, 1.0386679e-07, 2.7677422e-07, 2.562633e-07, 3.75767e-07, 2.257235e-06, 1.7437878e-07, 3.344044e-07, 1.4814579e-08, 1.4589527e-07, 9.747049e-06, 4.343081e-05, 2.4861931e-06, 4.7536546e-06, 1.7077704e-05, 4.1258518e-05, 0.00033958646, 2.7045458e-05, 2.2274935e-06, 0.0007474787, 7.4541317e-06, 3.46901e-06, 1.7526618e-06, 5.131311e-06, 5.9422626e-07, 5.283553e-06, 2.3466607e-06, 1.1319079e-06, 0.0075451825, 6.501963e-06, 1.8894696e-05, 4.274101e-05, 8.023169e-05, 4.3800146e-06, 5.5521894e-05, 7.4542e-05, 5.1395902e-05, 3.269665e-07, 6.566881e-07, 1.1947602e-06, 3.175389e-06, 4.53225e-06, 2.0213977e-06, 1.9833658e-05, 7.5548055e-06, 7.107016e-05, 5.446052e-05, 0.0003789736, 2.3369696e-05, 6.262723e-06, 8.743196e-05, 1.7344008e-05, 0.00054024416, 5.6735547e-07, 1.3520279e-06, 9.357407e-06, 1.2883928e-06, 4.2506967e-06, 2.1557997e-05, 2.9111889e-05, 6.968902e-05, 4.0829773e-06, 6.0506145e-06, 1.7851306e-05, 2.1046308e-06, 9.482371e-07, 1.9215779e-05, 1.5350582e-06, 9.981025e-07, 1.2606618e-07, 5.10712e-06, 2.7865839e-05, 7.178262e-06, 4.581452e-06, 1.8116694e-05, 5.820323e-06, 3.7669934e-06, 7.9357415e-05, 2.4837062e-05, 1.822985e-05, 1.283606e-06, 1.2288719e-05, 2.9043683e-06, 0.00011244261, 4.338252e-06, 0.00021625363, 2.9391324e-05, 1.0695196e-06, 0.00011266577, 0.40733615, 0.5772368, 0.0004146664, 1.229108e-06, 1.202781e-06, 2.706326e-05, 9.573294e-06, 1.6624028e-06, 1.3531604e-06, 4.1547428e-05, 1.3941313e-05, 3.45452e-06, 3.975819e-07, 1.14479995e-07, 3.2602754e-06, 2.4812724e-07, 5.9290517e-07, 9.359942e-06, 5.2608295e-05, 1.6942557e-05, 3.36582e-05, 3.7529944e-06, 1.7631246e-06, 6.8221685e-05, 1.913694e-07, 2.6620355e-06, 2.7133906e-06, 2.9902525e-05, 3.281977e-05, 0.00021295837, 1.4895779e-05, 1.5925854e-05, 7.54719e-07, 0.00016248274, 8.674957e-05, 7.4369905e-06, 1.7487591e-05, 5.6154302e-05, 1.0220183e-06, 2.6891967e-05, 1.2903312e-05, 1.98491e-05, 1.0393447e-05, 3.2990965e-06, 4.3320383e-06, 6.4613107e-07, 5.837787e-07, 9.864287e-06, 1.605011e-06, 5.953469e-06, 3.730937e-06, 7.4432987e-06, 1.6610296e-05, 2.930948e-05, 3.3447233e-07, 4.6493938e-06, 6.9082525e-08, 3.058961e-07, 1.7166974e-07, 1.0861472e-07, 1.1060474e-07, 9.991572e-06, 2.624814e-06, 7.4486564e-07, 1.1626346e-06, 8.506138e-07, 3.9435324e-07, 5.3783315e-06, 1.02206464e-07, 3.4676552e-07, 1.2598566e-07, 2.1253848e-07, 1.1014853e-07, 3.873366e-07, 5.1221065e-07, 7.8767215e-08, 3.3245198e-07, 1.7352278e-07, 1.7126436e-07, 7.1782176e-07, 5.609307e-07, 1.8716644e-07, 2.7722834e-07, 2.3728228e-07, 1.4343879e-06, 1.6317398e-07, 1.8679945e-07, 9.642633e-07, 6.046637e-07, 2.5942106e-07, 6.012982e-08, 3.1809796e-07, 3.3224913e-07, 2.943848e-07, 4.0296842e-07, 1.1191497e-06, 1.4413408e-07, 2.769044e-07, 1.8496877e-07, 1.05810884e-07, 5.328658e-07, 4.866973e-06, 3.224421e-05, 1.1619223e-05, 2.464877e-06, 1.8399638e-05, 1.6188493e-06, 2.0739626e-06, 9.169329e-05, 2.8408182e-05, 1.0441167e-06, 2.2457004e-06, 4.574065e-06, 6.362242e-07, 9.629592e-07, 1.0002079e-05, 7.1745495e-07, 2.1475851e-07, 1.01941005e-05, 8.618323e-07, 1.869045e-06, 6.2070296e-07, 1.1434444e-06, 2.3597759e-06, 6.7796793e-07, 1.271199e-05, 1.1577484e-06, 2.368491e-07, 4.55484e-07, 2.5140366e-06, 5.6595123e-08, 7.3302494e-06, 4.584962e-06, 2.463876e-06, 1.1013761e-07, 7.970608e-07, 1.06286215e-07, 6.5447483e-07, 5.9785117e-07, 3.151479e-07, 1.0185223e-06, 2.0698292e-06, 2.019361e-06, 1.1943068e-06, 9.928024e-07, 8.1185874e-07, 1.9383758e-06, 8.319345e-07, 6.315681e-07, 1.5005149e-07, 1.4024852e-06, 2.58234e-07, 4.5330773e-07, 1.0390844e-06, 6.014614e-07, 5.0667715e-08, 4.292828e-08, 2.428166e-07, 2.1167232e-07, 5.6699145e-07, 3.6647265e-07, 7.7877735e-07, 9.033721e-08, 7.109228e-08, 1.13489925e-07, 5.4377375e-07, 1.7701846e-07, 7.7141095e-08, 9.243051e-07, 2.0634916e-07, 7.603061e-07, 2.1217143e-06, 1.2784554e-06, 6.703036e-08, 2.8089295e-07, 8.370749e-07, 3.334441e-06, 4.4463496e-07, 2.1578214e-07, 8.6750146e-07, 2.0263879e-07, 1.757333e-06, 4.232752e-06, 5.2864135e-07, 1.0017834e-06, 8.292726e-06, 2.0218168e-07, 5.6843703e-07, 3.1687039e-06, 2.3628252e-06, 1.3313359e-06, 2.1216838e-06, 3.529718e-07, 1.0109056e-06, 5.4811755e-07, 1.5179745e-06, 1.1439428e-06, 2.7279775e-06, 2.491207e-05, 2.4989247e-06, 1.0711284e-06, 3.4363297e-06, 1.795127e-06, 5.6359966e-08, 4.6255496e-05, 7.617995e-06, 1.7480146e-06, 1.513277e-06, 4.223011e-06, 2.5502782e-06, 4.8256497e-06, 6.389756e-07, 2.050033e-06, 1.9788192e-06, 2.5552567e-06, 6.402927e-07, 1.5162955e-05, 1.9254494e-06, 1.4773743e-05, 1.5308382e-07, 6.756578e-07, 3.368628e-06, 3.2261296e-06, 1.3053974e-06, 8.031118e-06, 8.357667e-07, 4.2610545e-06, 5.840485e-06, 7.83309e-07, 6.467494e-07, 4.065039e-06, 1.2906764e-06, 3.6102936e-06, 1.5432053e-05, 1.8839935e-06, 9.1145074e-07, 9.640519e-07, 1.5100665e-06, 1.49469e-06, 3.234839e-06, 1.8960048e-06, 9.3880084e-07, 1.1120671e-06, 1.266507e-06, 3.204753e-06, 2.2967995e-06, 8.827923e-06, 1.5669655e-06, 2.128402e-06, 4.6287373e-06, 3.5791315e-06, 1.1039351e-06, 7.966436e-07, 2.8899622e-06, 1.6511663e-07, 2.5859703e-07, 5.687974e-06, 7.1050804e-06, 4.2829834e-07, 1.4763658e-05, 5.886087e-07, 3.8952072e-07, 1.2212666e-06, 4.2225375e-07, 2.9476307e-06, 2.9590444e-07, 1.647012e-06, 1.138658e-05, 6.443352e-06, 4.116174e-06, 2.3373326e-07, 4.594675e-06, 5.326156e-06, 2.7369208e-06, 9.157579e-06, 1.6969457e-06, 1.5630312e-06, 7.475492e-07, 3.4837499e-06, 4.043013e-06, 3.866793e-07, 1.3447072e-06, 2.7079177e-06, 7.07651e-06, 3.3592798e-06, 2.4041381e-05, 1.5609503e-06, 1.4794115e-07, 1.8450921e-06, 3.6126664e-06, 8.430876e-07, 1.6879924e-06, 1.49956795e-05, 8.848961e-06, 7.616669e-07, 8.442579e-06, 5.06263e-07, 4.8507803e-07, 9.1500846e-07, 2.4074225e-06, 6.2053016e-07, 9.895185e-07, 2.1231817e-06, 5.107417e-06, 1.9196445e-06, 2.279443e-06, 1.78816e-07, 2.918832e-06, 1.6303065e-06, 1.3995557e-05, 3.7271278e-08, 7.159881e-07, 1.059618e-06, 2.258917e-06, 2.0854193e-07, 2.4586356e-07, 5.275983e-07, 2.4461443e-07, 7.5761e-07, 4.2390033e-07, 1.6908271e-07, 1.8666652e-06, 1.8524983e-05, 1.4771988e-06, 2.6837728e-07, 1.7947718e-07, 1.140592e-06, 1.8138328e-06, 6.742733e-05, 2.4207802e-06, 7.482425e-07, 4.8928086e-07, 4.0176797e-06, 1.5053209e-05, 2.5682698e-07, 6.180497e-07, 1.2896313e-05, 1.0683698e-06, 7.219537e-06, 2.5994248e-06, 7.657017e-07, 1.0884103e-06, 4.019638e-06, 1.1041446e-06, 8.6559413e-07, 3.2376015e-06, 5.8260262e-08, 3.7094105e-07, 1.7066193e-05, 9.563972e-07, 2.7943852e-06, 8.446875e-07, 8.798296e-06, 2.0743909e-07, 3.720597e-06, 9.2031805e-05, 3.2299022e-07, 1.3909248e-06, 1.5150531e-06, 1.1316872e-05, 3.735218e-07, 8.270013e-06, 9.53729e-08, 5.2934156e-05, 1.0523074e-05, 2.0331552e-07, 1.072588e-06, 2.2581287e-06, 1.7145777e-06, 1.3304353e-07, 7.8983436e-07, 8.3505965e-06, 4.7793077e-07, 8.633999e-06, 5.063731e-07, 2.9272896e-06, 1.6326154e-06, 1.6821863e-06, 1.6723036e-05, 2.907346e-07, 1.0632824e-06, 1.8857632e-05, 8.1562877e-07, 3.4525522e-07, 2.9374132e-06, 1.6777803e-06, 2.093815e-07, 5.092932e-06, 2.2783825e-06, 3.2003124e-06, 3.72425e-06, 1.1175212e-06, 7.551085e-07, 1.2608814e-06, 7.110252e-06, 6.5194e-07, 1.294883e-06, 1.4719675e-06, 6.0391164e-07, 2.758968e-06, 1.3804836e-06, 1.03495595e-05, 1.0754078e-06, 4.651161e-07, 8.321808e-08, 3.883249e-07, 4.3750847e-06, 6.9287e-06, 1.0790991e-06, 2.0175928e-07, 1.1477083e-06, 9.2724866e-07, 4.4966427e-07, 3.768524e-06, 8.53336e-08, 7.5422184e-07, 5.943853e-06, 5.7822704e-06, 1.1200253e-06, 2.2967261e-07, 1.7970591e-06, 5.6860836e-07, 1.3928349e-06, 6.2377126e-06, 5.0388206e-07, 6.703836e-06, 2.8018142e-06, 3.1146221e-06, 5.6211627e-07, 2.3552186e-06, 7.2731456e-07, 8.2250574e-07, 1.7007797e-07, 1.4904311e-06, 2.3674602e-06, 8.284086e-06, 1.5256389e-06, 5.753343e-06, 1.6530263e-06, 4.4300623e-07, 2.975805e-06, 1.5939414e-06, 1.2028959e-06, 1.9596916e-06, 5.4057625e-07, 2.5539276e-07, 3.354089e-05, 4.7010667e-07, 7.318403e-06, 6.808285e-07, 1.2269537e-05, 2.980676e-06, 1.2634037e-07, 5.5363403e-06, 3.1773146e-05, 9.6223175e-08, 1.5475484e-07, 1.4305355e-06, 3.206239e-08, 5.6593562e-06, 2.2849756e-06, 4.1018943e-06, 4.2290035e-06, 7.9598516e-07, 8.535069e-08, 3.1764587e-06, 2.3981077e-06, 2.4379176e-06, 2.924787e-06, 1.9348952e-05, 3.043286e-05, 3.9040322e-07, 1.9553707e-07, 3.6785837e-06, 9.770004e-07, 2.1725493e-06, 9.120881e-07, 2.3135665e-06, 1.628987e-06, 2.6829575e-06, 5.118893e-07, 6.0893008e-06, 2.598494e-07, 2.110724e-07, 3.142682e-07, 8.9087973e-07, 1.6903034e-05, 1.8139677e-06, 4.9642364e-07, 9.3815464e-07, 1.1011582e-06, 8.299229e-06, 4.316495e-06, 1.5535351e-06, 1.2686715e-07, 7.4317227e-06, 2.2307408e-07, 7.130532e-07, 1.6027021e-05, 2.522636e-07, 2.934066e-07, 1.3851732e-06, 5.2406995e-07, 3.4384148e-06, 1.7058935e-07, 2.2353183e-06, 1.1527202e-06, 2.0424222e-06, 9.0871504e-07, 9.279175e-07, 1.2784419e-06, 1.0543832e-06, 1.288475e-06, 7.744535e-07, 6.339913e-06, 1.9381142e-07, 3.6516923e-07, 5.14587e-06, 1.8609968e-06, 2.1099022e-06, 6.7389306e-06, 1.0700466e-05, 6.8114905e-06, 2.9642656e-06, 7.136022e-07, 7.744875e-07, 3.7833279e-06, 3.0161364e-06, 6.936075e-07, 4.3780747e-07, 1.0460443e-06, 8.8778995e-07, 2.3039381e-06, 1.7693857e-06, 1.115564e-06, 1.661298e-06, 2.6530446e-07, 4.3556693e-07, 3.4118489e-06, 1.5530655e-06, 7.466016e-07, 5.063683e-07, 1.3009085e-06, 1.5598796e-05, 1.0366763e-06, 1.1196889e-06, 3.0060828e-06, 1.356155e-06, 9.820625e-07, 2.1989595e-06, 1.6922332e-06, 1.3272423e-06, 1.9030645e-06, 2.1997091e-07, 8.5709223e-07, 3.8253297e-06, 3.4569968e-07, 2.6077578e-06, 3.483567e-06, 3.6281364e-07, 1.7578661e-06, 1.9342892e-06, 1.7658742e-06, 3.907155e-06, 8.981448e-06, 8.9915403e-07, 2.0014852e-07, 4.210974e-06, 4.8389336e-07, 2.2567508e-06, 8.9751427e-07, 6.308779e-08, 5.069776e-07, 4.775052e-07, 8.122239e-06, 5.8180678e-05, 1.3861193e-06, 3.4600296e-06, 6.8834534e-06, 1.4960676e-06, 1.9307035e-07, 1.0058861e-05, 1.646274e-06, 2.423841e-06, 1.2812272e-06, 6.536019e-08, 1.56902e-06, 7.662522e-06, 1.6937428e-06, 1.2506719e-06, 3.169654e-07, 1.827239e-07, 5.736246e-07, 1.476168e-06, 5.8050346e-06, 2.8127104e-06, 8.41238e-07, 9.861714e-07, 3.964515e-06, 1.5863016e-06, 5.7537403e-07, 1.9392207e-06, 6.138479e-07, 1.4168915e-06, 1.0203871e-06, 1.4193434e-06, 7.2978736e-07, 2.6645528e-06, 4.362764e-06, 5.1138727e-07, 1.5418674e-06, 2.1608723e-06, 4.441137e-07, 2.1684011e-06, 5.7982095e-07, 4.208252e-06, 6.2694644e-06, 1.0910754e-07, 8.9542823e-07, 7.5162893e-07, 1.2211142e-06, 5.671848e-06, 8.001827e-05, 5.315375e-07, 1.2871044e-06, 2.5712022e-06, 1.929912e-07, 4.3394084e-07, 3.7587375e-07, 1.1067423e-06, 1.4056023e-06, 3.7882778e-06, 4.107226e-06, 8.979295e-07, 9.980064e-07, 7.4344875e-06, 8.000773e-07, 3.3249592e-06, 6.2859344e-06, 1.2670778e-05, 6.181024e-06, 5.992325e-07, 1.4047648e-06, 1.2503236e-06, 3.9425134e-07, 2.5586876e-06, 3.5965681e-06, 3.2103972e-06, 5.080649e-08, 7.5429407e-06, 1.1273129e-05, 4.050838e-07, 6.827945e-06, 6.557428e-06, 1.5494928e-06, 7.5670454e-07, 5.3732815e-06, 3.926059e-06, 6.872614e-07, 2.1881797e-06, 8.326354e-07, 4.418257e-07, 2.5343563e-06, 1.6838235e-06, 8.0051086e-07, 5.0417003e-07, 4.4274334e-06, 1.7458055e-06, 1.312581e-05, 6.8006834e-06, 6.6938344e-07, 1.9611891e-06, 4.05965e-06, 1.2272673e-05, 4.318187e-06, 2.5695033e-06, 3.420368e-06, 1.8288926e-06, 7.090443e-07, 1.1700457e-06, 1.5462392e-06, 8.471674e-07, 1.19016195e-05, 1.0063437e-07, 3.5280016e-07, 8.083196e-07, 7.831007e-07, 2.2469967e-06, 3.2198745e-06, 5.46896e-06, 2.3607876e-07, 4.5549596e-06, 1.777636e-06, 1.8209928e-06, 1.0895079e-06, 3.649106e-07, 3.4167476e-07, 3.4340349e-07, 8.9606124e-07, 1.8848374e-05, 4.4522218e-07, 1.1343252e-06, 6.218925e-06, 6.378472e-08, 6.0418674e-06, 4.010594e-06, 8.1876243e-07, 2.5506273e-05, 2.0598866e-06, 3.0352992e-06, 7.1552597e-07, 2.509204e-07, 7.4251875e-07, 1.1561078e-06, 8.450162e-07, 1.1495174e-07, 2.30015e-07, 4.216791e-07, 3.6176586e-06, 1.8345672e-07, 3.6328088e-06, 1.041257e-06, 6.1919064e-07, 1.6489703e-06, 1.23177115e-05, 3.0211654e-06, 1.7541284e-06, 1.4839601e-07, 6.1302976e-06, 2.0154073e-06, 3.6311096e-07, 7.3696737e-07, 4.370341e-07, 1.5177255e-06, 1.2590387e-07, 2.3164412e-06, 7.050975e-07, 8.3458315e-07, 4.139802e-06, 5.4616526e-06, 3.749249e-06, 1.1025905e-06, 8.433087e-07, 2.094582e-07, 3.424967e-06, 5.2668106e-06, 5.819143e-07, 1.4725024e-06, 1.2079207e-06, 3.0717872e-06, 1.0360299e-06, 2.678918e-06, 2.2673117e-05, 4.2749342e-08, 7.203822e-06, 1.2097364e-06, 3.5521066e-07, 1.5892792e-05, 7.6659603e-07, 1.5075173e-07, 1.3642651e-06, 1.4929412e-07, 2.595537e-07, 2.5446576e-07, 1.00538834e-07, 2.9553246e-07, 6.4130705e-07, 2.1728714e-07, 6.822549e-06, 7.864933e-05]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>62</td>\n",
+       "        <td>0.05564801</td>\n",
+       "        <td>4</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, [2.4545777e-06, 5.2808724e-07, 4.793985e-07, 3.0648587e-06, 1.7589542e-05, 7.423473e-06, 2.8404587e-05, 7.702516e-07, 2.9076324e-07, 4.9191232e-05, 7.6386027e-07, 1.6786764e-07, 2.8007811e-08, 1.10271536e-07, 6.7959e-08, 4.662994e-07, 1.4375345e-07, 1.0798308e-07, 3.0669946e-07, 2.0534213e-07, 7.262093e-07, 1.9336496e-06, 5.923924e-06, 2.124527e-06, 1.517234e-06, 9.527852e-05, 0.00018055105, 8.729679e-06, 1.3466038e-05, 4.9883015e-06, 0.00016520561, 8.931284e-07, 0.00010447865, 0.008126934, 0.0041637723, 0.00013488345, 0.0004189078, 1.1425285e-05, 0.0005037103, 0.0009210269, 1.1044925e-06, 0.00024138592, 9.045082e-05, 9.4167415e-05, 0.0013647893, 0.0005210496, 2.0178764e-05, 5.4065484e-05, 0.0009393721, 0.05819438, 0.027708823, 5.9237965e-07, 0.00053300196, 0.002545086, 0.037394878, 6.184113e-05, 0.000853419, 0.0009619954, 0.1408455, 0.00020672889, 0.014558754, 0.00151045, 0.055647947, 0.011832289, 0.0007671514, 0.5346089, 0.006554945, 0.041236367, 0.039578155, 1.4561356e-05, 1.6464638e-07, 3.165858e-05, 3.935167e-07, 1.0918371e-06, 5.218124e-07, 2.7307672e-07, 1.0199102e-05, 3.5511948e-06, 2.86876e-07, 4.0874023e-05, 2.6729484e-07, 1.0093462e-06, 4.629379e-06, 2.1517774e-06, 1.2261435e-05, 4.338481e-06, 9.929987e-07, 1.7831232e-07, 2.4536823e-07, 1.5457067e-07, 1.2165957e-07, 3.4199334e-07, 5.923435e-07, 2.5983104e-06, 1.6504788e-07, 7.1641075e-08, 5.894999e-07, 1.5662519e-06, 5.8751893e-06, 3.89696e-06, 4.1094467e-05, 4.955301e-06, 6.5522227e-06, 1.9406043e-05, 1.6132468e-05, 3.7659976e-07, 6.3438647e-06, 4.0902314e-06, 1.0292862e-06, 6.6630196e-06, 5.040724e-06, 8.1575585e-05, 7.456448e-05, 2.8282989e-06, 3.618538e-05, 3.8675685e-07, 7.651428e-05, 1.0225481e-06, 1.266903e-05, 4.5824258e-05, 7.288306e-05, 3.3498336e-06, 2.992845e-06, 1.8091552e-06, 1.875435e-05, 6.7368404e-05, 4.9985447e-06, 3.3620515e-07, 4.305059e-06, 1.7280902e-06, 7.6235165e-07, 5.0209233e-06, 7.799939e-06, 1.937166e-06, 9.611823e-07, 1.614425e-06, 1.8713204e-07, 2.9885357e-06, 4.8278425e-06, 2.0012436e-05, 7.2182515e-06, 4.359301e-06, 1.574922e-05, 2.238073e-06, 1.7358202e-06, 1.3882881e-06, 2.0444086e-06, 2.4902007e-05, 8.8671567e-07, 9.5118907e-07, 5.4420452e-05, 4.4048585e-07, 1.533093e-07, 2.2423917e-07, 1.2845265e-07, 5.4538754e-07, 6.9552794e-07, 3.5553444e-07, 4.2179462e-07, 1.1077553e-06, 1.831457e-06, 1.143593e-06, 5.030103e-07, 9.824084e-07, 6.9452412e-06, 1.0099182e-06, 3.9539648e-07, 8.06763e-08, 4.0702827e-07, 5.923435e-07, 5.3197714e-06, 1.4797711e-06, 2.7858605e-06, 1.0203154e-06, 1.270072e-06, 1.7932915e-06, 1.1223898e-06, 1.32488985e-05, 2.3145172e-07, 2.5678237e-06, 6.031348e-07, 7.921732e-07, 2.581308e-06, 3.6643971e-06, 7.416457e-07, 9.522029e-07, 7.175475e-07, 1.1451864e-06, 2.0917491e-06, 9.24253e-06, 3.8683064e-07, 1.0389535e-05, 7.724614e-07, 7.472543e-07, 5.071702e-07, 1.6446974e-06, 1.1704658e-06, 3.0614149e-06, 1.409079e-06, 8.466847e-07, 1.8017014e-06, 3.507826e-07, 2.1284595e-06, 8.698909e-07, 1.224563e-06, 1.7106437e-06, 5.696064e-06, 4.514067e-07, 1.1763666e-06, 3.3413244e-06, 3.4128946e-06, 5.175482e-07, 3.714022e-06, 2.2927e-07, 2.316875e-06, 3.377438e-07, 2.3052058e-07, 2.61643e-06, 6.701335e-07, 8.474925e-07, 2.7116255e-07, 6.2081126e-06, 1.5719688e-07, 1.4582722e-06, 6.9947583e-07, 1.865066e-06, 3.862484e-06, 2.6300042e-06, 3.1585807e-07, 5.6471627e-06, 2.8117404e-06, 3.3101567e-06, 4.781126e-06, 5.332083e-06, 4.262747e-06, 3.4367472e-06, 1.5752081e-06, 9.933036e-07, 4.4076637e-06, 3.7497364e-06, 1.7161192e-06, 2.5797578e-06, 1.5539875e-06, 1.019196e-06, 1.471617e-06, 2.4699254e-06, 2.445008e-06, 7.9564916e-07, 9.740301e-07, 3.635142e-06, 1.5163473e-06, 1.1429203e-06, 3.060701e-07, 2.6062207e-07, 1.1011417e-06, 1.8995628e-07, 1.8400178e-06, 2.40029e-07, 1.1530976e-06, 4.4667553e-07, 1.499923e-07, 1.0562828e-06, 2.805113e-07, 3.573091e-07, 4.562204e-06, 9.904574e-07, 4.4645708e-07, 8.6095275e-07, 3.799917e-07, 5.7280236e-06, 1.5454673e-06, 8.292788e-07, 1.09758175e-05, 1.1667296e-05, 9.930498e-07, 8.749859e-06, 5.806174e-05, 1.8263291e-06, 2.6493578e-06, 2.9941327e-06, 1.7455774e-05, 1.29098025e-05, 5.3980243e-06, 6.8297766e-08, 2.1631668e-07, 6.335388e-06, 1.089367e-06, 2.0598577e-06, 4.8624817e-05, 1.12490325e-05, 9.968333e-06, 1.8108658e-06, 3.7059235e-06, 4.596375e-05, 1.0402488e-06, 4.026173e-06, 1.0984137e-06, 2.3183823e-06, 0.00011660683, 8.382035e-06, 1.2012907e-07, 2.1115221e-07, 1.6709426e-07, 2.803805e-07, 9.1352774e-08, 8.547764e-06, 8.3231095e-07, 7.207249e-07, 4.964948e-07, 4.7260255e-07, 1.924433e-06, 5.233134e-07, 3.1484038e-07, 8.789158e-08, 3.3203023e-06, 1.5410322e-07, 7.599468e-07, 6.5865486e-08, 2.2502013e-06, 1.9028707e-06, 2.0829236e-07, 1.9414686e-07, 1.737569e-08, 4.187208e-08, 2.133833e-08, 5.217957e-08, 7.756315e-08, 0.00028948556, 3.2254661e-06, 0.00045745773, 3.6141873e-06, 3.6879846e-06, 4.3864155e-08, 1.1546421e-07, 6.098287e-06, 4.7297253e-06, 7.4544114e-06, 2.3579938e-05, 1.7185849e-07, 8.238342e-09, 6.6577663e-06, 7.1307113e-07, 6.6831553e-06, 4.188079e-06, 1.1908708e-05, 2.2311808e-06, 2.3669925e-05, 5.633239e-07, 2.5464924e-06, 3.673358e-06, 9.228042e-06, 3.7669442e-06, 2.5243994e-06, 3.923309e-06, 1.5492952e-06, 5.415315e-07, 1.2985793e-05, 7.727771e-06, 1.1638047e-05, 6.415278e-06, 0.000117049596, 1.844425e-05, 3.0543357e-05, 7.667886e-05, 4.6682158e-06, 5.8063925e-08, 3.4565298e-07, 1.3907674e-06, 3.9034413e-07, 2.3856628e-06, 1.3961561e-06, 1.0203523e-06, 9.6493195e-06, 1.916817e-06, 5.992382e-06, 1.3963438e-06, 1.6447373e-07, 1.1997806e-05, 3.522568e-06, 7.992761e-07, 2.7826318e-06, 2.9383807e-06, 1.2469625e-06, 6.400709e-06, 8.6687646e-07, 1.3604218e-05, 3.653841e-06, 3.270486e-07, 2.1591251e-07, 7.4862564e-06, 0.0022334387, 2.6425723e-06, 8.8534085e-07, 3.9918265e-07, 1.4015102e-05, 6.251236e-05, 3.7113346e-06, 4.424491e-05, 5.257468e-08, 2.3197601e-06, 8.55421e-09, 6.37598e-08, 1.7909628e-07, 8.214551e-08, 3.2871288e-08, 6.140439e-08, 1.6078678e-07, 1.8680272e-07, 5.777686e-06, 3.2538915e-07, 8.600066e-07, 2.7519786e-08, 1.1244575e-06, 4.4513232e-07, 1.3912985e-07, 2.2559327e-07, 7.5065955e-08, 2.1787962e-07, 6.5873184e-07, 1.6194141e-07, 4.2486533e-07, 9.449337e-08, 4.586137e-07, 6.8615904e-08, 3.4233057e-08, 2.3015714e-08, 8.529465e-08, 1.1009139e-06, 2.085397e-06, 1.1364241e-06, 5.969457e-08, 5.854545e-08, 1.2437502e-07, 2.8979143e-06, 3.7183227e-07, 6.1114713e-07, 1.8259102e-07, 5.007582e-06, 2.0794462e-07, 7.3661596e-08, 1.2289431e-06, 4.696466e-07, 8.236131e-08, 3.7003272e-07, 9.213571e-07, 6.144086e-07, 2.8531753e-07, 1.6841451e-06, 3.648355e-07, 3.489435e-07, 2.1981393e-07, 3.489891e-07, 6.616541e-08, 1.7107e-07, 5.8820905e-08, 2.657458e-06, 6.7280183e-07, 7.9155576e-08, 3.4055512e-07, 5.2415576e-08, 1.9475921e-05, 2.0859183e-08, 5.2399713e-07, 3.550162e-06, 6.954769e-07, 1.5623053e-07, 4.141466e-08, 7.0278865e-08, 1.1458085e-07, 1.1404557e-07, 3.500317e-07, 5.067037e-07, 2.6709915e-07, 6.325267e-07, 2.1115885e-07, 6.933021e-07, 4.481367e-08, 7.1330064e-08, 4.751686e-07, 6.4066296e-07, 2.3656271e-07, 1.1212038e-07, 5.1450616e-08, 4.898539e-07, 3.9345363e-07, 7.469027e-08, 2.8215636e-08, 1.4027972e-06, 1.5045851e-05, 5.45923e-07, 3.3606793e-07, 6.793033e-07, 3.2025406e-07, 5.567628e-08, 5.144404e-08, 4.7856094e-08, 8.7845734e-08, 2.304809e-08, 3.320628e-08, 1.531284e-07, 8.455002e-07, 4.030141e-08, 5.826353e-07, 2.0933905e-07, 2.1854659e-07, 1.8154402e-07, 8.5039017e-07, 9.850174e-07, 4.691499e-08, 2.9529717e-08, 5.921111e-06, 1.4848057e-07, 5.5719124e-07, 3.0771523e-07, 7.578906e-07, 1.2629686e-06, 4.9141647e-08, 7.4479624e-07, 1.2221409e-06, 1.6379033e-07, 3.4507277e-08, 3.4415274e-07, 1.2394273e-07, 2.48148e-06, 3.051856e-08, 7.7588567e-07, 3.814518e-08, 1.1678488e-08, 3.970682e-07, 3.907356e-07, 6.9163754e-08, 3.2372859e-07, 4.107152e-08, 1.32908e-07, 7.114948e-08, 1.3805939e-07, 4.362909e-07, 1.7327786e-06, 1.0840065e-07, 4.287437e-06, 1.3406026e-06, 2.7652288e-07, 5.110169e-07, 2.6476766e-07, 3.4128325e-06, 7.380665e-08, 2.5400274e-07, 6.9870474e-08, 5.537498e-08, 4.607658e-07, 1.6728343e-08, 1.2180828e-07, 3.1195963e-08, 3.9396355e-07, 2.7166365e-07, 1.8899802e-07, 1.434787e-07, 5.057714e-07, 1.0786199e-06, 9.368018e-08, 4.7229683e-08, 8.0803005e-07, 5.0072924e-07, 2.4026627e-07, 1.9182908e-08, 8.188763e-08, 1.274422e-07, 2.8407675e-07, 2.4712435e-07, 1.4823198e-07, 5.0134037e-07, 7.983131e-07, 1.8063096e-07, 7.675681e-06, 4.8255936e-07, 8.072598e-07, 1.6763661e-07, 1.07333484e-07, 6.337291e-08, 1.5299548e-08, 1.586752e-07, 2.2856735e-08, 1.3107238e-07, 3.1036965e-08, 4.9698727e-08, 5.08264e-07, 1.4144146e-06, 1.758415e-06, 2.6504657e-07, 9.582741e-08, 5.6582618e-08, 2.8249872e-07, 1.8841897e-08, 8.451918e-08, 1.1465327e-06, 6.435631e-07, 7.6025344e-07, 5.33686e-08, 1.4108464e-07, 4.900749e-07, 1.1835917e-05, 1.6986917e-08, 5.604835e-08, 3.662114e-07, 6.067581e-08, 2.396404e-07, 4.516629e-07, 4.7011735e-08, 3.1379034e-07, 2.5050101e-06, 3.2410483e-07, 1.0516861e-06, 1.4487472e-07, 3.085546e-08, 4.3942286e-08, 4.497369e-07, 4.5126813e-05, 5.931131e-08, 3.7003736e-07, 2.668337e-08, 8.2547594e-08, 7.860159e-07, 1.1476485e-06, 3.1438208e-06, 7.495808e-08, 2.8493167e-07, 1.9044575e-06, 1.75579e-07, 2.7747865e-07, 1.8837882e-07, 3.0251408e-07, 5.71522e-07, 3.9703924e-08, 2.5046242e-07, 7.552639e-08, 4.5776423e-07, 9.707653e-08, 1.4215377e-07, 1.4616903e-06, 1.1546075e-06, 2.3138962e-06, 1.7347833e-07, 4.9520267e-08, 3.0973612e-07, 1.7835579e-06, 3.6174768e-08, 6.2489794e-06, 2.0208286e-07, 7.262567e-08, 3.0372778e-06, 1.3853102e-07, 2.5439329e-07, 3.3290007e-07, 3.0902535e-07, 5.590861e-07, 1.06721984e-07, 6.7565156e-07, 2.1733736e-07, 3.322144e-07, 2.0757233e-07, 7.719017e-07, 8.4283094e-08, 1.0376899e-07, 1.9896996e-07, 2.2272106e-07, 7.127427e-07, 9.009337e-07, 3.2640187e-08, 2.0750068e-07, 4.8350348e-08, 7.0408504e-07, 5.6114324e-05, 7.5532836e-07, 8.887479e-08, 7.085125e-07, 1.1560844e-07, 3.8795744e-07, 3.976793e-06, 4.6256346e-07, 5.975669e-07, 3.0750516e-07, 7.345843e-08, 2.4090642e-07, 2.1353691e-07, 3.3497807e-07, 1.00545954e-07, 4.805937e-07, 8.9473104e-09, 1.5028067e-07, 2.299759e-08, 2.2111642e-07, 8.5874444e-07, 1.4600192e-07, 2.5260093e-07, 4.1094324e-08, 3.6426795e-06, 6.740626e-07, 1.0925928e-07, 1.2064076e-07, 2.6033217e-07, 2.3739936e-07, 4.02779e-06, 6.986927e-08, 4.927631e-07, 2.0377253e-07, 4.859285e-08, 6.001323e-07, 4.019476e-07, 6.5550914e-08, 3.255296e-08, 4.502512e-08, 4.5910076e-07, 1.9682864e-07, 1.5400917e-07, 2.1057043e-07, 3.4854838e-07, 2.8710346e-07, 7.8076866e-08, 3.7246292e-07, 8.849859e-08, 2.2556638e-07, 8.8687415e-08, 1.4012883e-07, 2.5905608e-08, 4.2669419e-07, 3.5269767e-07, 9.342652e-08, 1.16506385e-07, 1.8241785e-07, 2.592098e-08, 2.0602221e-07, 5.343529e-07, 4.3310727e-08, 3.1328245e-06, 3.7680095e-07, 1.6470683e-07, 1.3352549e-08, 4.713009e-07, 4.3408545e-07, 3.0829065e-07, 4.148033e-07, 3.5043252e-07, 4.1328406e-08, 1.039676e-07, 2.0944228e-07, 3.6792827e-08, 1.9913675e-06, 7.9480465e-08, 3.3763465e-07, 3.8642725e-07, 3.1329128e-07, 1.0457145e-06, 5.3874476e-07, 5.3555846e-07, 2.299623e-08, 2.9225612e-07, 1.2956369e-06, 5.7133235e-07, 1.7188988e-06, 1.951809e-07, 3.692544e-07, 3.600926e-07, 5.115637e-08, 2.873042e-07, 3.5520603e-07, 4.8817757e-08, 3.2432975e-08, 2.3311536e-07, 2.5904757e-07, 5.7130944e-07, 5.5044434e-06, 5.5146495e-07, 3.4003781e-07, 1.2421808e-07, 1.0033226e-05, 1.1899007e-06, 1.1852086e-06, 2.8190502e-07, 3.9343806e-08, 6.526603e-08, 2.8353597e-07, 3.716835e-08, 7.739528e-08, 1.2510313e-06, 1.1097918e-06, 1.4137441e-08, 5.196871e-08, 1.2651475e-07, 4.3768156e-07, 5.16784e-08, 1.8941446e-07, 3.4604793e-06, 2.1872393e-06, 1.603818e-06, 2.8407675e-07, 7.211409e-07, 4.7171773e-07, 7.0506945e-07, 2.37784e-08, 2.6139796e-07, 1.0857099e-08, 4.7737427e-05, 1.7715913e-06, 2.4736508e-07, 5.6247893e-07, 1.7013748e-06, 5.3935753e-07, 6.814604e-08, 8.114711e-07, 2.0709443e-07, 2.1679611e-08, 5.5730574e-08, 1.0388673e-07, 2.0995584e-07, 3.1774303e-07, 1.2546808e-07, 1.5703132e-07, 9.148216e-08, 1.1042813e-07, 1.16445406e-07, 2.0244965e-08, 1.4764365e-07, 9.9541595e-08, 2.9003402e-07, 5.2911406e-07, 3.6716312e-06, 1.8401056e-06, 2.6981215e-07, 2.3241279e-07, 4.6320945e-08, 6.2793663e-07, 4.0248636e-08, 7.398446e-08, 7.438862e-07, 8.786995e-08, 2.891773e-07, 7.3661136e-07, 6.0076565e-07, 2.798309e-06, 2.8826727e-07, 2.20146e-07, 1.3246057e-07, 4.587592e-06, 1.1943971e-06, 4.5323137e-07, 6.210182e-07, 7.987308e-08, 1.1633191e-06, 2.6830708e-08, 2.667916e-07, 9.157102e-08, 2.2695318e-07, 2.539181e-06, 2.993917e-07, 1.3205372e-08, 5.2766785e-07, 4.7152092e-08, 4.660542e-08, 4.905271e-07, 1.408878e-07, 4.298302e-08, 6.095104e-07, 1.6381455e-07, 6.7567737e-07, 1.3618406e-07, 2.9217654e-08, 6.0456813e-07, 3.481089e-07, 2.199539e-06, 3.1379585e-08, 4.0919085e-06, 1.3511705e-07, 1.7438792e-06, 5.164891e-07, 6.14058e-08, 1.692717e-07, 6.0094214e-07, 4.616263e-08, 1.7136415e-08, 1.8789655e-07, 1.664482e-05, 1.109561e-08, 2.0408984e-06, 5.144324e-07, 2.2324717e-07, 7.9635775e-08, 2.3219015e-07, 1.4640878e-07, 1.3904564e-07, 6.791909e-08, 5.3922327e-07, 1.7450128e-07, 2.2044287e-07, 8.2480156e-08, 7.791018e-08, 3.608704e-08, 1.4282757e-06, 4.4595026e-07, 6.1917463e-06, 1.2274731e-06, 6.541434e-08, 5.510292e-07, 6.2777127e-07, 9.852104e-08, 9.2759166e-08, 1.0801202e-07, 5.1773508e-08, 7.03263e-07, 5.48382e-08, 9.163828e-08, 1.8281544e-07, 2.2309499e-07, 1.5063874e-06, 2.9042156e-05, 8.7185555e-08, 3.4663535e-07, 2.3967124e-07, 6.1186444e-07, 2.6795408e-07, 1.7159874e-07, 2.5335256e-08, 1.0267665e-06, 1.2672062e-07, 1.19088995e-07, 9.9702916e-08, 6.4066995e-08, 1.2536097e-08, 9.928059e-08, 7.231101e-08, 2.8704923e-07, 6.1002176e-08, 2.986067e-07, 1.2906129e-06, 5.925127e-08, 1.8925866e-07, 9.22468e-08, 1.4960415e-07, 1.06911564e-07, 4.2838074e-08, 3.6908082e-07, 1.1071727e-07, 3.8998283e-07, 1.568034e-07, 3.122595e-07, 8.165866e-08, 1.6952727e-07, 1.9333584e-08, 1.3010535e-07, 3.7023185e-07, 3.297176e-08, 6.786729e-08, 2.6440307e-07, 4.7650342e-08, 2.6313728e-07, 1.0631267e-06, 7.3527224e-07, 1.0764516e-06, 7.899638e-08, 4.3475617e-07, 8.123929e-08, 9.337006e-08, 5.917117e-07, 8.175583e-08, 2.3680897e-07, 1.224908e-07, 1.1772941e-07, 7.237772e-08, 1.7921716e-08, 2.1354629e-07, 1.7915333e-07, 4.6141595e-06, 1.3726662e-06, 4.41399e-06, 1.4019968e-05, 5.0816493e-06, 3.158545e-05, 1.0377246e-05, 0.00013469988, 0.00012541312, 8.943293e-07, 3.5171972e-06, 4.143929e-07, 5.997899e-08, 3.7551363e-06, 2.5072768e-07, 4.117451e-08, 1.9050432e-07, 3.9115857e-06, 2.6173373e-07, 2.9716453e-08, 1.8971113e-07, 2.3804073e-07, 5.234724e-08, 1.01332226e-07, 4.7450158e-07, 1.5748033e-07, 6.902779e-07, 8.266525e-07, 3.983048e-06, 1.5279669e-07]),\n",
-       " (2, [1.5398843e-09, 2.3107729e-08, 7.636333e-08, 2.2277517e-07, 1.7589974e-07, 5.438902e-08, 9.2527934e-08, 4.70354e-08, 9.018498e-09, 1.8840204e-08, 2.2168626e-07, 5.591516e-09, 3.5177464e-08, 4.6703323e-07, 4.6959094e-09, 4.3828187e-08, 1.6602337e-08, 2.8010957e-08, 1.4341178e-07, 8.532091e-08, 5.8298728e-08, 8.00204e-09, 3.6414684e-07, 3.851048e-08, 2.0718534e-08, 2.6480208e-08, 3.3010747e-09, 4.47217e-08, 2.8583589e-08, 6.513382e-10, 1.9336234e-08, 5.272263e-10, 2.404317e-09, 1.12124745e-07, 7.488657e-07, 6.5691674e-09, 4.623265e-09, 1.6595015e-09, 1.9597215e-08, 1.6531517e-07, 1.6838877e-08, 8.48814e-09, 2.9826325e-08, 4.543374e-09, 2.3641775e-08, 4.0592163e-08, 6.912111e-09, 3.3884773e-09, 1.042833e-08, 7.361764e-09, 4.2130747e-08, 1.6772022e-07, 1.4776109e-09, 5.525266e-09, 4.0264826e-08, 1.278795e-09, 1.4832867e-07, 4.2117554e-09, 1.3611935e-07, 3.0520868e-09, 6.4132884e-08, 1.4925305e-08, 5.7810297e-08, 6.520122e-09, 2.525428e-09, 2.0901005e-07, 3.504386e-08, 2.659847e-07, 1.3491556e-08, 8.361713e-08, 7.3479924e-08, 1.3090515e-08, 4.685739e-08, 1.2734573e-07, 1.3352158e-07, 9.6114135e-08, 3.8392837e-08, 1.2469758e-07, 5.4070338e-08, 1.2920589e-07, 3.2127724e-07, 1.414701e-05, 3.858608e-07, 1.1320818e-08, 1.5107693e-08, 3.15158e-08, 3.0302337e-07, 3.696147e-09, 2.5316362e-08, 1.6199753e-08, 5.834453e-09, 6.6545938e-09, 7.8418395e-08, 2.9896134e-09, 9.421624e-09, 2.3791766e-09, 2.7823246e-09, 1.8130814e-09, 4.316991e-09, 1.6660833e-08, 1.0815284e-08, 1.2120864e-08, 7.508208e-09, 2.4426359e-09, 7.537742e-08, 2.1063304e-09, 3.549428e-09, 4.384449e-08, 7.2657627e-09, 3.1592776e-08, 2.4405764e-10, 1.6091322e-08, 1.8533458e-06, 2.4308275e-08, 3.7633074e-09, 1.6087233e-09, 2.277431e-08, 5.8490266e-09, 4.0977223e-08, 3.8039954e-07, 1.3494928e-08, 1.781187e-07, 6.407897e-08, 5.389317e-07, 8.723511e-08, 7.2745024e-08, 2.6316968e-08, 6.543285e-08, 7.232058e-08, 4.530396e-08, 1.2370523e-07, 1.0784221e-08, 4.778658e-08, 8.2433225e-09, 1.152689e-07, 1.4541708e-08, 4.6979163e-09, 1.9269741e-08, 6.921267e-09, 2.708216e-07, 5.5321866e-08, 1.6940814e-07, 2.0328073e-08, 2.275105e-07, 7.400611e-08, 4.040531e-06, 5.9436662e-08, 3.5894548e-08, 5.200469e-08, 1.9099544e-09, 2.3085869e-07, 6.950096e-08, 9.922675e-09, 8.168692e-08, 7.0257973e-09, 8.1655614e-08, 1.0280881e-06, 3.198189e-08, 1.8011564e-08, 1.1670971e-06, 4.824007e-08, 4.8371714e-07, 2.3133298e-07, 9.200232e-07, 2.2442546e-07, 4.7174932e-08, 1.1280227e-07, 2.7898892e-07, 8.8787215e-07, 9.2161355e-07, 2.3470897e-05, 1.7707903e-06, 6.7424407e-06, 6.0599414e-06, 2.8161035e-06, 9.434914e-09, 1.0188343e-06, 5.5756454e-06, 3.1946593e-06, 8.136942e-07, 1.1003609e-07, 4.576751e-07, 3.3645716e-07, 2.740537e-07, 5.4528084e-07, 5.777307e-07, 2.9695158e-07, 7.171372e-08, 2.6257868e-07, 1.6330781e-06, 9.9922396e-08, 6.695125e-07, 8.737203e-07, 8.1306325e-08, 5.515203e-08, 6.864878e-07, 7.5751146e-08, 1.1799731e-06, 1.4831099e-07, 2.772437e-07, 1.7763999e-07, 2.098404e-08, 1.940811e-07, 2.6700523e-06, 1.0012253e-07, 6.0266035e-07, 4.6589884e-07, 6.262852e-07, 3.776922e-06, 7.038604e-06, 2.2661327e-06, 1.1063678e-06, 6.118894e-07, 2.5951251e-06, 2.1583949e-06, 2.2275087e-06, 5.9048972e-08, 9.353487e-07, 8.8109886e-07, 1.6752422e-07, 4.7095458e-08, 1.1280829e-07, 6.4681115e-07, 1.4628699e-06, 1.5081841e-06, 6.317585e-07, 1.1813028e-06, 1.900376e-07, 2.4231375e-07, 2.6553482e-07, 4.894391e-07, 2.6518939e-07, 1.5399695e-07, 8.8166956e-07, 6.533642e-07, 4.457954e-07, 1.8340435e-06, 1.855212e-07, 4.893554e-06, 2.3288433e-06, 2.3774019e-06, 5.201625e-06, 7.6095256e-07, 2.3237685e-07, 1.4151934e-07, 8.257309e-08, 2.1411304e-06, 2.2388014e-05, 1.3785373e-05, 5.831647e-06, 1.3944214e-05, 2.2903537e-06, 4.6498332e-08, 4.6375638e-07, 6.242959e-07, 2.3912223e-06, 1.4558304e-06, 9.471192e-07, 5.5140313e-06, 9.166841e-08, 1.2893312e-07, 3.9749816e-07, 4.1223927e-08, 3.76367e-07, 1.4717683e-06, 9.8032444e-08, 3.420696e-07, 1.1348545e-06, 1.233211e-07, 6.2267026e-07, 1.1291587e-06, 6.029464e-08, 4.1942053e-06, 3.126346e-07, 2.5256027e-08, 3.411787e-08, 1.6709652e-08, 1.0040704e-06, 2.5781976e-08, 9.700623e-07, 1.6046155e-07, 2.0606766e-08, 2.4433362e-08, 4.4592547e-09, 1.1156811e-08, 1.8641177e-08, 1.579223e-07, 2.0955612e-07, 2.9641933e-09, 4.2151328e-08, 8.4548657e-10, 1.2786364e-09, 5.8412592e-08, 7.3222175e-09, 3.4796508e-07, 6.028762e-08, 2.8341196e-06, 2.5817788e-09, 1.6543543e-08, 5.1597104e-09, 6.515199e-09, 2.4591106e-08, 1.10343e-08, 2.3506432e-08, 2.5722677e-09, 9.193776e-09, 7.1656943e-09, 6.2450964e-09, 4.8909826e-08, 2.9583264e-08, 1.0357064e-07, 7.1164735e-08, 3.6447915e-08, 3.097284e-07, 4.2578133e-08, 1.8602746e-08, 4.852659e-09, 4.6380855e-09, 8.3475065e-08, 7.4244093e-07, 2.558261e-08, 3.2909315e-07, 7.988803e-09, 2.3116634e-08, 3.85773e-08, 1.5388956e-08, 7.1112463e-09, 1.4074487e-07, 2.9982766e-07, 1.564242e-08, 1.0789998e-07, 1.299596e-07, 1.7822981e-09, 2.1977655e-09, 3.8496357e-09, 1.3551873e-09, 5.3080055e-07, 5.1470135e-08, 8.4123347e-10, 7.115103e-08, 1.0199323e-07, 8.621156e-08, 3.209869e-07, 1.0228442e-08, 3.5695646e-09, 1.6592921e-06, 5.754166e-08, 4.5394786e-06, 6.6713767e-07, 8.642847e-06, 8.179076e-06, 8.936807e-08, 7.0527335e-08, 9.755931e-08, 1.8077302e-06, 2.6266648e-06, 1.8360647e-06, 1.18492515e-07, 1.0602841e-08, 2.8182551e-08, 8.855424e-09, 3.9218904e-08, 9.4692405e-09, 1.8187903e-08, 6.0960176e-10, 2.8660767e-09, 2.7049174e-09, 2.9135108e-09, 2.0202557e-09, 3.7000687e-09, 1.3453749e-09, 7.608665e-08, 1.4827498e-08, 1.8008379e-07, 3.214444e-08, 3.3369876e-09, 1.1487082e-08, 1.0620547e-09, 1.0122127e-08, 2.7668134e-09, 1.38777265e-08, 5.459645e-09, 9.779465e-09, 1.4373716e-08, 1.4275604e-08, 3.1167882e-08, 1.6579142e-08, 1.6735798e-09, 1.7655769e-09, 2.4233465e-08, 7.761856e-08, 8.390994e-07, 8.398905e-09, 1.0605148e-08, 3.585211e-07, 5.846406e-07, 2.773881e-07, 9.707507e-09, 1.7990827e-08, 7.335421e-07, 5.715663e-08, 1.6425967e-06, 2.6860542e-07, 1.945019e-07, 4.4494698e-08, 2.3012471e-07, 1.0821924e-07, 2.522256e-07, 5.9017445e-08, 4.4775834e-07, 2.4033554e-06, 5.852322e-08, 3.763297e-07, 1.2767528e-05, 3.413865e-07, 2.4368111e-08, 1.2132007e-07, 2.718393e-07, 4.8032852e-08, 4.3237666e-08, 3.820392e-07, 2.2197873e-06, 3.518223e-08, 6.4883015e-09, 1.7396843e-08, 1.4981359e-05, 9.289176e-08, 5.7502575e-07, 2.5261088e-06, 7.218663e-08, 1.6187707e-08, 2.5262677e-08, 1.4110595e-06, 2.7774984e-06, 5.1647742e-08, 1.7309242e-07, 6.657365e-08, 1.1532635e-06, 2.5521492e-08, 1.2916955e-07, 1.5800223e-06, 5.573535e-08, 6.0709255e-08, 2.8454885e-08, 2.6240545e-07, 2.2590486e-06, 6.349486e-08, 2.5054753e-06, 1.0332632e-06, 7.5764866e-08, 3.0934818e-05, 1.0285893e-06, 4.9956363e-08, 3.806668e-08, 1.12557945e-08, 4.212406e-09, 9.062275e-06, 2.1683526e-07, 7.8610384e-08, 3.9823655e-07, 1.2365699e-05, 2.6495582e-07, 7.8097855e-06, 1.8755473e-07, 4.612106e-09, 1.1201466e-07, 1.1676837e-08, 3.508029e-09, 9.1218254e-08, 7.643896e-08, 1.5362502e-08, 9.381413e-07, 2.6301302e-07, 6.57294e-09, 7.180635e-07, 3.8983737e-07, 1.554282e-08, 8.0090764e-08, 8.545088e-08, 1.569008e-08, 5.355029e-08, 2.2874363e-07, 3.753023e-08, 3.225784e-07, 1.6325798e-06, 3.007842e-09, 3.2840123e-07, 3.130718e-08, 5.363401e-08, 9.742059e-08, 3.938637e-08, 1.09314815e-05, 2.7909737e-08, 1.9837347e-07, 1.0354215e-06, 5.9552523e-08, 1.6291214e-07, 2.0342968e-06, 2.5287791e-08, 1.59485e-07, 9.795091e-07, 3.1564498e-06, 1.7559394e-08, 1.6565266e-08, 2.620732e-08, 4.9752394e-08, 1.6515578e-08, 1.9081748e-08, 7.422872e-09, 3.3019376e-09, 1.7074969e-07, 1.0950456e-08, 8.516746e-09, 1.1135265e-06, 1.636078e-07, 1.3219143e-06, 3.5221102e-08, 4.5833468e-07, 1.7809934e-07, 1.052475e-07, 2.4009454e-08, 4.471895e-09, 3.8112885e-06, 4.0083058e-05, 1.1635215e-07, 5.101486e-07, 3.6760483e-08, 1.9992733e-08, 9.947765e-09, 3.0854395e-08, 1.118227e-08, 4.7650698e-08, 8.499971e-08, 2.4238595e-08, 1.4947384e-08, 3.6674674e-09, 6.124752e-08, 0.00038468122, 1.8884897e-07, 2.5177487e-08, 2.9864592e-07, 1.9823315e-07, 1.1138956e-07, 3.4749004e-08, 4.807366e-07, 1.1343858e-08, 1.24109e-07, 3.708061e-08, 1.3761401e-07, 7.507371e-07, 8.474337e-10, 3.2129975e-08, 1.6138574e-07, 4.289849e-08, 4.1769333e-08, 1.524812e-07, 4.4421853e-08, 3.3816161e-06, 1.883187e-06, 8.6855124e-07, 3.4589572e-08, 1.10152946e-07, 2.6198538e-07, 1.5651674e-08, 2.0048878e-07, 1.0421765e-07, 4.6393686e-07, 7.667737e-09, 3.3911301e-06, 5.303072e-07, 6.010618e-07, 9.7195084e-08, 4.687357e-08, 1.2324756e-08, 3.2322134e-07, 7.846987e-08, 2.5114367e-08, 7.899244e-08, 2.5696616e-07, 1.01712125e-07, 5.3800395e-08, 4.404471e-09, 7.0128237e-09, 4.1187302e-08, 2.6257606e-08, 7.664451e-08, 3.842652e-07, 6.999354e-07, 5.2077125e-09, 1.8243146e-08, 8.38386e-09, 1.0335663e-07, 6.9752186e-09, 5.2016794e-08, 8.570752e-07, 1.0925754e-08, 3.7993168e-06, 2.1593397e-08, 8.062468e-08, 1.6209674e-08, 9.4739295e-07, 3.8672835e-07, 2.2282576e-07, 4.2491993e-07, 4.789252e-08, 5.8676232e-08, 5.8623435e-08, 1.3501982e-08, 2.2437293e-06, 1.702915e-06, 9.943773e-07, 3.649515e-08, 1.6232762e-07, 3.140646e-08, 1.11932536e-07, 3.590689e-07, 1.3766297e-07, 5.5616647e-07, 5.2407668e-08, 2.748883e-08, 9.189626e-08, 2.5330624e-07, 6.336275e-08, 4.12668e-08, 6.258344e-09, 1.3503933e-07, 1.733054e-08, 1.9494284e-08, 1.6185105e-07, 1.3785838e-08, 6.109794e-08, 1.8560323e-08, 1.6609394e-07, 2.476262e-08, 2.6778312e-07, 4.351084e-08, 3.084329e-07, 7.817878e-06, 1.8975626e-06, 2.0993969e-06, 2.6453353e-08, 2.2118513e-07, 8.731006e-07, 1.6439921e-07, 2.3696006e-07, 1.787288e-06, 4.8213653e-05, 8.37574e-09, 6.81651e-09, 2.8432564e-06, 2.704404e-07, 4.226339e-08, 9.407418e-06, 9.6293405e-08, 3.725218e-07, 1.2643505e-06, 9.604207e-07, 7.47385e-06, 3.566186e-07, 9.538723e-09, 1.6844483e-06, 7.3710655e-09, 5.2484666e-09, 8.958981e-07, 7.355466e-08, 3.6778154e-08, 6.467584e-08, 1.0149527e-07, 1.1401684e-07, 2.0010434e-08, 2.130884e-08, 3.8332364e-06, 0.0007202051, 8.997154e-10, 3.1479754e-08, 8.319596e-08, 2.7916498e-08, 1.4024812e-06, 8.000587e-08, 6.105344e-08, 1.6910072e-08, 2.397463e-08, 2.5698992e-07, 8.2904137e-07, 2.7703791e-08, 2.4609814e-09, 8.351855e-10, 5.907249e-09, 2.3557195e-08, 5.1269083e-08, 3.1052042e-07, 1.9230353e-07, 1.3536194e-07, 5.51488e-07, 1.9878282e-08, 7.9125144e-08, 8.0264836e-05, 2.3167297e-07, 9.360291e-08, 3.483506e-07, 1.8976078e-06, 9.991526e-06, 7.9052325e-07, 2.5110057e-06, 1.2902883e-05, 1.06492195e-07, 1.8583303e-07, 5.769002e-08, 3.6822023e-08, 1.5638037e-08, 1.6243138e-07, 1.0981206e-07, 1.0313105e-09, 2.2036936e-08, 1.6936184e-08, 5.209431e-07, 1.066118e-05, 3.8097987e-08, 6.077448e-08, 8.576721e-09, 2.7099158e-09, 6.393335e-07, 6.024069e-09, 8.597185e-07, 2.4906703e-07, 1.7457806e-08, 9.4648165e-09, 2.0813505e-07, 1.3174564e-06, 4.096353e-09, 4.0667996e-06, 8.2589827e-07, 3.04197e-08, 2.339224e-05, 3.6062332e-08, 1.1635309e-06, 4.037927e-08, 6.19253e-08, 3.770228e-08, 6.161791e-08, 1.5994248e-06, 6.009784e-08, 8.941896e-08, 1.1777902e-07, 2.4193057e-06, 9.981634e-09, 3.2275516e-06, 5.3249114e-08, 1.5546792e-08, 6.834216e-07, 3.9778675e-07, 4.532384e-08, 6.9061976e-08, 1.1572694e-08, 5.0055938e-08, 6.2338563e-07, 4.9324466e-08, 5.308013e-06, 3.0234658e-06, 1.6129864e-07, 7.341118e-07, 4.529463e-08, 4.721031e-08, 2.7191814e-07, 9.214177e-05, 3.4954944e-08, 5.892889e-09, 6.9181067e-07, 3.0443786e-08, 9.5250334e-07, 6.399774e-08, 1.8667293e-08, 4.4535962e-08, 7.128982e-09, 6.305315e-08, 1.3619449e-06, 8.321957e-07, 6.1739435e-08, 7.1391058e-09, 3.2976266e-07, 4.1574063e-07, 9.305059e-09, 1.4348497e-07, 1.12088614e-07, 1.717341e-07, 9.155937e-10, 8.49131e-08, 2.274656e-07, 1.31738735e-08, 6.4238044e-08, 2.5300033e-09, 4.735305e-07, 0.0016679094, 4.3191326e-07, 1.3948901e-08, 0.6965568, 0.00032454787, 3.5461414e-06, 3.661261e-09, 1.7207987e-07, 9.268941e-09, 2.3562737e-07, 0.0002525007, 8.555244e-05, 1.0181198e-09, 6.2870043e-07, 1.4091459e-06, 1.3009806e-06, 2.7303082e-07, 1.4669624e-09, 8.454194e-09, 1.8131272e-08, 2.4682942e-07, 1.1727215e-06, 1.1580163e-07, 4.3070994e-08, 1.063989e-07, 1.1100011e-08, 9.354254e-07, 1.5596758e-07, 1.1565727e-07, 1.0455736e-08, 1.5751957e-06, 7.423669e-08, 2.7763226e-06, 6.379263e-06, 8.358341e-08, 1.1284831e-07, 8.936478e-09, 2.7141138e-07, 1.1885099e-06, 3.5644707e-08, 4.211571e-05, 8.188094e-07, 4.756142e-07, 4.3325363e-07, 3.3440047e-06, 3.8567996e-06, 2.567217e-07, 2.7330253e-07, 5.7382164e-07, 1.7412797e-06, 1.52012235e-05, 3.6163758e-06, 4.464074e-08, 1.5914673e-06, 2.5722297e-08, 5.8252516e-07, 5.265234e-08, 8.234005e-09, 6.593523e-08, 7.2670247e-07, 6.157772e-08, 1.9472304e-06, 1.6877687e-08, 5.4864344e-08, 1.2159968e-07, 8.8625045e-09, 3.85879e-08, 1.18826236e-07, 5.029152e-09, 7.548831e-09, 4.8796755e-06, 3.368214e-06, 3.206473e-07, 2.0191665e-08, 7.773727e-07, 8.836705e-07, 2.781339e-08, 8.5779556e-07, 6.021054e-07, 5.5811405e-08, 0.00017171177, 1.3041937e-07, 1.03372e-07, 9.105134e-07, 1.9368134e-07, 3.4401072e-08, 6.993172e-10, 1.4681606e-05, 2.6645605e-06, 5.3564694e-08, 1.8430373e-07, 3.574854e-08, 4.3808715e-08, 1.1790208e-07, 2.903986e-08, 1.7527881e-07, 9.018778e-08, 5.909041e-07, 1.3226456e-06, 9.100276e-10, 9.4933505e-08, 1.7600302e-08, 1.237418e-07, 2.4791518e-06, 1.6265204e-08, 5.6816635e-08, 3.0899054e-07, 2.259528e-07, 1.2160329e-07, 1.7874806e-08, 1.1018447e-07, 8.6053426e-08, 1.6846097e-09, 1.3612078e-09, 4.0714268e-09, 3.1117708e-07, 1.8836668e-06, 2.4274023e-09, 1.6711957e-07, 1.2489967e-07, 5.7767593e-06, 0.00023884398, 4.883025e-07, 0.00018770229, 6.8687876e-07, 3.7221625e-08, 1.396456e-07, 9.718689e-07, 9.019608e-07, 3.1364755e-07, 1.7217104e-08, 4.8390746e-08, 7.264543e-09, 7.605233e-10, 1.654826e-10, 6.4277764e-08, 1.371566e-07, 9.566331e-08, 6.1089866e-09, 1.1865948e-08, 9.200144e-09, 2.860801e-09, 3.0422314e-08, 1.2621048e-08, 5.286336e-08, 6.646337e-09, 1.1934101e-09, 1.2263828e-09, 1.2506999e-09, 1.0561177e-09, 1.5004105e-09, 4.2410333e-09, 1.4444183e-08, 2.3039358e-10, 7.3044473e-09, 7.0201445e-09, 4.5222894e-08, 6.5248993e-09, 3.0367296e-08, 1.61907e-08, 2.812724e-08, 7.16595e-10, 2.4030102e-09, 1.4037422e-09, 2.8542007e-09, 6.8244588e-09, 7.394124e-07, 8.272793e-09, 9.1326186e-08, 3.4254477e-08, 3.938575e-09, 1.2116357e-08, 2.4269255e-09, 2.9024413e-08, 1.6277409e-07, 4.0866994e-09, 4.3306834e-08, 7.559003e-09, 0.29820144, 6.001986e-07, 4.3510576e-05, 6.434583e-08, 4.367675e-05, 6.187613e-06, 5.1900697e-06, 4.6378314e-06, 8.057671e-06, 5.7155186e-05, 1.4133632e-05, 1.1964084e-07, 1.1058091e-07, 4.345451e-07, 3.3152878e-07, 8.424061e-09, 8.0378465e-10, 4.2880637e-09, 4.369939e-08, 6.1343945e-07, 2.029015e-08, 3.3226485e-08, 7.760297e-09, 3.8018118e-09, 3.5106462e-09, 1.7808583e-08, 1.4098259e-08, 3.8490683e-10, 3.2085192e-09, 3.8055336e-07]),\n",
-       " (3, [1.6689529e-07, 2.2206162e-07, 2.820838e-07, 9.411482e-08, 9.196814e-08, 6.835522e-08, 1.4712062e-07, 9.119155e-06, 6.7199594e-06, 1.9530382e-06, 2.4907507e-07, 6.704823e-07, 1.3618245e-07, 3.1312987e-07, 1.2618708e-07, 8.8433814e-07, 1.3651416e-07, 1.0374823e-06, 9.697097e-07, 3.1258924e-07, 1.154241e-07, 2.7880057e-07, 7.071258e-07, 1.1773246e-06, 2.7102183e-07, 8.7961396e-08, 5.8193404e-08, 2.5870187e-07, 3.1527037e-08, 2.715556e-07, 1.865584e-07, 3.0306205e-07, 1.0248689e-07, 2.2406402e-07, 2.4086475e-07, 8.182266e-08, 3.4661014e-07, 2.965118e-07, 1.560052e-07, 2.3101217e-07, 5.0157405e-07, 1.6106436e-07, 1.7701154e-07, 7.382814e-07, 4.017762e-07, 8.348868e-08, 4.505219e-07, 3.507288e-08, 1.9216787e-07, 2.1005312e-07, 1.7100543e-06, 1.2874328e-07, 1.7951021e-07, 1.1528065e-07, 3.0493317e-07, 8.817166e-07, 7.344486e-07, 8.2197954e-07, 4.4888705e-07, 6.063366e-07, 9.589398e-07, 3.6966567e-06, 1.7984804e-06, 3.5922747e-07, 4.3303217e-07, 1.4106764e-07, 3.4898633e-07, 2.1168786e-07, 2.1905065e-07, 7.523292e-08, 1.2124268e-07, 1.6527275e-07, 6.333906e-08, 5.592894e-08, 9.339264e-08, 3.3048057e-08, 2.849944e-07, 5.0003926e-08, 3.2996897e-07, 1.093866e-06, 4.7521535e-08, 6.467596e-08, 6.746868e-07, 7.208058e-07, 1.3252958e-06, 4.4775916e-07, 1.3629414e-06, 2.6028246e-07, 1.2213138e-07, 2.378058e-07, 4.0746293e-08, 4.4012336e-07, 1.5162593e-07, 1.3352037e-06, 1.0124106e-06, 1.385228e-07, 2.0494727e-07, 1.7858578e-05, 1.1512349e-06, 8.851452e-06, 2.1538575e-07, 6.265657e-08, 5.9420086e-07, 2.688158e-07, 6.4805367e-06, 5.182306e-07, 2.372954e-07, 2.2233793e-07, 3.245263e-07, 5.0860205e-08, 1.017551e-07, 8.684927e-08, 3.1345842e-06, 2.1760503e-07, 2.1266662e-07, 5.1490655e-08, 5.8833677e-08, 7.994336e-07, 1.437624e-06, 1.8312319e-07, 4.1118618e-08, 6.079955e-07, 7.317862e-07, 1.2565258e-07, 8.145708e-07, 7.439697e-07, 4.4777835e-07, 8.953791e-08, 2.082923e-07, 3.7194494e-07, 6.495325e-08, 1.15024555e-07, 4.6843113e-07, 1.9805561e-06, 1.128121e-06, 3.502328e-08, 7.75236e-07, 1.0817615e-06, 2.0329721e-06, 3.4939458e-07, 1.3838962e-06, 5.9606386e-08, 1.0386679e-07, 2.7677422e-07, 2.562633e-07, 3.75767e-07, 2.257235e-06, 1.7437878e-07, 3.344044e-07, 1.4814579e-08, 1.4589527e-07, 9.747049e-06, 4.343081e-05, 2.4861931e-06, 4.7536546e-06, 1.7077704e-05, 4.1258518e-05, 0.00033958646, 2.7045458e-05, 2.2274935e-06, 0.0007474787, 7.4541317e-06, 3.46901e-06, 1.7526618e-06, 5.131311e-06, 5.9422626e-07, 5.283553e-06, 2.3466607e-06, 1.1319079e-06, 0.0075451825, 6.501963e-06, 1.8894696e-05, 4.274101e-05, 8.023169e-05, 4.3800146e-06, 5.5521894e-05, 7.4542e-05, 5.1395902e-05, 3.269665e-07, 6.566881e-07, 1.1947602e-06, 3.175389e-06, 4.53225e-06, 2.0213977e-06, 1.9833658e-05, 7.5548055e-06, 7.107016e-05, 5.446052e-05, 0.0003789736, 2.3369696e-05, 6.262723e-06, 8.743196e-05, 1.7344008e-05, 0.00054024416, 5.6735547e-07, 1.3520279e-06, 9.357407e-06, 1.2883928e-06, 4.2506967e-06, 2.1557997e-05, 2.9111889e-05, 6.968902e-05, 4.0829773e-06, 6.0506145e-06, 1.7851306e-05, 2.1046308e-06, 9.482371e-07, 1.9215779e-05, 1.5350582e-06, 9.981025e-07, 1.2606618e-07, 5.10712e-06, 2.7865839e-05, 7.178262e-06, 4.581452e-06, 1.8116694e-05, 5.820323e-06, 3.7669934e-06, 7.9357415e-05, 2.4837062e-05, 1.822985e-05, 1.283606e-06, 1.2288719e-05, 2.9043683e-06, 0.00011244261, 4.338252e-06, 0.00021625363, 2.9391324e-05, 1.0695196e-06, 0.00011266577, 0.40733615, 0.5772368, 0.0004146664, 1.229108e-06, 1.202781e-06, 2.706326e-05, 9.573294e-06, 1.6624028e-06, 1.3531604e-06, 4.1547428e-05, 1.3941313e-05, 3.45452e-06, 3.975819e-07, 1.14479995e-07, 3.2602754e-06, 2.4812724e-07, 5.9290517e-07, 9.359942e-06, 5.2608295e-05, 1.6942557e-05, 3.36582e-05, 3.7529944e-06, 1.7631246e-06, 6.8221685e-05, 1.913694e-07, 2.6620355e-06, 2.7133906e-06, 2.9902525e-05, 3.281977e-05, 0.00021295837, 1.4895779e-05, 1.5925854e-05, 7.54719e-07, 0.00016248274, 8.674957e-05, 7.4369905e-06, 1.7487591e-05, 5.6154302e-05, 1.0220183e-06, 2.6891967e-05, 1.2903312e-05, 1.98491e-05, 1.0393447e-05, 3.2990965e-06, 4.3320383e-06, 6.4613107e-07, 5.837787e-07, 9.864287e-06, 1.605011e-06, 5.953469e-06, 3.730937e-06, 7.4432987e-06, 1.6610296e-05, 2.930948e-05, 3.3447233e-07, 4.6493938e-06, 6.9082525e-08, 3.058961e-07, 1.7166974e-07, 1.0861472e-07, 1.1060474e-07, 9.991572e-06, 2.624814e-06, 7.4486564e-07, 1.1626346e-06, 8.506138e-07, 3.9435324e-07, 5.3783315e-06, 1.02206464e-07, 3.4676552e-07, 1.2598566e-07, 2.1253848e-07, 1.1014853e-07, 3.873366e-07, 5.1221065e-07, 7.8767215e-08, 3.3245198e-07, 1.7352278e-07, 1.7126436e-07, 7.1782176e-07, 5.609307e-07, 1.8716644e-07, 2.7722834e-07, 2.3728228e-07, 1.4343879e-06, 1.6317398e-07, 1.8679945e-07, 9.642633e-07, 6.046637e-07, 2.5942106e-07, 6.012982e-08, 3.1809796e-07, 3.3224913e-07, 2.943848e-07, 4.0296842e-07, 1.1191497e-06, 1.4413408e-07, 2.769044e-07, 1.8496877e-07, 1.05810884e-07, 5.328658e-07, 4.866973e-06, 3.224421e-05, 1.1619223e-05, 2.464877e-06, 1.8399638e-05, 1.6188493e-06, 2.0739626e-06, 9.169329e-05, 2.8408182e-05, 1.0441167e-06, 2.2457004e-06, 4.574065e-06, 6.362242e-07, 9.629592e-07, 1.0002079e-05, 7.1745495e-07, 2.1475851e-07, 1.01941005e-05, 8.618323e-07, 1.869045e-06, 6.2070296e-07, 1.1434444e-06, 2.3597759e-06, 6.7796793e-07, 1.271199e-05, 1.1577484e-06, 2.368491e-07, 4.55484e-07, 2.5140366e-06, 5.6595123e-08, 7.3302494e-06, 4.584962e-06, 2.463876e-06, 1.1013761e-07, 7.970608e-07, 1.06286215e-07, 6.5447483e-07, 5.9785117e-07, 3.151479e-07, 1.0185223e-06, 2.0698292e-06, 2.019361e-06, 1.1943068e-06, 9.928024e-07, 8.1185874e-07, 1.9383758e-06, 8.319345e-07, 6.315681e-07, 1.5005149e-07, 1.4024852e-06, 2.58234e-07, 4.5330773e-07, 1.0390844e-06, 6.014614e-07, 5.0667715e-08, 4.292828e-08, 2.428166e-07, 2.1167232e-07, 5.6699145e-07, 3.6647265e-07, 7.7877735e-07, 9.033721e-08, 7.109228e-08, 1.13489925e-07, 5.4377375e-07, 1.7701846e-07, 7.7141095e-08, 9.243051e-07, 2.0634916e-07, 7.603061e-07, 2.1217143e-06, 1.2784554e-06, 6.703036e-08, 2.8089295e-07, 8.370749e-07, 3.334441e-06, 4.4463496e-07, 2.1578214e-07, 8.6750146e-07, 2.0263879e-07, 1.757333e-06, 4.232752e-06, 5.2864135e-07, 1.0017834e-06, 8.292726e-06, 2.0218168e-07, 5.6843703e-07, 3.1687039e-06, 2.3628252e-06, 1.3313359e-06, 2.1216838e-06, 3.529718e-07, 1.0109056e-06, 5.4811755e-07, 1.5179745e-06, 1.1439428e-06, 2.7279775e-06, 2.491207e-05, 2.4989247e-06, 1.0711284e-06, 3.4363297e-06, 1.795127e-06, 5.6359966e-08, 4.6255496e-05, 7.617995e-06, 1.7480146e-06, 1.513277e-06, 4.223011e-06, 2.5502782e-06, 4.8256497e-06, 6.389756e-07, 2.050033e-06, 1.9788192e-06, 2.5552567e-06, 6.402927e-07, 1.5162955e-05, 1.9254494e-06, 1.4773743e-05, 1.5308382e-07, 6.756578e-07, 3.368628e-06, 3.2261296e-06, 1.3053974e-06, 8.031118e-06, 8.357667e-07, 4.2610545e-06, 5.840485e-06, 7.83309e-07, 6.467494e-07, 4.065039e-06, 1.2906764e-06, 3.6102936e-06, 1.5432053e-05, 1.8839935e-06, 9.1145074e-07, 9.640519e-07, 1.5100665e-06, 1.49469e-06, 3.234839e-06, 1.8960048e-06, 9.3880084e-07, 1.1120671e-06, 1.266507e-06, 3.204753e-06, 2.2967995e-06, 8.827923e-06, 1.5669655e-06, 2.128402e-06, 4.6287373e-06, 3.5791315e-06, 1.1039351e-06, 7.966436e-07, 2.8899622e-06, 1.6511663e-07, 2.5859703e-07, 5.687974e-06, 7.1050804e-06, 4.2829834e-07, 1.4763658e-05, 5.886087e-07, 3.8952072e-07, 1.2212666e-06, 4.2225375e-07, 2.9476307e-06, 2.9590444e-07, 1.647012e-06, 1.138658e-05, 6.443352e-06, 4.116174e-06, 2.3373326e-07, 4.594675e-06, 5.326156e-06, 2.7369208e-06, 9.157579e-06, 1.6969457e-06, 1.5630312e-06, 7.475492e-07, 3.4837499e-06, 4.043013e-06, 3.866793e-07, 1.3447072e-06, 2.7079177e-06, 7.07651e-06, 3.3592798e-06, 2.4041381e-05, 1.5609503e-06, 1.4794115e-07, 1.8450921e-06, 3.6126664e-06, 8.430876e-07, 1.6879924e-06, 1.49956795e-05, 8.848961e-06, 7.616669e-07, 8.442579e-06, 5.06263e-07, 4.8507803e-07, 9.1500846e-07, 2.4074225e-06, 6.2053016e-07, 9.895185e-07, 2.1231817e-06, 5.107417e-06, 1.9196445e-06, 2.279443e-06, 1.78816e-07, 2.918832e-06, 1.6303065e-06, 1.3995557e-05, 3.7271278e-08, 7.159881e-07, 1.059618e-06, 2.258917e-06, 2.0854193e-07, 2.4586356e-07, 5.275983e-07, 2.4461443e-07, 7.5761e-07, 4.2390033e-07, 1.6908271e-07, 1.8666652e-06, 1.8524983e-05, 1.4771988e-06, 2.6837728e-07, 1.7947718e-07, 1.140592e-06, 1.8138328e-06, 6.742733e-05, 2.4207802e-06, 7.482425e-07, 4.8928086e-07, 4.0176797e-06, 1.5053209e-05, 2.5682698e-07, 6.180497e-07, 1.2896313e-05, 1.0683698e-06, 7.219537e-06, 2.5994248e-06, 7.657017e-07, 1.0884103e-06, 4.019638e-06, 1.1041446e-06, 8.6559413e-07, 3.2376015e-06, 5.8260262e-08, 3.7094105e-07, 1.7066193e-05, 9.563972e-07, 2.7943852e-06, 8.446875e-07, 8.798296e-06, 2.0743909e-07, 3.720597e-06, 9.2031805e-05, 3.2299022e-07, 1.3909248e-06, 1.5150531e-06, 1.1316872e-05, 3.735218e-07, 8.270013e-06, 9.53729e-08, 5.2934156e-05, 1.0523074e-05, 2.0331552e-07, 1.072588e-06, 2.2581287e-06, 1.7145777e-06, 1.3304353e-07, 7.8983436e-07, 8.3505965e-06, 4.7793077e-07, 8.633999e-06, 5.063731e-07, 2.9272896e-06, 1.6326154e-06, 1.6821863e-06, 1.6723036e-05, 2.907346e-07, 1.0632824e-06, 1.8857632e-05, 8.1562877e-07, 3.4525522e-07, 2.9374132e-06, 1.6777803e-06, 2.093815e-07, 5.092932e-06, 2.2783825e-06, 3.2003124e-06, 3.72425e-06, 1.1175212e-06, 7.551085e-07, 1.2608814e-06, 7.110252e-06, 6.5194e-07, 1.294883e-06, 1.4719675e-06, 6.0391164e-07, 2.758968e-06, 1.3804836e-06, 1.03495595e-05, 1.0754078e-06, 4.651161e-07, 8.321808e-08, 3.883249e-07, 4.3750847e-06, 6.9287e-06, 1.0790991e-06, 2.0175928e-07, 1.1477083e-06, 9.2724866e-07, 4.4966427e-07, 3.768524e-06, 8.53336e-08, 7.5422184e-07, 5.943853e-06, 5.7822704e-06, 1.1200253e-06, 2.2967261e-07, 1.7970591e-06, 5.6860836e-07, 1.3928349e-06, 6.2377126e-06, 5.0388206e-07, 6.703836e-06, 2.8018142e-06, 3.1146221e-06, 5.6211627e-07, 2.3552186e-06, 7.2731456e-07, 8.2250574e-07, 1.7007797e-07, 1.4904311e-06, 2.3674602e-06, 8.284086e-06, 1.5256389e-06, 5.753343e-06, 1.6530263e-06, 4.4300623e-07, 2.975805e-06, 1.5939414e-06, 1.2028959e-06, 1.9596916e-06, 5.4057625e-07, 2.5539276e-07, 3.354089e-05, 4.7010667e-07, 7.318403e-06, 6.808285e-07, 1.2269537e-05, 2.980676e-06, 1.2634037e-07, 5.5363403e-06, 3.1773146e-05, 9.6223175e-08, 1.5475484e-07, 1.4305355e-06, 3.206239e-08, 5.6593562e-06, 2.2849756e-06, 4.1018943e-06, 4.2290035e-06, 7.9598516e-07, 8.535069e-08, 3.1764587e-06, 2.3981077e-06, 2.4379176e-06, 2.924787e-06, 1.9348952e-05, 3.043286e-05, 3.9040322e-07, 1.9553707e-07, 3.6785837e-06, 9.770004e-07, 2.1725493e-06, 9.120881e-07, 2.3135665e-06, 1.628987e-06, 2.6829575e-06, 5.118893e-07, 6.0893008e-06, 2.598494e-07, 2.110724e-07, 3.142682e-07, 8.9087973e-07, 1.6903034e-05, 1.8139677e-06, 4.9642364e-07, 9.3815464e-07, 1.1011582e-06, 8.299229e-06, 4.316495e-06, 1.5535351e-06, 1.2686715e-07, 7.4317227e-06, 2.2307408e-07, 7.130532e-07, 1.6027021e-05, 2.522636e-07, 2.934066e-07, 1.3851732e-06, 5.2406995e-07, 3.4384148e-06, 1.7058935e-07, 2.2353183e-06, 1.1527202e-06, 2.0424222e-06, 9.0871504e-07, 9.279175e-07, 1.2784419e-06, 1.0543832e-06, 1.288475e-06, 7.744535e-07, 6.339913e-06, 1.9381142e-07, 3.6516923e-07, 5.14587e-06, 1.8609968e-06, 2.1099022e-06, 6.7389306e-06, 1.0700466e-05, 6.8114905e-06, 2.9642656e-06, 7.136022e-07, 7.744875e-07, 3.7833279e-06, 3.0161364e-06, 6.936075e-07, 4.3780747e-07, 1.0460443e-06, 8.8778995e-07, 2.3039381e-06, 1.7693857e-06, 1.115564e-06, 1.661298e-06, 2.6530446e-07, 4.3556693e-07, 3.4118489e-06, 1.5530655e-06, 7.466016e-07, 5.063683e-07, 1.3009085e-06, 1.5598796e-05, 1.0366763e-06, 1.1196889e-06, 3.0060828e-06, 1.356155e-06, 9.820625e-07, 2.1989595e-06, 1.6922332e-06, 1.3272423e-06, 1.9030645e-06, 2.1997091e-07, 8.5709223e-07, 3.8253297e-06, 3.4569968e-07, 2.6077578e-06, 3.483567e-06, 3.6281364e-07, 1.7578661e-06, 1.9342892e-06, 1.7658742e-06, 3.907155e-06, 8.981448e-06, 8.9915403e-07, 2.0014852e-07, 4.210974e-06, 4.8389336e-07, 2.2567508e-06, 8.9751427e-07, 6.308779e-08, 5.069776e-07, 4.775052e-07, 8.122239e-06, 5.8180678e-05, 1.3861193e-06, 3.4600296e-06, 6.8834534e-06, 1.4960676e-06, 1.9307035e-07, 1.0058861e-05, 1.646274e-06, 2.423841e-06, 1.2812272e-06, 6.536019e-08, 1.56902e-06, 7.662522e-06, 1.6937428e-06, 1.2506719e-06, 3.169654e-07, 1.827239e-07, 5.736246e-07, 1.476168e-06, 5.8050346e-06, 2.8127104e-06, 8.41238e-07, 9.861714e-07, 3.964515e-06, 1.5863016e-06, 5.7537403e-07, 1.9392207e-06, 6.138479e-07, 1.4168915e-06, 1.0203871e-06, 1.4193434e-06, 7.2978736e-07, 2.6645528e-06, 4.362764e-06, 5.1138727e-07, 1.5418674e-06, 2.1608723e-06, 4.441137e-07, 2.1684011e-06, 5.7982095e-07, 4.208252e-06, 6.2694644e-06, 1.0910754e-07, 8.9542823e-07, 7.5162893e-07, 1.2211142e-06, 5.671848e-06, 8.001827e-05, 5.315375e-07, 1.2871044e-06, 2.5712022e-06, 1.929912e-07, 4.3394084e-07, 3.7587375e-07, 1.1067423e-06, 1.4056023e-06, 3.7882778e-06, 4.107226e-06, 8.979295e-07, 9.980064e-07, 7.4344875e-06, 8.000773e-07, 3.3249592e-06, 6.2859344e-06, 1.2670778e-05, 6.181024e-06, 5.992325e-07, 1.4047648e-06, 1.2503236e-06, 3.9425134e-07, 2.5586876e-06, 3.5965681e-06, 3.2103972e-06, 5.080649e-08, 7.5429407e-06, 1.1273129e-05, 4.050838e-07, 6.827945e-06, 6.557428e-06, 1.5494928e-06, 7.5670454e-07, 5.3732815e-06, 3.926059e-06, 6.872614e-07, 2.1881797e-06, 8.326354e-07, 4.418257e-07, 2.5343563e-06, 1.6838235e-06, 8.0051086e-07, 5.0417003e-07, 4.4274334e-06, 1.7458055e-06, 1.312581e-05, 6.8006834e-06, 6.6938344e-07, 1.9611891e-06, 4.05965e-06, 1.2272673e-05, 4.318187e-06, 2.5695033e-06, 3.420368e-06, 1.8288926e-06, 7.090443e-07, 1.1700457e-06, 1.5462392e-06, 8.471674e-07, 1.19016195e-05, 1.0063437e-07, 3.5280016e-07, 8.083196e-07, 7.831007e-07, 2.2469967e-06, 3.2198745e-06, 5.46896e-06, 2.3607876e-07, 4.5549596e-06, 1.777636e-06, 1.8209928e-06, 1.0895079e-06, 3.649106e-07, 3.4167476e-07, 3.4340349e-07, 8.9606124e-07, 1.8848374e-05, 4.4522218e-07, 1.1343252e-06, 6.218925e-06, 6.378472e-08, 6.0418674e-06, 4.010594e-06, 8.1876243e-07, 2.5506273e-05, 2.0598866e-06, 3.0352992e-06, 7.1552597e-07, 2.509204e-07, 7.4251875e-07, 1.1561078e-06, 8.450162e-07, 1.1495174e-07, 2.30015e-07, 4.216791e-07, 3.6176586e-06, 1.8345672e-07, 3.6328088e-06, 1.041257e-06, 6.1919064e-07, 1.6489703e-06, 1.23177115e-05, 3.0211654e-06, 1.7541284e-06, 1.4839601e-07, 6.1302976e-06, 2.0154073e-06, 3.6311096e-07, 7.3696737e-07, 4.370341e-07, 1.5177255e-06, 1.2590387e-07, 2.3164412e-06, 7.050975e-07, 8.3458315e-07, 4.139802e-06, 5.4616526e-06, 3.749249e-06, 1.1025905e-06, 8.433087e-07, 2.094582e-07, 3.424967e-06, 5.2668106e-06, 5.819143e-07, 1.4725024e-06, 1.2079207e-06, 3.0717872e-06, 1.0360299e-06, 2.678918e-06, 2.2673117e-05, 4.2749342e-08, 7.203822e-06, 1.2097364e-06, 3.5521066e-07, 1.5892792e-05, 7.6659603e-07, 1.5075173e-07, 1.3642651e-06, 1.4929412e-07, 2.595537e-07, 2.5446576e-07, 1.00538834e-07, 2.9553246e-07, 6.4130705e-07, 2.1728714e-07, 6.822549e-06, 7.864933e-05])]"
+       "[(1, u'dependent_var', u'58', 0.14084539, 2),\n",
+       " (1, u'dependent_var', u'49', 0.05819455, 3),\n",
+       " (1, u'dependent_var', u'62', 0.05564801, 4)]"
       ]
      },
-     "execution_count": 7,
+     "execution_count": 13,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -4849,15 +4845,22 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 11,
+   "execution_count": 14,
    "metadata": {},
    "outputs": [
     {
+     "name": "stderr",
+     "output_type": "stream",
+     "text": [
+      "Using TensorFlow backend.\n"
+     ]
+    },
+    {
      "name": "stdout",
      "output_type": "stream",
      "text": [
-      "10 rows affected.\n",
-      "(10, 1, 1000)\n"
+      "10000 rows affected.\n",
+      "(10000, 1)\n"
      ]
     }
    ],
@@ -4865,14 +4868,14 @@
     "from keras.applications.vgg16 import decode_predictions\n",
     "import numpy as np\n",
     "\n",
-    "prob_vector = %sql select prob from imagenet_predict_vgg16_prob order by id;\n",
+    "prob_vector = %sql select prob from imagenet_predict_vgg16_prob order by id, rank;\n",
     "print np.array(prob_vector).shape\n",
     "label = decode_predictions(np.array(prob_vector).reshape(10,1000))"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 9,
+   "execution_count": 15,
    "metadata": {},
    "outputs": [
     {
@@ -4882,7 +4885,7 @@
        "<IPython.core.display.Image object>"
       ]
      },
-     "execution_count": 9,
+     "execution_count": 15,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -4894,20 +4897,20 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 12,
+   "execution_count": 17,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "text/plain": [
-       "[(u'n01751748', u'sea_snake', 0.5346089),\n",
-       " (u'n01737021', u'water_snake', 0.1408455),\n",
-       " (u'n01697457', u'African_crocodile', 0.05819438),\n",
-       " (u'n01744401', u'rock_python', 0.055647947),\n",
-       " (u'n01755581', u'diamondback', 0.041236367)]"
+       "[(u'n01440764', u'tench', 0.5346085),\n",
+       " (u'n01443537', u'goldfish', 0.14084539),\n",
+       " (u'n01484850', u'great_white_shark', 0.05819455),\n",
+       " (u'n01491361', u'tiger_shark', 0.05564801),\n",
+       " (u'n01494475', u'hammerhead', 0.0412363)]"
       ]
      },
-     "execution_count": 12,
+     "execution_count": 17,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -4918,7 +4921,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 49,
+   "execution_count": 8,
    "metadata": {},
    "outputs": [
     {
@@ -4928,7 +4931,7 @@
        "<IPython.core.display.Image object>"
       ]
      },
-     "execution_count": 49,
+     "execution_count": 8,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -4939,20 +4942,20 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 13,
+   "execution_count": 18,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "text/plain": [
-       "[(u'n04228054', u'ski', 0.6965568),\n",
-       " (u'n09193705', u'alp', 0.29820144),\n",
-       " (u'n04208210', u'shovel', 0.0016679094),\n",
-       " (u'n03792972', u'mountain_tent', 0.0007202051),\n",
-       " (u'n03218198', u'dogsled', 0.00038468122)]"
+       "[(u'n01440764', u'tench', 0.6965566),\n",
+       " (u'n01443537', u'goldfish', 0.29820165),\n",
+       " (u'n01484850', u'great_white_shark', 0.0016679137),\n",
+       " (u'n01491361', u'tiger_shark', 0.0007202063),\n",
+       " (u'n01494475', u'hammerhead', 0.00038468183)]"
       ]
      },
-     "execution_count": 13,
+     "execution_count": 18,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -4984,20 +4987,20 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 14,
+   "execution_count": 19,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "text/plain": [
-       "[(u'n02106030', u'collie', 0.5772368),\n",
-       " (u'n02105855', u'Shetland_sheepdog', 0.40733615),\n",
-       " (u'n02090622', u'borzoi', 0.0075451825),\n",
-       " (u'n02088094', u'Afghan_hound', 0.0007474787),\n",
-       " (u'n02096294', u'Australian_terrier', 0.00054024416)]"
+       "[(u'n01440764', u'tench', 0.57723767),\n",
+       " (u'n01443537', u'goldfish', 0.40733525),\n",
+       " (u'n01484850', u'great_white_shark', 0.007545187),\n",
+       " (u'n01491361', u'tiger_shark', 0.00074747804),\n",
+       " (u'n01494475', u'hammerhead', 0.000540244)]"
       ]
      },
-     "execution_count": 14,
+     "execution_count": 19,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -5076,7 +5079,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 10,
+   "execution_count": 13,
    "metadata": {},
    "outputs": [
     {
@@ -5102,7 +5105,7 @@
        "[(15867L,)]"
       ]
      },
-     "execution_count": 10,
+     "execution_count": 13,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -5122,7 +5125,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 4,
+   "execution_count": 14,
    "metadata": {},
    "outputs": [
     {
@@ -5148,7 +5151,7 @@
        "[(Decimal('68.27'),)]"
       ]
      },
-     "execution_count": 4,
+     "execution_count": 14,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -5179,7 +5182,7 @@
    "name": "python",
    "nbconvert_exporter": "python",
    "pygments_lexer": "ipython2",
-   "version": "2.7.10"
+   "version": "2.7.16"
   }
  },
  "nbformat": 4,
diff --git a/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-model-selection-MLP-v1-checkpoint.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-model-selection-MLP-v1-checkpoint.ipynb
new file mode 100644
index 0000000..55085b9
--- /dev/null
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/MADlib-Keras-model-selection-MLP-v1-checkpoint.ipynb
@@ -0,0 +1,6276 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Model Selection for Multilayer Perceptron Using Keras and MADlib\n",
+    "\n",
+    "E2E classification example using MADlib calling a Keras MLP for different hyperparameters and model architectures.\n",
+    "\n",
+    "Deep learning works best on very large datasets, but that is not convenient for a quick introduction to the syntax.  So in this workbook we use the well known iris data set from https://archive.ics.uci.edu/ml/datasets/iris to help get you started.  It is similar to the example in user docs http://madlib.apache.org/docs/latest/index.html\n",
+    "\n",
+    "For more realistic examples please refer to the deep learning notebooks at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#class\">Classification</a>\n",
+    "\n",
+    "* <a href=\"#create_input_data\">1. Create input data</a>\n",
+    "\n",
+    "* <a href=\"#pp\">2. Call preprocessor for deep learning</a>\n",
+    "\n",
+    "* <a href=\"#load\">3. Define and load model architecture</a>\n",
+    "\n",
+    "* <a href=\"#def_mst\">4. Define and load model selection tuples</a>\n",
+    "\n",
+    "* <a href=\"#train\">5. Train</a>\n",
+    "\n",
+    "* <a href=\"#eval\">6. Evaluate</a>\n",
+    "\n",
+    "* <a href=\"#pred\">7. Predict</a>\n",
+    "\n",
+    "<a href=\"#class2\">Classification with Other Parameters</a>\n",
+    "\n",
+    "* <a href=\"#val_dataset\">1. Validation dataset</a>\n",
+    "\n",
+    "* <a href=\"#pred_prob\">2. Predict probabilities</a>\n",
+    "\n",
+    "* <a href=\"#warm_start\">3. Warm start</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class\"></a>\n",
+    "# Classification\n",
+    "\n",
+    "<a id=\"create_input_data\"></a>\n",
+    "# 1.  Create input data\n",
+    "\n",
+    "Load iris data set."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "150 rows affected.\n",
+      "150 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>attributes</th>\n",
+       "        <th>class_text</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>[Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>21</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>22</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>25</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>35</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>36</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>37</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>39</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>40</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>42</td>\n",
+       "        <td>[Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>44</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>45</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>46</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>47</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>48</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>[Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>50</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>[Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>53</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>58</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>59</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>62</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>65</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>69</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>79</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>86</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>87</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>88</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>91</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>95</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>97</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>[Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>105</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>106</td>\n",
+       "        <td>[Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>[Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>109</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>110</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>118</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>124</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>129</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>130</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>[Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>[Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>133</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>139</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>140</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>141</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>142</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (2, [Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (3, [Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (4, [Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (5, [Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (6, [Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (7, [Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (8, [Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (9, [Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (10, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (11, [Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (12, [Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (13, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (14, [Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (15, [Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (16, [Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (17, [Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (18, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (19, [Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (20, [Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (21, [Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (22, [Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (23, [Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (24, [Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')], u'Iris-setosa'),\n",
+       " (25, [Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (26, [Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (27, [Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (28, [Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (29, [Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (30, [Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (31, [Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (32, [Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (33, [Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (34, [Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (35, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (36, [Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (37, [Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (38, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (39, [Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (40, [Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (41, [Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (42, [Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (43, [Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (44, [Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')], u'Iris-setosa'),\n",
+       " (45, [Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (46, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (47, [Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (48, [Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (49, [Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (50, [Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (51, [Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (52, [Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (53, [Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (54, [Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (55, [Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (56, [Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (57, [Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (58, [Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (59, [Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (60, [Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (61, [Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (62, [Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (63, [Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (64, [Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (65, [Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (66, [Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (67, [Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (68, [Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (69, [Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (70, [Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (71, [Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')], u'Iris-versicolor'),\n",
+       " (72, [Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (73, [Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (74, [Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (75, [Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (76, [Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (77, [Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (78, [Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')], u'Iris-versicolor'),\n",
+       " (79, [Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (80, [Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (81, [Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (82, [Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (83, [Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (84, [Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (85, [Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (86, [Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (87, [Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (88, [Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (89, [Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (90, [Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (91, [Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (92, [Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (93, [Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (94, [Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (95, [Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (96, [Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (97, [Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (98, [Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (99, [Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (100, [Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (101, [Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (102, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (103, [Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (104, [Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (105, [Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (106, [Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (107, [Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')], u'Iris-virginica'),\n",
+       " (108, [Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (109, [Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (110, [Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (111, [Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (112, [Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (113, [Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (114, [Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (115, [Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (116, [Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (117, [Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (118, [Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (119, [Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (120, [Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (121, [Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (122, [Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (123, [Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (124, [Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (125, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (126, [Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (127, [Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (128, [Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (129, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (130, [Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')], u'Iris-virginica'),\n",
+       " (131, [Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (132, [Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (133, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (134, [Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (135, [Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')], u'Iris-virginica'),\n",
+       " (136, [Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (137, [Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (138, [Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (139, [Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (140, [Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (141, [Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (142, [Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (143, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (144, [Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (145, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (146, [Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (147, [Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (148, [Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (149, [Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (150, [Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')], u'Iris-virginica')]"
+      ]
+     },
+     "execution_count": 4,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql \n",
+    "DROP TABLE IF EXISTS iris_data;\n",
+    "\n",
+    "CREATE TABLE iris_data(\n",
+    "    id serial,\n",
+    "    attributes numeric[],\n",
+    "    class_text varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO iris_data(id, attributes, class_text) VALUES\n",
+    "(1,ARRAY[5.1,3.5,1.4,0.2],'Iris-setosa'),\n",
+    "(2,ARRAY[4.9,3.0,1.4,0.2],'Iris-setosa'),\n",
+    "(3,ARRAY[4.7,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(4,ARRAY[4.6,3.1,1.5,0.2],'Iris-setosa'),\n",
+    "(5,ARRAY[5.0,3.6,1.4,0.2],'Iris-setosa'),\n",
+    "(6,ARRAY[5.4,3.9,1.7,0.4],'Iris-setosa'),\n",
+    "(7,ARRAY[4.6,3.4,1.4,0.3],'Iris-setosa'),\n",
+    "(8,ARRAY[5.0,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(9,ARRAY[4.4,2.9,1.4,0.2],'Iris-setosa'),\n",
+    "(10,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(11,ARRAY[5.4,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(12,ARRAY[4.8,3.4,1.6,0.2],'Iris-setosa'),\n",
+    "(13,ARRAY[4.8,3.0,1.4,0.1],'Iris-setosa'),\n",
+    "(14,ARRAY[4.3,3.0,1.1,0.1],'Iris-setosa'),\n",
+    "(15,ARRAY[5.8,4.0,1.2,0.2],'Iris-setosa'),\n",
+    "(16,ARRAY[5.7,4.4,1.5,0.4],'Iris-setosa'),\n",
+    "(17,ARRAY[5.4,3.9,1.3,0.4],'Iris-setosa'),\n",
+    "(18,ARRAY[5.1,3.5,1.4,0.3],'Iris-setosa'),\n",
+    "(19,ARRAY[5.7,3.8,1.7,0.3],'Iris-setosa'),\n",
+    "(20,ARRAY[5.1,3.8,1.5,0.3],'Iris-setosa'),\n",
+    "(21,ARRAY[5.4,3.4,1.7,0.2],'Iris-setosa'),\n",
+    "(22,ARRAY[5.1,3.7,1.5,0.4],'Iris-setosa'),\n",
+    "(23,ARRAY[4.6,3.6,1.0,0.2],'Iris-setosa'),\n",
+    "(24,ARRAY[5.1,3.3,1.7,0.5],'Iris-setosa'),\n",
+    "(25,ARRAY[4.8,3.4,1.9,0.2],'Iris-setosa'),\n",
+    "(26,ARRAY[5.0,3.0,1.6,0.2],'Iris-setosa'),\n",
+    "(27,ARRAY[5.0,3.4,1.6,0.4],'Iris-setosa'),\n",
+    "(28,ARRAY[5.2,3.5,1.5,0.2],'Iris-setosa'),\n",
+    "(29,ARRAY[5.2,3.4,1.4,0.2],'Iris-setosa'),\n",
+    "(30,ARRAY[4.7,3.2,1.6,0.2],'Iris-setosa'),\n",
+    "(31,ARRAY[4.8,3.1,1.6,0.2],'Iris-setosa'),\n",
+    "(32,ARRAY[5.4,3.4,1.5,0.4],'Iris-setosa'),\n",
+    "(33,ARRAY[5.2,4.1,1.5,0.1],'Iris-setosa'),\n",
+    "(34,ARRAY[5.5,4.2,1.4,0.2],'Iris-setosa'),\n",
+    "(35,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(36,ARRAY[5.0,3.2,1.2,0.2],'Iris-setosa'),\n",
+    "(37,ARRAY[5.5,3.5,1.3,0.2],'Iris-setosa'),\n",
+    "(38,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(39,ARRAY[4.4,3.0,1.3,0.2],'Iris-setosa'),\n",
+    "(40,ARRAY[5.1,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(41,ARRAY[5.0,3.5,1.3,0.3],'Iris-setosa'),\n",
+    "(42,ARRAY[4.5,2.3,1.3,0.3],'Iris-setosa'),\n",
+    "(43,ARRAY[4.4,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(44,ARRAY[5.0,3.5,1.6,0.6],'Iris-setosa'),\n",
+    "(45,ARRAY[5.1,3.8,1.9,0.4],'Iris-setosa'),\n",
+    "(46,ARRAY[4.8,3.0,1.4,0.3],'Iris-setosa'),\n",
+    "(47,ARRAY[5.1,3.8,1.6,0.2],'Iris-setosa'),\n",
+    "(48,ARRAY[4.6,3.2,1.4,0.2],'Iris-setosa'),\n",
+    "(49,ARRAY[5.3,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(50,ARRAY[5.0,3.3,1.4,0.2],'Iris-setosa'),\n",
+    "(51,ARRAY[7.0,3.2,4.7,1.4],'Iris-versicolor'),\n",
+    "(52,ARRAY[6.4,3.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(53,ARRAY[6.9,3.1,4.9,1.5],'Iris-versicolor'),\n",
+    "(54,ARRAY[5.5,2.3,4.0,1.3],'Iris-versicolor'),\n",
+    "(55,ARRAY[6.5,2.8,4.6,1.5],'Iris-versicolor'),\n",
+    "(56,ARRAY[5.7,2.8,4.5,1.3],'Iris-versicolor'),\n",
+    "(57,ARRAY[6.3,3.3,4.7,1.6],'Iris-versicolor'),\n",
+    "(58,ARRAY[4.9,2.4,3.3,1.0],'Iris-versicolor'),\n",
+    "(59,ARRAY[6.6,2.9,4.6,1.3],'Iris-versicolor'),\n",
+    "(60,ARRAY[5.2,2.7,3.9,1.4],'Iris-versicolor'),\n",
+    "(61,ARRAY[5.0,2.0,3.5,1.0],'Iris-versicolor'),\n",
+    "(62,ARRAY[5.9,3.0,4.2,1.5],'Iris-versicolor'),\n",
+    "(63,ARRAY[6.0,2.2,4.0,1.0],'Iris-versicolor'),\n",
+    "(64,ARRAY[6.1,2.9,4.7,1.4],'Iris-versicolor'),\n",
+    "(65,ARRAY[5.6,2.9,3.6,1.3],'Iris-versicolor'),\n",
+    "(66,ARRAY[6.7,3.1,4.4,1.4],'Iris-versicolor'),\n",
+    "(67,ARRAY[5.6,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(68,ARRAY[5.8,2.7,4.1,1.0],'Iris-versicolor'),\n",
+    "(69,ARRAY[6.2,2.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(70,ARRAY[5.6,2.5,3.9,1.1],'Iris-versicolor'),\n",
+    "(71,ARRAY[5.9,3.2,4.8,1.8],'Iris-versicolor'),\n",
+    "(72,ARRAY[6.1,2.8,4.0,1.3],'Iris-versicolor'),\n",
+    "(73,ARRAY[6.3,2.5,4.9,1.5],'Iris-versicolor'),\n",
+    "(74,ARRAY[6.1,2.8,4.7,1.2],'Iris-versicolor'),\n",
+    "(75,ARRAY[6.4,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(76,ARRAY[6.6,3.0,4.4,1.4],'Iris-versicolor'),\n",
+    "(77,ARRAY[6.8,2.8,4.8,1.4],'Iris-versicolor'),\n",
+    "(78,ARRAY[6.7,3.0,5.0,1.7],'Iris-versicolor'),\n",
+    "(79,ARRAY[6.0,2.9,4.5,1.5],'Iris-versicolor'),\n",
+    "(80,ARRAY[5.7,2.6,3.5,1.0],'Iris-versicolor'),\n",
+    "(81,ARRAY[5.5,2.4,3.8,1.1],'Iris-versicolor'),\n",
+    "(82,ARRAY[5.5,2.4,3.7,1.0],'Iris-versicolor'),\n",
+    "(83,ARRAY[5.8,2.7,3.9,1.2],'Iris-versicolor'),\n",
+    "(84,ARRAY[6.0,2.7,5.1,1.6],'Iris-versicolor'),\n",
+    "(85,ARRAY[5.4,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(86,ARRAY[6.0,3.4,4.5,1.6],'Iris-versicolor'),\n",
+    "(87,ARRAY[6.7,3.1,4.7,1.5],'Iris-versicolor'),\n",
+    "(88,ARRAY[6.3,2.3,4.4,1.3],'Iris-versicolor'),\n",
+    "(89,ARRAY[5.6,3.0,4.1,1.3],'Iris-versicolor'),\n",
+    "(90,ARRAY[5.5,2.5,4.0,1.3],'Iris-versicolor'),\n",
+    "(91,ARRAY[5.5,2.6,4.4,1.2],'Iris-versicolor'),\n",
+    "(92,ARRAY[6.1,3.0,4.6,1.4],'Iris-versicolor'),\n",
+    "(93,ARRAY[5.8,2.6,4.0,1.2],'Iris-versicolor'),\n",
+    "(94,ARRAY[5.0,2.3,3.3,1.0],'Iris-versicolor'),\n",
+    "(95,ARRAY[5.6,2.7,4.2,1.3],'Iris-versicolor'),\n",
+    "(96,ARRAY[5.7,3.0,4.2,1.2],'Iris-versicolor'),\n",
+    "(97,ARRAY[5.7,2.9,4.2,1.3],'Iris-versicolor'),\n",
+    "(98,ARRAY[6.2,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(99,ARRAY[5.1,2.5,3.0,1.1],'Iris-versicolor'),\n",
+    "(100,ARRAY[5.7,2.8,4.1,1.3],'Iris-versicolor'),\n",
+    "(101,ARRAY[6.3,3.3,6.0,2.5],'Iris-virginica'),\n",
+    "(102,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(103,ARRAY[7.1,3.0,5.9,2.1],'Iris-virginica'),\n",
+    "(104,ARRAY[6.3,2.9,5.6,1.8],'Iris-virginica'),\n",
+    "(105,ARRAY[6.5,3.0,5.8,2.2],'Iris-virginica'),\n",
+    "(106,ARRAY[7.6,3.0,6.6,2.1],'Iris-virginica'),\n",
+    "(107,ARRAY[4.9,2.5,4.5,1.7],'Iris-virginica'),\n",
+    "(108,ARRAY[7.3,2.9,6.3,1.8],'Iris-virginica'),\n",
+    "(109,ARRAY[6.7,2.5,5.8,1.8],'Iris-virginica'),\n",
+    "(110,ARRAY[7.2,3.6,6.1,2.5],'Iris-virginica'),\n",
+    "(111,ARRAY[6.5,3.2,5.1,2.0],'Iris-virginica'),\n",
+    "(112,ARRAY[6.4,2.7,5.3,1.9],'Iris-virginica'),\n",
+    "(113,ARRAY[6.8,3.0,5.5,2.1],'Iris-virginica'),\n",
+    "(114,ARRAY[5.7,2.5,5.0,2.0],'Iris-virginica'),\n",
+    "(115,ARRAY[5.8,2.8,5.1,2.4],'Iris-virginica'),\n",
+    "(116,ARRAY[6.4,3.2,5.3,2.3],'Iris-virginica'),\n",
+    "(117,ARRAY[6.5,3.0,5.5,1.8],'Iris-virginica'),\n",
+    "(118,ARRAY[7.7,3.8,6.7,2.2],'Iris-virginica'),\n",
+    "(119,ARRAY[7.7,2.6,6.9,2.3],'Iris-virginica'),\n",
+    "(120,ARRAY[6.0,2.2,5.0,1.5],'Iris-virginica'),\n",
+    "(121,ARRAY[6.9,3.2,5.7,2.3],'Iris-virginica'),\n",
+    "(122,ARRAY[5.6,2.8,4.9,2.0],'Iris-virginica'),\n",
+    "(123,ARRAY[7.7,2.8,6.7,2.0],'Iris-virginica'),\n",
+    "(124,ARRAY[6.3,2.7,4.9,1.8],'Iris-virginica'),\n",
+    "(125,ARRAY[6.7,3.3,5.7,2.1],'Iris-virginica'),\n",
+    "(126,ARRAY[7.2,3.2,6.0,1.8],'Iris-virginica'),\n",
+    "(127,ARRAY[6.2,2.8,4.8,1.8],'Iris-virginica'),\n",
+    "(128,ARRAY[6.1,3.0,4.9,1.8],'Iris-virginica'),\n",
+    "(129,ARRAY[6.4,2.8,5.6,2.1],'Iris-virginica'),\n",
+    "(130,ARRAY[7.2,3.0,5.8,1.6],'Iris-virginica'),\n",
+    "(131,ARRAY[7.4,2.8,6.1,1.9],'Iris-virginica'),\n",
+    "(132,ARRAY[7.9,3.8,6.4,2.0],'Iris-virginica'),\n",
+    "(133,ARRAY[6.4,2.8,5.6,2.2],'Iris-virginica'),\n",
+    "(134,ARRAY[6.3,2.8,5.1,1.5],'Iris-virginica'),\n",
+    "(135,ARRAY[6.1,2.6,5.6,1.4],'Iris-virginica'),\n",
+    "(136,ARRAY[7.7,3.0,6.1,2.3],'Iris-virginica'),\n",
+    "(137,ARRAY[6.3,3.4,5.6,2.4],'Iris-virginica'),\n",
+    "(138,ARRAY[6.4,3.1,5.5,1.8],'Iris-virginica'),\n",
+    "(139,ARRAY[6.0,3.0,4.8,1.8],'Iris-virginica'),\n",
+    "(140,ARRAY[6.9,3.1,5.4,2.1],'Iris-virginica'),\n",
+    "(141,ARRAY[6.7,3.1,5.6,2.4],'Iris-virginica'),\n",
+    "(142,ARRAY[6.9,3.1,5.1,2.3],'Iris-virginica'),\n",
+    "(143,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(144,ARRAY[6.8,3.2,5.9,2.3],'Iris-virginica'),\n",
+    "(145,ARRAY[6.7,3.3,5.7,2.5],'Iris-virginica'),\n",
+    "(146,ARRAY[6.7,3.0,5.2,2.3],'Iris-virginica'),\n",
+    "(147,ARRAY[6.3,2.5,5.0,1.9],'Iris-virginica'),\n",
+    "(148,ARRAY[6.5,3.0,5.2,2.0],'Iris-virginica'),\n",
+    "(149,ARRAY[6.2,3.4,5.4,2.3],'Iris-virginica'),\n",
+    "(150,ARRAY[5.9,3.0,5.1,1.8],'Iris-virginica');\n",
+    "\n",
+    "SELECT * FROM iris_data ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Create a test/validation dataset from the training data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(120L,)]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train, iris_test;\n",
+    "\n",
+    "-- Set seed so results are reproducible\n",
+    "SELECT setseed(0);\n",
+    "\n",
+    "SELECT madlib.train_test_split('iris_data',     -- Source table\n",
+    "                               'iris',          -- Output table root name\n",
+    "                                0.8,            -- Train proportion\n",
+    "                                NULL,           -- Test proportion (0.2)\n",
+    "                                NULL,           -- Strata definition\n",
+    "                                NULL,           -- Output all columns\n",
+    "                                NULL,           -- Sample without replacement\n",
+    "                                TRUE            -- Separate output tables\n",
+    "                              );\n",
+    "\n",
+    "SELECT COUNT(*) FROM iris_train;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pp\"></a>\n",
+    "# 2. Call preprocessor for deep learning\n",
+    "Training dataset (uses training preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>attributes_shape</th>\n",
+       "        <th>class_text_shape</th>\n",
+       "        <th>buffer_id</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[60, 4]</td>\n",
+       "        <td>[60, 3]</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[60, 4]</td>\n",
+       "        <td>[60, 3]</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[([60, 4], [60, 3], 0), ([60, 4], [60, 3], 1)]"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train_packed, iris_train_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('iris_train',         -- Source table\n",
+    "                                       'iris_train_packed',  -- Output table\n",
+    "                                       'class_text',        -- Dependent variable\n",
+    "                                       'attributes'         -- Independent variable\n",
+    "                                        ); \n",
+    "\n",
+    "SELECT attributes_shape, class_text_shape, buffer_id FROM iris_train_packed ORDER BY buffer_id;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train</td>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>60</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train', u'iris_train_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 60, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 8,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_train_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Validation dataset (uses validation preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>attributes_shape</th>\n",
+       "        <th>class_text_shape</th>\n",
+       "        <th>buffer_id</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 4]</td>\n",
+       "        <td>[15, 3]</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 4]</td>\n",
+       "        <td>[15, 3]</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[([15, 4], [15, 3], 0), ([15, 4], [15, 3], 1)]"
+      ]
+     },
+     "execution_count": 9,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_test_packed, iris_test_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('iris_test',          -- Source table\n",
+    "                                         'iris_test_packed',   -- Output table\n",
+    "                                         'class_text',         -- Dependent variable\n",
+    "                                         'attributes',         -- Independent variable\n",
+    "                                         'iris_train_packed'   -- From training preprocessor step\n",
+    "                                          ); \n",
+    "\n",
+    "SELECT attributes_shape, class_text_shape, buffer_id FROM iris_test_packed ORDER BY buffer_id;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_test</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>15</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_test', u'iris_test_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 15, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_test_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load\"></a>\n",
+    "# 3. Define and load model architecture\n",
+    "Import Keras libraries"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 1 hidden layer:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 13,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_1\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_3 (Dense)              (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_4 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_5 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 193\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model1 = Sequential()\n",
+    "model1.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model1.add(Dense(10, activation='relu'))\n",
+    "model1.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model1.summary()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 14,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_1\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 14,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model1.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 2 hidden layers:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 15,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_2\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_6 (Dense)              (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_7 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_8 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_9 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 303\n",
+      "Trainable params: 303\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model2 = Sequential()\n",
+    "model2.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model2.summary()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 14,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 14,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model2.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 16,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>model_weights</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>__internal_madlib_id__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>MLP with 1 hidden layer</td>\n",
+       "        <td>__madlib_temp_99030268_1614985897_73934030__</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>MLP with 2 hidden layers</td>\n",
+       "        <td>__madlib_temp_69765081_1614985897_31307059__</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'MLP with 1 hidden layer', u'__madlib_temp_99030268_1614985897_73934030__'),\n",
+       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1835 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'MLP with 2 hidden layers', u'__madlib_temp_69765081_1614985897_31307059__')]"
+      ]
+     },
+     "execution_count": 16,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS model_arch_library;\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Sophie',              -- Name\n",
+    "                               'MLP with 1 hidden layer'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Maria',               -- Name\n",
+    "                               'MLP with 2 hidden layers'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT * FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"def_mst\"></a>\n",
+    "# 4.  Define and load model selection tuples"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Generate model configurations using grid search. The output table for grid search contains the unique combinations of model architectures, compile and fit parameters."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 17,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8')]"
+      ]
+     },
+     "execution_count": 17,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'],\n",
+    "                                             'optimizer_params_list': [ {'optimizer': ['Adam'], 'lr': [0.001, 0.01, 0.1]} ],\n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid\n",
+    "                                         $$\n",
+    "                                         { 'batch_size': [4, 8],\n",
+    "                                           'epochs': [1]\n",
+    "                                         }\n",
+    "                                         $$,                  -- fit_param_grid\n",
+    "                                         'grid'               -- search_type\n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "This is the name of the model architecture table that corresponds to the model selection table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 18,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>object_table</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'model_arch_library', None)]"
+      ]
+     },
+     "execution_count": 18,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM mst_table_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train\"></a>\n",
+    "# 5.  Train\n",
+    "Train multiple models:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 19,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit_multiple_model</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 19,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_multi_model, iris_multi_model_summary, iris_multi_model_info;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
+    "                                              'iris_multi_model',     -- model_output_table\n",
+    "                                              'mst_table',            -- model_selection_table\n",
+    "                                              10,                     -- num_iterations\n",
+    "                                              FALSE                   -- use gpus\n",
+    "                                             );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 20,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>warm_start</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>iris_multi_model</td>\n",
+       "        <td>iris_multi_model_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>mst_table</td>\n",
+       "        <td>None</td>\n",
+       "        <td>10</td>\n",
+       "        <td>10</td>\n",
+       "        <td>False</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2021-03-05 23:17:53.015997</td>\n",
+       "        <td>2021-03-05 23:19:25.828328</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[10]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', None, u'iris_multi_model', u'iris_multi_model_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'mst_table', None, 10, 10, False, None, None, datetime.datetime(2021, 3, 5, 23, 17, 53, 15997), datetime.datetime(2021, 3, 5, 23, 19, 25, 828328), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0, [10])]"
+      ]
+     },
+     "execution_count": 20,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View results for each model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 22,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[90.4152021408081]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.983333349228</td>\n",
+       "        <td>0.0784567892551</td>\n",
+       "        <td>[0.983333349227905]</td>\n",
+       "        <td>[0.0784567892551422]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[89.9267370700836]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.370112359524</td>\n",
+       "        <td>[0.949999988079071]</td>\n",
+       "        <td>[0.370112359523773]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[90.6731050014496]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.120034113526</td>\n",
+       "        <td>[0.933333337306976]</td>\n",
+       "        <td>[0.120034113526344]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[91.596951007843]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.908333361149</td>\n",
+       "        <td>0.221316426992</td>\n",
+       "        <td>[0.908333361148834]</td>\n",
+       "        <td>[0.221316426992416]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[92.810093164444]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.256317943335</td>\n",
+       "        <td>[0.866666674613953]</td>\n",
+       "        <td>[0.256317943334579]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[92.3147950172424]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.774999976158</td>\n",
+       "        <td>0.937343239784</td>\n",
+       "        <td>[0.774999976158142]</td>\n",
+       "        <td>[0.937343239784241]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[90.1556451320648]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.725000023842</td>\n",
+       "        <td>0.670977592468</td>\n",
+       "        <td>[0.725000023841858]</td>\n",
+       "        <td>[0.670977592468262]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[91.1536350250244]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.691666662693</td>\n",
+       "        <td>0.679427325726</td>\n",
+       "        <td>[0.691666662693024]</td>\n",
+       "        <td>[0.679427325725555]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[92.044242143631]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.691666662693</td>\n",
+       "        <td>0.818258941174</td>\n",
+       "        <td>[0.691666662693024]</td>\n",
+       "        <td>[0.818258941173553]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[90.9328460693359]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.641666650772</td>\n",
+       "        <td>0.462306410074</td>\n",
+       "        <td>[0.641666650772095]</td>\n",
+       "        <td>[0.462306410074234]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[92.5436642169952]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.441666662693</td>\n",
+       "        <td>1.47737956047</td>\n",
+       "        <td>[0.441666662693024]</td>\n",
+       "        <td>[1.47737956047058]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[91.3729450702667]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.358333319426</td>\n",
+       "        <td>1.09699964523</td>\n",
+       "        <td>[0.358333319425583]</td>\n",
+       "        <td>[1.09699964523315]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [90.4152021408081], [u'accuracy'], u'categorical_crossentropy', 0.983333349227905, 0.0784567892551422, [0.983333349227905], [0.0784567892551422], None, None, None, None),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [89.9267370700836], [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.370112359523773, [0.949999988079071], [0.370112359523773], None, None, None, None),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [90.6731050014496], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.120034113526344, [0.933333337306976], [0.120034113526344], None, None, None, None),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [91.596951007843], [u'accuracy'], u'categorical_crossentropy', 0.908333361148834, 0.221316426992416, [0.908333361148834], [0.221316426992416], None, None, None, None),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [92.810093164444], [u'accuracy'], u'categorical_crossentropy', 0.866666674613953, 0.256317943334579, [0.866666674613953], [0.256317943334579], None, None, None, None),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [92.3147950172424], [u'accuracy'], u'categorical_crossentropy', 0.774999976158142, 0.937343239784241, [0.774999976158142], [0.937343239784241], None, None, None, None),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [90.1556451320648], [u'accuracy'], u'categorical_crossentropy', 0.725000023841858, 0.670977592468262, [0.725000023841858], [0.670977592468262], None, None, None, None),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [91.1536350250244], [u'accuracy'], u'categorical_crossentropy', 0.691666662693024, 0.679427325725555, [0.691666662693024], [0.679427325725555], None, None, None, None),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [92.044242143631], [u'accuracy'], u'categorical_crossentropy', 0.691666662693024, 0.818258941173553, [0.691666662693024], [0.818258941173553], None, None, None, None),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [90.9328460693359], [u'accuracy'], u'categorical_crossentropy', 0.641666650772095, 0.462306410074234, [0.641666650772095], [0.462306410074234], None, None, None, None),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [92.5436642169952], [u'accuracy'], u'categorical_crossentropy', 0.441666662693024, 1.47737956047058, [0.441666662693024], [1.47737956047058], None, None, None, None),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [91.3729450702667], [u'accuracy'], u'categorical_crossentropy', 0.358333319425583, 1.09699964523315, [0.358333319425583], [1.09699964523315], None, None, None, None)]"
+      ]
+     },
+     "execution_count": 22,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_info ORDER BY training_metrics_final DESC, training_loss_final;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"eval\"></a>\n",
+    "# 6. Evaluate\n",
+    "\n",
+    "Now run evaluate using model we built above:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 24,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>loss</th>\n",
+       "        <th>metric</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>0.143714919686</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(0.143714919686317, 0.966666638851166, [u'accuracy'], u'categorical_crossentropy')]"
+      ]
+     },
+     "execution_count": 24,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_validate;\n",
+    "SELECT madlib.madlib_keras_evaluate('iris_multi_model',  -- model\n",
+    "                                    'iris_test_packed',  -- test table\n",
+    "                                    'iris_validate',     -- output table\n",
+    "                                     NULL,               -- use gpus\n",
+    "                                     9                   -- mst_key to use\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_validate;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred\"></a>\n",
+    "# 7. Predict\n",
+    "\n",
+    "Now predict using model we built.  We will use the validation data set for prediction as well, which is not usual but serves to show the syntax. The prediction is in the estimated_class_text column:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 25,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "30 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99811894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9342552</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.81576407</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8137554</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9515729</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.68559366</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9376807</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9403642</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.70344603</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9156011</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.92681235</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.59969354</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.90776694</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8773563</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.92992663</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.97188693</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9858701</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.749179</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5419714</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.907297</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9711316</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (5, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (7, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (13, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (23, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (24, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (27, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (30, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (32, u'class_text', u'Iris-setosa', 0.99811894),\n",
+       " (52, u'class_text', u'Iris-versicolor', 0.9342552),\n",
+       " (57, u'class_text', u'Iris-versicolor', 0.81576407),\n",
+       " (64, u'class_text', u'Iris-versicolor', 0.8137554),\n",
+       " (66, u'class_text', u'Iris-versicolor', 0.9515729),\n",
+       " (67, u'class_text', u'Iris-versicolor', 0.68559366),\n",
+       " (68, u'class_text', u'Iris-versicolor', 0.9376807),\n",
+       " (70, u'class_text', u'Iris-versicolor', 0.9403642),\n",
+       " (71, u'class_text', u'Iris-virginica', 0.70344603),\n",
+       " (74, u'class_text', u'Iris-versicolor', 0.9156011),\n",
+       " (80, u'class_text', u'Iris-versicolor', 0.92681235),\n",
+       " (85, u'class_text', u'Iris-versicolor', 0.59969354),\n",
+       " (90, u'class_text', u'Iris-versicolor', 0.90776694),\n",
+       " (92, u'class_text', u'Iris-versicolor', 0.8773563),\n",
+       " (107, u'class_text', u'Iris-virginica', 0.92992663),\n",
+       " (108, u'class_text', u'Iris-virginica', 0.97188693),\n",
+       " (116, u'class_text', u'Iris-virginica', 0.9858701),\n",
+       " (127, u'class_text', u'Iris-virginica', 0.749179),\n",
+       " (134, u'class_text', u'Iris-virginica', 0.5419714),\n",
+       " (138, u'class_text', u'Iris-virginica', 0.907297),\n",
+       " (143, u'class_text', u'Iris-virginica', 0.9711316)]"
+      ]
+     },
+     "execution_count": 25,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_multi_model', -- model\n",
+    "                                   'iris_test',        -- test_table\n",
+    "                                   'id',               -- id column\n",
+    "                                   'attributes',       -- independent var\n",
+    "                                   'iris_predict',     -- output table\n",
+    "                                    'response',        -- prediction type\n",
+    "                                    FALSE,             -- use gpus\n",
+    "                                    9                  -- mst_key to use\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 29,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1L,)]"
+      ]
+     },
+     "execution_count": 29,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict JOIN iris_test USING (id) \n",
+    "WHERE iris_predict.class_value != iris_test.class_text;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 30,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96.67</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('96.67'),)]"
+      ]
+     },
+     "execution_count": 30,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict.class_value as estimated\n",
+    "     from iris_predict inner join iris_test\n",
+    "     on iris_test.id=iris_predict.id) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class2\"></a>\n",
+    "# Classification with Other Parameters\n",
+    "\n",
+    "<a id=\"val_dataset\"></a>\n",
+    "# 1.  Validation dataset\n",
+    "\n",
+    "Now use a validation dataset and compute metrics every 2nd iteration using the 'metrics_compute_frequency' parameter.  This can help reduce run time if you do not need metrics computed at every iteration."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 31,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit_multiple_model</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 31,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_multi_model, iris_multi_model_summary, iris_multi_model_info;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
+    "                                              'iris_multi_model',     -- model_output_table\n",
+    "                                              'mst_table',            -- model_selection_table\n",
+    "                                               10,                     -- num_iterations\n",
+    "                                               FALSE,                 -- use gpus\n",
+    "                                              'iris_test_packed',     -- validation dataset\n",
+    "                                               3,                     -- metrics compute frequency\n",
+    "                                               FALSE,                 -- warm start\n",
+    "                                              'Sophie L.',            -- name\n",
+    "                                              'Model selection for iris dataset'  -- description\n",
+    "                                             );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 32,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>warm_start</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>iris_multi_model</td>\n",
+       "        <td>iris_multi_model_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>mst_table</td>\n",
+       "        <td>None</td>\n",
+       "        <td>10</td>\n",
+       "        <td>3</td>\n",
+       "        <td>False</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Model selection for iris dataset</td>\n",
+       "        <td>2021-03-05 23:30:29.361525</td>\n",
+       "        <td>2021-03-05 23:32:25.242549</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3, 6, 9, 10]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_test_packed', u'iris_multi_model', u'iris_multi_model_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'mst_table', None, 10, 3, False, u'Sophie L.', u'Model selection for iris dataset', datetime.datetime(2021, 3, 5, 23, 30, 29, 361525), datetime.datetime(2021, 3, 5, 23, 32, 25, 242549), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0, [3, 6, 9, 10])]"
+      ]
+     },
+     "execution_count": 32,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View performance of each model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 33,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[30.6736240386963, 63.8817899227142, 97.0417211055756, 113.172317028046]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.975000023842</td>\n",
+       "        <td>0.0857621207833</td>\n",
+       "        <td>[0.866666674613953, 0.908333361148834, 0.824999988079071, 0.975000023841858]</td>\n",
+       "        <td>[0.283501744270325, 0.201569080352783, 0.365902632474899, 0.085762120783329]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.122390650213</td>\n",
+       "        <td>[0.733333349227905, 0.933333337306976, 0.866666674613953, 0.933333337306976]</td>\n",
+       "        <td>[0.402765065431595, 0.179033249616623, 0.302312880754471, 0.122390650212765]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[33.3402421474457, 66.678878068924, 100.063696146011, 115.878378152847]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.122639112175</td>\n",
+       "        <td>[0.841666638851166, 0.866666674613953, 0.983333349227905, 0.966666638851166]</td>\n",
+       "        <td>[0.927325308322906, 0.235888451337814, 0.152433648705482, 0.122639112174511]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.213843882084</td>\n",
+       "        <td>[0.800000011920929, 0.766666650772095, 0.966666638851166, 0.933333337306976]</td>\n",
+       "        <td>[1.31817901134491, 0.403295516967773, 0.225061357021332, 0.213843882083893]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[32.3545279502869, 65.5994129180908, 98.8639390468597, 114.628155946732]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.340993762016</td>\n",
+       "        <td>[0.725000023841858, 0.975000023841858, 0.966666638851166, 0.949999988079071]</td>\n",
+       "        <td>[0.634531021118164, 0.48712894320488, 0.342138230800629, 0.340993762016296]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.375507682562</td>\n",
+       "        <td>[0.633333325386047, 0.933333337306976, 0.966666638851166, 0.933333337306976]</td>\n",
+       "        <td>[0.697710037231445, 0.519804179668427, 0.398834854364395, 0.375507682561874]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[31.1844570636749, 64.4128880500793, 97.6602430343628, 113.686207056046]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.0938184261322</td>\n",
+       "        <td>[0.975000023841858, 0.949999988079071, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.107082977890968, 0.102665595710278, 0.0681213364005089, 0.0938184261322021]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.166130110621</td>\n",
+       "        <td>[0.966666638851166, 0.933333337306976, 0.933333337306976, 0.933333337306976]</td>\n",
+       "        <td>[0.141592919826508, 0.121055454015732, 0.0925953686237335, 0.166130110621452]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[32.1151111125946, 65.3588180541992, 98.6281480789185, 114.400741100311]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.875</td>\n",
+       "        <td>0.260402351618</td>\n",
+       "        <td>[0.975000023841858, 0.649999976158142, 0.774999976158142, 0.875]</td>\n",
+       "        <td>[0.350311905145645, 0.55135190486908, 0.319230705499649, 0.260402351617813]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.267347544432</td>\n",
+       "        <td>[0.966666638851166, 0.466666668653488, 0.666666686534882, 0.866666674613953]</td>\n",
+       "        <td>[0.401203900575638, 0.844793558120728, 0.492979735136032, 0.267347544431686]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[30.9298729896545, 64.1563010215759, 97.3999631404877, 113.427273988724]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.908333361149</td>\n",
+       "        <td>0.262101709843</td>\n",
+       "        <td>[0.725000023841858, 0.958333313465118, 0.916666686534882, 0.908333361148834]</td>\n",
+       "        <td>[0.42991915345192, 0.225951835513115, 0.209349796175957, 0.262101709842682]</td>\n",
+       "        <td>0.800000011921</td>\n",
+       "        <td>0.303691267967</td>\n",
+       "        <td>[0.633333325386047, 1.0, 0.966666638851166, 0.800000011920929]</td>\n",
+       "        <td>[0.514671266078949, 0.251676499843597, 0.234429702162743, 0.303691267967224]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[30.4536979198456, 63.6586000919342, 96.8189079761505, 112.950505018234]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.441878378391</td>\n",
+       "        <td>[0.474999994039536, 0.691666662693024, 0.741666674613953, 0.866666674613953]</td>\n",
+       "        <td>[0.957030355930328, 0.640025198459625, 0.472760319709778, 0.441878378391266]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.470352619886</td>\n",
+       "        <td>[0.433333337306976, 0.566666662693024, 0.600000023841858, 0.766666650772095]</td>\n",
+       "        <td>[0.983189880847931, 0.697314381599426, 0.513611257076263, 0.470352619886398]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[32.8489799499512, 66.1956899166107, 99.4372820854187, 115.379984140396]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.850000023842</td>\n",
+       "        <td>0.454301446676</td>\n",
+       "        <td>[0.691666662693024, 0.691666662693024, 0.725000023841858, 0.850000023841858]</td>\n",
+       "        <td>[0.845666646957397, 0.635031342506409, 0.49227574467659, 0.454301446676254]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.534793019295</td>\n",
+       "        <td>[0.566666662693024, 0.566666662693024, 0.633333325386047, 0.766666650772095]</td>\n",
+       "        <td>[0.97626668214798, 0.749890267848969, 0.588431000709534, 0.534793019294739]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[33.0761120319366, 66.4210600852966, 99.6609480381012, 115.608960151672]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.850000023842</td>\n",
+       "        <td>0.456756323576</td>\n",
+       "        <td>[0.966666638851166, 0.958333313465118, 0.566666662693024, 0.850000023841858]</td>\n",
+       "        <td>[0.138208195567131, 0.242085874080658, 2.31771349906921, 0.456756323575974]</td>\n",
+       "        <td>0.733333349228</td>\n",
+       "        <td>0.720676660538</td>\n",
+       "        <td>[0.966666638851166, 0.966666638851166, 0.666666686534882, 0.733333349227905]</td>\n",
+       "        <td>[0.188635662198067, 0.311398893594742, 2.20467662811279, 0.72067666053772]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[31.4386551380157, 64.6823661327362, 97.9228360652924, 113.945574998856]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.691666662693</td>\n",
+       "        <td>0.461679339409</td>\n",
+       "        <td>[0.858333349227905, 0.691666662693024, 0.641666650772095, 0.691666662693024]</td>\n",
+       "        <td>[0.436321765184402, 0.492918580770493, 0.463972359895706, 0.461679339408875]</td>\n",
+       "        <td>0.566666662693</td>\n",
+       "        <td>0.466376572847</td>\n",
+       "        <td>[0.800000011920929, 0.566666662693024, 0.766666650772095, 0.566666662693024]</td>\n",
+       "        <td>[0.436118185520172, 0.50031191110611, 0.46391037106514, 0.466376572847366]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[31.660609960556, 64.9007411003113, 98.1513090133667, 114.166445016861]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.699999988079</td>\n",
+       "        <td>0.612414062023</td>\n",
+       "        <td>[0.358333319425583, 0.691666662693024, 0.691666662693024, 0.699999988079071]</td>\n",
+       "        <td>[1.08384656906128, 0.789247930049896, 0.642031610012054, 0.612414062023163]</td>\n",
+       "        <td>0.566666662693</td>\n",
+       "        <td>0.721881330013</td>\n",
+       "        <td>[0.233333334326744, 0.566666662693024, 0.566666662693024, 0.566666662693024]</td>\n",
+       "        <td>[1.24377524852753, 0.895631670951843, 0.770445883274078, 0.721881330013275]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[32.5858130455017, 65.8329451084137, 99.097393989563, 115.102792024612]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.641666650772</td>\n",
+       "        <td>0.865168988705</td>\n",
+       "        <td>[0.358333319425583, 0.541666686534882, 0.641666650772095, 0.641666650772095]</td>\n",
+       "        <td>[0.977362334728241, 0.920270144939423, 0.881129205226898, 0.865168988704681]</td>\n",
+       "        <td>0.533333361149</td>\n",
+       "        <td>0.939725458622</td>\n",
+       "        <td>[0.233333334326744, 0.466666668653488, 0.533333361148834, 0.533333361148834]</td>\n",
+       "        <td>[1.08130764961243, 0.966315567493439, 0.950002133846283, 0.939725458621979]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [30.6736240386963, 63.8817899227142, 97.0417211055756, 113.172317028046], [u'accuracy'], u'categorical_crossentropy', 0.975000023841858, 0.085762120783329, [0.866666674613953, 0.908333361148834, 0.824999988079071, 0.975000023841858], [0.283501744270325, 0.201569080352783, 0.365902632474899, 0.085762120783329], 0.933333337306976, 0.122390650212765, [0.733333349227905, 0.933333337306976, 0.866666674613953, 0.933333337306976], [0.402765065431595, 0.179033249616623, 0.302312880754471, 0.122390650212765]),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [33.3402421474457, 66.678878068924, 100.063696146011, 115.878378152847], [u'accuracy'], u'categorical_crossentropy', 0.966666638851166, 0.122639112174511, [0.841666638851166, 0.866666674613953, 0.983333349227905, 0.966666638851166], [0.927325308322906, 0.235888451337814, 0.152433648705482, 0.122639112174511], 0.933333337306976, 0.213843882083893, [0.800000011920929, 0.766666650772095, 0.966666638851166, 0.933333337306976], [1.31817901134491, 0.403295516967773, 0.225061357021332, 0.213843882083893]),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [32.3545279502869, 65.5994129180908, 98.8639390468597, 114.628155946732], [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.340993762016296, [0.725000023841858, 0.975000023841858, 0.966666638851166, 0.949999988079071], [0.634531021118164, 0.48712894320488, 0.342138230800629, 0.340993762016296], 0.933333337306976, 0.375507682561874, [0.633333325386047, 0.933333337306976, 0.966666638851166, 0.933333337306976], [0.697710037231445, 0.519804179668427, 0.398834854364395, 0.375507682561874]),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [31.1844570636749, 64.4128880500793, 97.6602430343628, 113.686207056046], [u'accuracy'], u'categorical_crossentropy', 0.966666638851166, 0.0938184261322021, [0.975000023841858, 0.949999988079071, 0.966666638851166, 0.966666638851166], [0.107082977890968, 0.102665595710278, 0.0681213364005089, 0.0938184261322021], 0.933333337306976, 0.166130110621452, [0.966666638851166, 0.933333337306976, 0.933333337306976, 0.933333337306976], [0.141592919826508, 0.121055454015732, 0.0925953686237335, 0.166130110621452]),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [32.1151111125946, 65.3588180541992, 98.6281480789185, 114.400741100311], [u'accuracy'], u'categorical_crossentropy', 0.875, 0.260402351617813, [0.975000023841858, 0.649999976158142, 0.774999976158142, 0.875], [0.350311905145645, 0.55135190486908, 0.319230705499649, 0.260402351617813], 0.866666674613953, 0.267347544431686, [0.966666638851166, 0.466666668653488, 0.666666686534882, 0.866666674613953], [0.401203900575638, 0.844793558120728, 0.492979735136032, 0.267347544431686]),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [30.9298729896545, 64.1563010215759, 97.3999631404877, 113.427273988724], [u'accuracy'], u'categorical_crossentropy', 0.908333361148834, 0.262101709842682, [0.725000023841858, 0.958333313465118, 0.916666686534882, 0.908333361148834], [0.42991915345192, 0.225951835513115, 0.209349796175957, 0.262101709842682], 0.800000011920929, 0.303691267967224, [0.633333325386047, 1.0, 0.966666638851166, 0.800000011920929], [0.514671266078949, 0.251676499843597, 0.234429702162743, 0.303691267967224]),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [30.4536979198456, 63.6586000919342, 96.8189079761505, 112.950505018234], [u'accuracy'], u'categorical_crossentropy', 0.866666674613953, 0.441878378391266, [0.474999994039536, 0.691666662693024, 0.741666674613953, 0.866666674613953], [0.957030355930328, 0.640025198459625, 0.472760319709778, 0.441878378391266], 0.766666650772095, 0.470352619886398, [0.433333337306976, 0.566666662693024, 0.600000023841858, 0.766666650772095], [0.983189880847931, 0.697314381599426, 0.513611257076263, 0.470352619886398]),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [32.8489799499512, 66.1956899166107, 99.4372820854187, 115.379984140396], [u'accuracy'], u'categorical_crossentropy', 0.850000023841858, 0.454301446676254, [0.691666662693024, 0.691666662693024, 0.725000023841858, 0.850000023841858], [0.845666646957397, 0.635031342506409, 0.49227574467659, 0.454301446676254], 0.766666650772095, 0.534793019294739, [0.566666662693024, 0.566666662693024, 0.633333325386047, 0.766666650772095], [0.97626668214798, 0.749890267848969, 0.588431000709534, 0.534793019294739]),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [33.0761120319366, 66.4210600852966, 99.6609480381012, 115.608960151672], [u'accuracy'], u'categorical_crossentropy', 0.850000023841858, 0.456756323575974, [0.966666638851166, 0.958333313465118, 0.566666662693024, 0.850000023841858], [0.138208195567131, 0.242085874080658, 2.31771349906921, 0.456756323575974], 0.733333349227905, 0.72067666053772, [0.966666638851166, 0.966666638851166, 0.666666686534882, 0.733333349227905], [0.188635662198067, 0.311398893594742, 2.20467662811279, 0.72067666053772]),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [31.4386551380157, 64.6823661327362, 97.9228360652924, 113.945574998856], [u'accuracy'], u'categorical_crossentropy', 0.691666662693024, 0.461679339408875, [0.858333349227905, 0.691666662693024, 0.641666650772095, 0.691666662693024], [0.436321765184402, 0.492918580770493, 0.463972359895706, 0.461679339408875], 0.566666662693024, 0.466376572847366, [0.800000011920929, 0.566666662693024, 0.766666650772095, 0.566666662693024], [0.436118185520172, 0.50031191110611, 0.46391037106514, 0.466376572847366]),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [31.660609960556, 64.9007411003113, 98.1513090133667, 114.166445016861], [u'accuracy'], u'categorical_crossentropy', 0.699999988079071, 0.612414062023163, [0.358333319425583, 0.691666662693024, 0.691666662693024, 0.699999988079071], [1.08384656906128, 0.789247930049896, 0.642031610012054, 0.612414062023163], 0.566666662693024, 0.721881330013275, [0.233333334326744, 0.566666662693024, 0.566666662693024, 0.566666662693024], [1.24377524852753, 0.895631670951843, 0.770445883274078, 0.721881330013275]),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [32.5858130455017, 65.8329451084137, 99.097393989563, 115.102792024612], [u'accuracy'], u'categorical_crossentropy', 0.641666650772095, 0.865168988704681, [0.358333319425583, 0.541666686534882, 0.641666650772095, 0.641666650772095], [0.977362334728241, 0.920270144939423, 0.881129205226898, 0.865168988704681], 0.533333361148834, 0.939725458621979, [0.233333334326744, 0.466666668653488, 0.533333361148834, 0.533333361148834], [1.08130764961243, 0.966315567493439, 0.950002133846283, 0.939725458621979])]"
+      ]
+     },
+     "execution_count": 33,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_info ORDER BY validation_metrics_final DESC;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Plot validation results"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 34,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%matplotlib notebook\n",
+    "import matplotlib.pyplot as plt\n",
+    "from matplotlib.ticker import MaxNLocator\n",
+    "from collections import defaultdict\n",
+    "import pandas as pd\n",
+    "import seaborn as sns\n",
+    "sns.set_palette(sns.color_palette(\"hls\", 20))\n",
+    "plt.rcParams.update({'font.size': 12})\n",
+    "pd.set_option('display.max_colwidth', -1)"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 36,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "7 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM iris_multi_model_info ORDER BY validation_loss ASC LIMIT 7;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "df_summary = %sql SELECT * FROM iris_multi_model_summary;\n",
+    "df_summary = df_summary.DataFrame()\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Iteration')\n",
+    "ax_metric.set_ylabel('Metric')\n",
+    "ax_metric.set_title('Validation metric curve')\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Iteration')\n",
+    "ax_loss.set_ylabel('Loss')\n",
+    "ax_loss.set_title('Validation loss curve')\n",
+    "\n",
+    "iters = df_summary['metrics_iters'][0]\n",
+    "\n",
+    "for mst_key in df_results['mst_key']:\n",
+    "    df_output_info = %sql SELECT validation_metrics,validation_loss FROM iris_multi_model_info WHERE mst_key = $mst_key\n",
+    "    df_output_info = df_output_info.DataFrame()\n",
+    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
+    "    validation_loss = df_output_info['validation_loss'][0]\n",
+    "    \n",
+    "    ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
+    "    ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred_prob\"></a>\n",
+    "# 2.  Predict probabilities\n",
+    "\n",
+    "Predict with probabilities for each class:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 38,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "90 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9998863</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.00011367707</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.2402717e-12</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999517</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>4.8225505e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>9.1029716e-14</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9998826</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.00011736089</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.3640493e-12</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9998809</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.0001190398</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.4304882e-12</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999548</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>4.5139022e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>7.4557155e-14</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9998492</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.00015075288</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>2.8751205e-12</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9998938</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.000106220104</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>9.999105e-13</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9998385</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.00016145893</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>3.608496e-12</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9998399</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.00016005547</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>3.511208e-12</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999478</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>5.218504e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.1477291e-13</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9810807</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.018912822</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>6.402063e-06</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.92223966</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.07775978</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>5.209471e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.77119005</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.22880979</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.5663863e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.99599</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0039465656</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>6.3496285e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.64926726</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.35073274</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>9.930102e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9886833</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.011225804</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>9.090092e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9772394</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.022695182</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>6.550627e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.64295816</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.35704187</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.5136303e-09</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.84566176</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.15433785</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>4.1220466e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9888799</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.010267593</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.00085249124</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.50881517</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.49118474</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.561445e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.88514596</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.11484939</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>4.6489695e-06</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.89321506</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.1067843</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>6.6432494e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.98068523</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.019314792</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.12604604e-11</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9925465</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.00745355</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>6.7306818e-15</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.97831887</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.021681104</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>6.130188e-13</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.74438953</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.2556105</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.6590068e-09</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7506184</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.24938157</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0018011e-09</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.958408</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.041591965</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>2.9998048e-12</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.98765844</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.01234158</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>5.268986e-13</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2, u'class_text', u'Iris-setosa', 0.9998863, 1),\n",
+       " (2, u'class_text', u'Iris-versicolor', 0.00011367707, 2),\n",
+       " (2, u'class_text', u'Iris-virginica', 1.2402717e-12, 3),\n",
+       " (5, u'class_text', u'Iris-setosa', 0.9999517, 1),\n",
+       " (5, u'class_text', u'Iris-versicolor', 4.8225505e-05, 2),\n",
+       " (5, u'class_text', u'Iris-virginica', 9.1029716e-14, 3),\n",
+       " (7, u'class_text', u'Iris-setosa', 0.9998826, 1),\n",
+       " (7, u'class_text', u'Iris-versicolor', 0.00011736089, 2),\n",
+       " (7, u'class_text', u'Iris-virginica', 1.3640493e-12, 3),\n",
+       " (13, u'class_text', u'Iris-setosa', 0.9998809, 1),\n",
+       " (13, u'class_text', u'Iris-versicolor', 0.0001190398, 2),\n",
+       " (13, u'class_text', u'Iris-virginica', 1.4304882e-12, 3),\n",
+       " (23, u'class_text', u'Iris-setosa', 0.9999548, 1),\n",
+       " (23, u'class_text', u'Iris-versicolor', 4.5139022e-05, 2),\n",
+       " (23, u'class_text', u'Iris-virginica', 7.4557155e-14, 3),\n",
+       " (24, u'class_text', u'Iris-setosa', 0.9998492, 1),\n",
+       " (24, u'class_text', u'Iris-versicolor', 0.00015075288, 2),\n",
+       " (24, u'class_text', u'Iris-virginica', 2.8751205e-12, 3),\n",
+       " (27, u'class_text', u'Iris-setosa', 0.9998938, 1),\n",
+       " (27, u'class_text', u'Iris-versicolor', 0.000106220104, 2),\n",
+       " (27, u'class_text', u'Iris-virginica', 9.999105e-13, 3),\n",
+       " (30, u'class_text', u'Iris-setosa', 0.9998385, 1),\n",
+       " (30, u'class_text', u'Iris-versicolor', 0.00016145893, 2),\n",
+       " (30, u'class_text', u'Iris-virginica', 3.608496e-12, 3),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.9998399, 1),\n",
+       " (31, u'class_text', u'Iris-versicolor', 0.00016005547, 2),\n",
+       " (31, u'class_text', u'Iris-virginica', 3.511208e-12, 3),\n",
+       " (32, u'class_text', u'Iris-setosa', 0.9999478, 1),\n",
+       " (32, u'class_text', u'Iris-versicolor', 5.218504e-05, 2),\n",
+       " (32, u'class_text', u'Iris-virginica', 1.1477291e-13, 3),\n",
+       " (52, u'class_text', u'Iris-versicolor', 0.9810807, 1),\n",
+       " (52, u'class_text', u'Iris-virginica', 0.018912822, 2),\n",
+       " (52, u'class_text', u'Iris-setosa', 6.402063e-06, 3),\n",
+       " (57, u'class_text', u'Iris-versicolor', 0.92223966, 1),\n",
+       " (57, u'class_text', u'Iris-virginica', 0.07775978, 2),\n",
+       " (57, u'class_text', u'Iris-setosa', 5.209471e-07, 3),\n",
+       " (64, u'class_text', u'Iris-versicolor', 0.77119005, 1),\n",
+       " (64, u'class_text', u'Iris-virginica', 0.22880979, 2),\n",
+       " (64, u'class_text', u'Iris-setosa', 1.5663863e-07, 3),\n",
+       " (66, u'class_text', u'Iris-versicolor', 0.99599, 1),\n",
+       " (66, u'class_text', u'Iris-virginica', 0.0039465656, 2),\n",
+       " (66, u'class_text', u'Iris-setosa', 6.3496285e-05, 3),\n",
+       " (67, u'class_text', u'Iris-versicolor', 0.64926726, 1),\n",
+       " (67, u'class_text', u'Iris-virginica', 0.35073274, 2),\n",
+       " (67, u'class_text', u'Iris-setosa', 9.930102e-08, 3),\n",
+       " (68, u'class_text', u'Iris-versicolor', 0.9886833, 1),\n",
+       " (68, u'class_text', u'Iris-virginica', 0.011225804, 2),\n",
+       " (68, u'class_text', u'Iris-setosa', 9.090092e-05, 3),\n",
+       " (70, u'class_text', u'Iris-versicolor', 0.9772394, 1),\n",
+       " (70, u'class_text', u'Iris-virginica', 0.022695182, 2),\n",
+       " (70, u'class_text', u'Iris-setosa', 6.550627e-05, 3),\n",
+       " (71, u'class_text', u'Iris-virginica', 0.64295816, 1),\n",
+       " (71, u'class_text', u'Iris-versicolor', 0.35704187, 2),\n",
+       " (71, u'class_text', u'Iris-setosa', 3.5136303e-09, 3),\n",
+       " (74, u'class_text', u'Iris-versicolor', 0.84566176, 1),\n",
+       " (74, u'class_text', u'Iris-virginica', 0.15433785, 2),\n",
+       " (74, u'class_text', u'Iris-setosa', 4.1220466e-07, 3),\n",
+       " (80, u'class_text', u'Iris-versicolor', 0.9888799, 1),\n",
+       " (80, u'class_text', u'Iris-setosa', 0.010267593, 2),\n",
+       " (80, u'class_text', u'Iris-virginica', 0.00085249124, 3),\n",
+       " (85, u'class_text', u'Iris-virginica', 0.50881517, 1),\n",
+       " (85, u'class_text', u'Iris-versicolor', 0.49118474, 2),\n",
+       " (85, u'class_text', u'Iris-setosa', 3.561445e-08, 3),\n",
+       " (90, u'class_text', u'Iris-versicolor', 0.88514596, 1),\n",
+       " (90, u'class_text', u'Iris-virginica', 0.11484939, 2),\n",
+       " (90, u'class_text', u'Iris-setosa', 4.6489695e-06, 3),\n",
+       " (92, u'class_text', u'Iris-versicolor', 0.89321506, 1),\n",
+       " (92, u'class_text', u'Iris-virginica', 0.1067843, 2),\n",
+       " (92, u'class_text', u'Iris-setosa', 6.6432494e-07, 3),\n",
+       " (107, u'class_text', u'Iris-virginica', 0.98068523, 1),\n",
+       " (107, u'class_text', u'Iris-versicolor', 0.019314792, 2),\n",
+       " (107, u'class_text', u'Iris-setosa', 1.12604604e-11, 3),\n",
+       " (108, u'class_text', u'Iris-virginica', 0.9925465, 1),\n",
+       " (108, u'class_text', u'Iris-versicolor', 0.00745355, 2),\n",
+       " (108, u'class_text', u'Iris-setosa', 6.7306818e-15, 3),\n",
+       " (116, u'class_text', u'Iris-virginica', 0.97831887, 1),\n",
+       " (116, u'class_text', u'Iris-versicolor', 0.021681104, 2),\n",
+       " (116, u'class_text', u'Iris-setosa', 6.130188e-13, 3),\n",
+       " (127, u'class_text', u'Iris-virginica', 0.74438953, 1),\n",
+       " (127, u'class_text', u'Iris-versicolor', 0.2556105, 2),\n",
+       " (127, u'class_text', u'Iris-setosa', 1.6590068e-09, 3),\n",
+       " (134, u'class_text', u'Iris-virginica', 0.7506184, 1),\n",
+       " (134, u'class_text', u'Iris-versicolor', 0.24938157, 2),\n",
+       " (134, u'class_text', u'Iris-setosa', 1.0018011e-09, 3),\n",
+       " (138, u'class_text', u'Iris-virginica', 0.958408, 1),\n",
+       " (138, u'class_text', u'Iris-versicolor', 0.041591965, 2),\n",
+       " (138, u'class_text', u'Iris-setosa', 2.9998048e-12, 3),\n",
+       " (143, u'class_text', u'Iris-virginica', 0.98765844, 1),\n",
+       " (143, u'class_text', u'Iris-versicolor', 0.01234158, 2),\n",
+       " (143, u'class_text', u'Iris-setosa', 5.268986e-13, 3)]"
+      ]
+     },
+     "execution_count": 38,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_multi_model', -- model\n",
+    "                                   'iris_test',        -- test_table\n",
+    "                                   'id',               -- id column\n",
+    "                                   'attributes',       -- independent var\n",
+    "                                   'iris_predict',     -- output table\n",
+    "                                    'prob',            -- prediction type\n",
+    "                                    FALSE,             -- use gpus\n",
+    "                                    3                  -- mst_key to use\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id, rank;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"warm_start\"></a>\n",
+    "# 3.  Warm start\n",
+    "\n",
+    "Next, use the warm_start parameter to continue learning, using the coefficients from the run above. Note that we don't drop the model table or model summary table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 39,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit_multiple_model</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 39,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
+    "                                              'iris_multi_model',     -- model_output_table\n",
+    "                                              'mst_table',            -- model_selection_table\n",
+    "                                               3,                     -- num_iterations\n",
+    "                                               FALSE,                 -- use gpus\n",
+    "                                              'iris_test_packed',     -- validation dataset\n",
+    "                                               1,                     -- metrics compute frequency\n",
+    "                                               TRUE,                  -- warm start\n",
+    "                                              'Sophie L.',            -- name\n",
+    "                                              'Simple MLP for iris dataset'  -- description\n",
+    "                                             );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 40,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>warm_start</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>iris_multi_model</td>\n",
+       "        <td>iris_multi_model_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>mst_table</td>\n",
+       "        <td>None</td>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>True</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Simple MLP for iris dataset</td>\n",
+       "        <td>2021-03-05 23:33:49.889172</td>\n",
+       "        <td>2021-03-05 23:34:35.696218</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[1, 2, 3]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_test_packed', u'iris_multi_model', u'iris_multi_model_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'mst_table', None, 3, 1, True, u'Sophie L.', u'Simple MLP for iris dataset', datetime.datetime(2021, 3, 5, 23, 33, 49, 889172), datetime.datetime(2021, 3, 5, 23, 34, 35, 696218), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0, [1, 2, 3])]"
+      ]
+     },
+     "execution_count": 40,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View performance of each model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 41,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[12.4149520397186, 27.5328221321106, 42.8307020664215]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.925000011921</td>\n",
+       "        <td>0.378388017416</td>\n",
+       "        <td>[0.866666674613953, 0.883333325386047, 0.925000011920929]</td>\n",
+       "        <td>[0.418134957551956, 0.397540986537933, 0.378388017416]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.401163935661</td>\n",
+       "        <td>[0.766666650772095, 0.800000011920929, 0.933333337306976]</td>\n",
+       "        <td>[0.450369209051132, 0.430310726165771, 0.401163935661316]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[13.1818931102753, 28.5486171245575, 43.8156039714813]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.975000023842</td>\n",
+       "        <td>0.0919722244143</td>\n",
+       "        <td>[0.966666638851166, 0.841666638851166, 0.975000023841858]</td>\n",
+       "        <td>[0.0829650238156319, 0.478152126073837, 0.0919722244143486]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.181837588549</td>\n",
+       "        <td>[0.966666638851166, 0.733333349227905, 0.933333337306976]</td>\n",
+       "        <td>[0.108266495168209, 0.985528528690338, 0.18183758854866]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[13.8922369480133, 29.253427028656, 44.6270771026611]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.163653150201</td>\n",
+       "        <td>[0.891666650772095, 0.491666674613953, 0.933333337306976]</td>\n",
+       "        <td>[0.223677828907967, 1.31514668464661, 0.163653150200844]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.214205592871</td>\n",
+       "        <td>[0.766666650772095, 0.600000023841858, 0.899999976158142]</td>\n",
+       "        <td>[0.346942394971848, 1.53849768638611, 0.214205592870712]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[15.0511469841003, 30.1774880886078, 45.5478649139404]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.159554898739</td>\n",
+       "        <td>[0.75, 0.941666662693024, 0.933333337306976]</td>\n",
+       "        <td>[0.376825720071793, 0.160363256931305, 0.159554898738861]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.249670743942</td>\n",
+       "        <td>[0.666666686534882, 0.899999976158142, 0.899999976158142]</td>\n",
+       "        <td>[0.572766482830048, 0.20397062599659, 0.249670743942261]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[14.5801939964294, 29.9574360847473, 45.324392080307]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.941666662693</td>\n",
+       "        <td>0.370102822781</td>\n",
+       "        <td>[0.891666650772095, 0.916666686534882, 0.941666662693024]</td>\n",
+       "        <td>[0.420222342014313, 0.39033767580986, 0.370102822780609]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.429161161184</td>\n",
+       "        <td>[0.800000011920929, 0.833333313465118, 0.866666674613953]</td>\n",
+       "        <td>[0.495854884386063, 0.458509385585785, 0.429161161184311]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[13.4523401260376, 28.810662984848, 44.1752660274506]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.641666650772</td>\n",
+       "        <td>0.462273716927</td>\n",
+       "        <td>[0.691666662693024, 0.691666662693024, 0.641666650772095]</td>\n",
+       "        <td>[0.461356490850449, 0.460641026496887, 0.462273716926575]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.461440712214</td>\n",
+       "        <td>[0.566666662693024, 0.566666662693024, 0.766666650772095]</td>\n",
+       "        <td>[0.494220763444901, 0.470757216215134, 0.461440712213516]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[15.3230481147766, 30.4359600543976, 45.8051240444183]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.641666650772</td>\n",
+       "        <td>0.465268939734</td>\n",
+       "        <td>[0.691666662693024, 0.691666662693024, 0.641666650772095]</td>\n",
+       "        <td>[0.468634635210037, 0.460501492023468, 0.465268939733505]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.453260302544</td>\n",
+       "        <td>[0.566666662693024, 0.566666662693024, 0.766666650772095]</td>\n",
+       "        <td>[0.527487695217133, 0.486553341150284, 0.45326030254364]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[12.9218399524689, 28.2852990627289, 43.3031449317932]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.908333361149</td>\n",
+       "        <td>0.282380491495</td>\n",
+       "        <td>[0.975000023841858, 0.916666686534882, 0.908333361148834]</td>\n",
+       "        <td>[0.133402094244957, 0.194111600518227, 0.282380491495132]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.321279972792</td>\n",
+       "        <td>[0.966666638851166, 0.966666638851166, 0.766666650772095]</td>\n",
+       "        <td>[0.14349028468132, 0.207735612988472, 0.321279972791672]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[12.6596629619598, 28.0149381160736, 43.0508260726929]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.841666638851</td>\n",
+       "        <td>0.374685227871</td>\n",
+       "        <td>[0.933333337306976, 0.899999976158142, 0.841666638851166]</td>\n",
+       "        <td>[0.162063658237457, 0.223208039999008, 0.374685227870941]</td>\n",
+       "        <td>0.733333349228</td>\n",
+       "        <td>0.698961436749</td>\n",
+       "        <td>[0.866666674613953, 0.733333349227905, 0.733333349227905]</td>\n",
+       "        <td>[0.297752887010574, 0.422860831022263, 0.698961436748505]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[14.1109259128571, 29.4764680862427, 44.847892999649]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.858333349228</td>\n",
+       "        <td>0.323420703411</td>\n",
+       "        <td>[0.975000023841858, 0.975000023841858, 0.858333349227905]</td>\n",
+       "        <td>[0.260769069194794, 0.237523972988129, 0.323420703411102]</td>\n",
+       "        <td>0.699999988079</td>\n",
+       "        <td>0.45600476861</td>\n",
+       "        <td>[0.933333337306976, 0.933333337306976, 0.699999988079071]</td>\n",
+       "        <td>[0.302719950675964, 0.286018937826157, 0.456004768610001]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[13.6740920543671, 29.0340840816498, 44.4022340774536]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.699999988079</td>\n",
+       "        <td>0.52513474226</td>\n",
+       "        <td>[0.699999988079071, 0.699999988079071, 0.699999988079071]</td>\n",
+       "        <td>[0.579447448253632, 0.552316129207611, 0.525134742259979]</td>\n",
+       "        <td>0.566666662693</td>\n",
+       "        <td>0.634039282799</td>\n",
+       "        <td>[0.566666662693024, 0.566666662693024, 0.566666662693024]</td>\n",
+       "        <td>[0.681210398674011, 0.654673635959625, 0.634039282798767]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[14.3272860050201, 29.6978969573975, 45.0662181377411]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.550000011921</td>\n",
+       "        <td>0.824133753777</td>\n",
+       "        <td>[0.600000023841858, 0.600000023841858, 0.550000011920929]</td>\n",
+       "        <td>[0.849512219429016, 0.836838126182556, 0.82413375377655]</td>\n",
+       "        <td>0.433333337307</td>\n",
+       "        <td>0.862674951553</td>\n",
+       "        <td>[0.533333361148834, 0.533333361148834, 0.433333337306976]</td>\n",
+       "        <td>[0.907541394233704, 0.897742569446564, 0.862674951553345]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [12.4149520397186, 27.5328221321106, 42.8307020664215], [u'accuracy'], u'categorical_crossentropy', 0.925000011920929, 0.378388017416, [0.866666674613953, 0.883333325386047, 0.925000011920929], [0.418134957551956, 0.397540986537933, 0.378388017416], 0.933333337306976, 0.401163935661316, [0.766666650772095, 0.800000011920929, 0.933333337306976], [0.450369209051132, 0.430310726165771, 0.401163935661316]),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [13.1818931102753, 28.5486171245575, 43.8156039714813], [u'accuracy'], u'categorical_crossentropy', 0.975000023841858, 0.0919722244143486, [0.966666638851166, 0.841666638851166, 0.975000023841858], [0.0829650238156319, 0.478152126073837, 0.0919722244143486], 0.933333337306976, 0.18183758854866, [0.966666638851166, 0.733333349227905, 0.933333337306976], [0.108266495168209, 0.985528528690338, 0.18183758854866]),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [13.8922369480133, 29.253427028656, 44.6270771026611], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.163653150200844, [0.891666650772095, 0.491666674613953, 0.933333337306976], [0.223677828907967, 1.31514668464661, 0.163653150200844], 0.899999976158142, 0.214205592870712, [0.766666650772095, 0.600000023841858, 0.899999976158142], [0.346942394971848, 1.53849768638611, 0.214205592870712]),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [15.0511469841003, 30.1774880886078, 45.5478649139404], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.159554898738861, [0.75, 0.941666662693024, 0.933333337306976], [0.376825720071793, 0.160363256931305, 0.159554898738861], 0.899999976158142, 0.249670743942261, [0.666666686534882, 0.899999976158142, 0.899999976158142], [0.572766482830048, 0.20397062599659, 0.249670743942261]),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [14.5801939964294, 29.9574360847473, 45.324392080307], [u'accuracy'], u'categorical_crossentropy', 0.941666662693024, 0.370102822780609, [0.891666650772095, 0.916666686534882, 0.941666662693024], [0.420222342014313, 0.39033767580986, 0.370102822780609], 0.866666674613953, 0.429161161184311, [0.800000011920929, 0.833333313465118, 0.866666674613953], [0.495854884386063, 0.458509385585785, 0.429161161184311]),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [13.4523401260376, 28.810662984848, 44.1752660274506], [u'accuracy'], u'categorical_crossentropy', 0.641666650772095, 0.462273716926575, [0.691666662693024, 0.691666662693024, 0.641666650772095], [0.461356490850449, 0.460641026496887, 0.462273716926575], 0.766666650772095, 0.461440712213516, [0.566666662693024, 0.566666662693024, 0.766666650772095], [0.494220763444901, 0.470757216215134, 0.461440712213516]),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [15.3230481147766, 30.4359600543976, 45.8051240444183], [u'accuracy'], u'categorical_crossentropy', 0.641666650772095, 0.465268939733505, [0.691666662693024, 0.691666662693024, 0.641666650772095], [0.468634635210037, 0.460501492023468, 0.465268939733505], 0.766666650772095, 0.45326030254364, [0.566666662693024, 0.566666662693024, 0.766666650772095], [0.527487695217133, 0.486553341150284, 0.45326030254364]),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [12.9218399524689, 28.2852990627289, 43.3031449317932], [u'accuracy'], u'categorical_crossentropy', 0.908333361148834, 0.282380491495132, [0.975000023841858, 0.916666686534882, 0.908333361148834], [0.133402094244957, 0.194111600518227, 0.282380491495132], 0.766666650772095, 0.321279972791672, [0.966666638851166, 0.966666638851166, 0.766666650772095], [0.14349028468132, 0.207735612988472, 0.321279972791672]),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [12.6596629619598, 28.0149381160736, 43.0508260726929], [u'accuracy'], u'categorical_crossentropy', 0.841666638851166, 0.374685227870941, [0.933333337306976, 0.899999976158142, 0.841666638851166], [0.162063658237457, 0.223208039999008, 0.374685227870941], 0.733333349227905, 0.698961436748505, [0.866666674613953, 0.733333349227905, 0.733333349227905], [0.297752887010574, 0.422860831022263, 0.698961436748505]),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [14.1109259128571, 29.4764680862427, 44.847892999649], [u'accuracy'], u'categorical_crossentropy', 0.858333349227905, 0.323420703411102, [0.975000023841858, 0.975000023841858, 0.858333349227905], [0.260769069194794, 0.237523972988129, 0.323420703411102], 0.699999988079071, 0.456004768610001, [0.933333337306976, 0.933333337306976, 0.699999988079071], [0.302719950675964, 0.286018937826157, 0.456004768610001]),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [13.6740920543671, 29.0340840816498, 44.4022340774536], [u'accuracy'], u'categorical_crossentropy', 0.699999988079071, 0.525134742259979, [0.699999988079071, 0.699999988079071, 0.699999988079071], [0.579447448253632, 0.552316129207611, 0.525134742259979], 0.566666662693024, 0.634039282798767, [0.566666662693024, 0.566666662693024, 0.566666662693024], [0.681210398674011, 0.654673635959625, 0.634039282798767]),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [14.3272860050201, 29.6978969573975, 45.0662181377411], [u'accuracy'], u'categorical_crossentropy', 0.550000011920929, 0.82413375377655, [0.600000023841858, 0.600000023841858, 0.550000011920929], [0.849512219429016, 0.836838126182556, 0.82413375377655], 0.433333337306976, 0.862674951553345, [0.533333361148834, 0.533333361148834, 0.433333337306976], [0.907541394233704, 0.897742569446564, 0.862674951553345])]"
+      ]
+     },
+     "execution_count": 41,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_info ORDER BY validation_metrics_final DESC;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Plot validation results:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 42,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "7 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM iris_multi_model_info ORDER BY validation_loss ASC LIMIT 7;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "df_summary = %sql SELECT * FROM iris_multi_model_summary;\n",
+    "df_summary = df_summary.DataFrame()\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Iteration')\n",
+    "ax_metric.set_ylabel('Metric')\n",
+    "ax_metric.set_title('Validation metric curve')\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Iteration')\n",
+    "ax_loss.set_ylabel('Loss')\n",
+    "ax_loss.set_title('Validation loss curve')\n",
+    "\n",
+    "iters = df_summary['metrics_iters'][0]\n",
+    "\n",
+    "for mst_key in df_results['mst_key']:\n",
+    "    df_output_info = %sql SELECT validation_metrics,validation_loss FROM iris_multi_model_info WHERE mst_key = $mst_key\n",
+    "    df_output_info = df_output_info.DataFrame()\n",
+    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
+    "    validation_loss = df_output_info['validation_loss'][0]\n",
+    "    \n",
+    "    ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
+    "    ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Preprocessor-for-images-distribution-rules-v1.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/Preprocessor-for-images-distribution-rules-v1-checkpoint.ipynb
similarity index 98%
rename from community-artifacts/Deep-learning/Preprocessor-for-images-distribution-rules-v1.ipynb
rename to community-artifacts/Deep-learning/.ipynb_checkpoints/Preprocessor-for-images-distribution-rules-v1-checkpoint.ipynb
index b457303..0ae2b4c 100644
--- a/community-artifacts/Deep-learning/Preprocessor-for-images-distribution-rules-v1.ipynb
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/Preprocessor-for-images-distribution-rules-v1-checkpoint.ipynb
@@ -1198,7 +1198,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1465,7 +1465,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1657,7 +1657,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1805,7 +1805,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1938,7 +1938,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_val_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_val_packed ORDER BY __dist_key__;"
    ]
   }
  ],
diff --git a/community-artifacts/Deep-learning/Preprocessor-for-images-v2.ipynb b/community-artifacts/Deep-learning/.ipynb_checkpoints/Preprocessor-for-images-v2-checkpoint.ipynb
similarity index 60%
copy from community-artifacts/Deep-learning/Preprocessor-for-images-v2.ipynb
copy to community-artifacts/Deep-learning/.ipynb_checkpoints/Preprocessor-for-images-v2-checkpoint.ipynb
index cb76d1e..3f74a2e 100644
--- a/community-artifacts/Deep-learning/Preprocessor-for-images-v2.ipynb
+++ b/community-artifacts/Deep-learning/.ipynb_checkpoints/Preprocessor-for-images-v2-checkpoint.ipynb
@@ -39,42 +39,17 @@
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
     "%load_ext sql"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 2,
+   "execution_count": 3,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
     "# Greenplum Database 5.x on GCP - via tunnel\n",
     "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
@@ -84,7 +59,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 3,
+   "execution_count": 5,
    "metadata": {},
    "outputs": [
     {
@@ -102,15 +77,15 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-85-g4bac900, cmake configuration time: Wed Mar  3 20:37:11 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-85-g4bac900, cmake configuration time: Wed Mar  3 20:37:11 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
-     "execution_count": 3,
+     "execution_count": 5,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -132,7 +107,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 4,
+   "execution_count": 6,
    "metadata": {},
    "outputs": [
     {
@@ -153,271 +128,271 @@
        "        <th>species</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[152, 186, 35], [102, 145, 138]], [[40, 249, 108], [175, 207, 70]]]</td>\n",
+       "        <td>[[[17, 201, 110], [175, 136, 179]], [[102, 57, 24], [110, 199, 64]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[205, 85, 56], [209, 11, 117]], [[86, 82, 41], [226, 192, 132]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[234, 110, 251], [147, 18, 158]], [[55, 79, 14], [140, 50, 143]]]</td>\n",
+       "        <td>[[[209, 227, 160], [86, 88, 177]], [[31, 198, 96], [167, 122, 198]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[146, 52, 167], [210, 33, 116]], [[38, 89, 69], [50, 207, 155]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[247, 125, 68], [124, 196, 20]], [[95, 100, 107], [183, 21, 138]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[117, 49, 248], [59, 18, 137]], [[110, 186, 91], [143, 46, 129]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[115, 179, 183], [14, 54, 175]], [[138, 122, 42], [79, 142, 137]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[249, 65, 200], [131, 191, 61]], [[180, 182, 119], [199, 63, 230]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[154, 117, 174], [27, 94, 33]], [[206, 21, 46], [4, 196, 185]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[238, 8, 12], [120, 187, 4]], [[184, 130, 135], [119, 191, 59]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[179, 202, 20], [219, 198, 173]], [[149, 233, 18], [38, 115, 59]]]</td>\n",
+       "        <td>[[[55, 2, 109], [28, 130, 7]], [[146, 48, 34], [240, 81, 240]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[223, 234, 239], [37, 253, 217]], [[147, 248, 108], [166, 150, 162]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[164, 46, 39], [51, 130, 218]], [[253, 150, 181], [195, 66, 75]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[85, 113, 32], [144, 145, 255]], [[122, 127, 36], [118, 88, 183]]]</td>\n",
+       "        <td>[[[128, 244, 200], [57, 113, 182]], [[64, 125, 46], [251, 129, 230]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[195, 93, 4], [102, 81, 168]], [[148, 120, 219], [21, 82, 217]]]</td>\n",
+       "        <td>[[[8, 93, 61], [67, 139, 115]], [[69, 248, 144], [199, 255, 33]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[8, 156, 237], [82, 72, 66]], [[196, 104, 210], [84, 103, 75]]]</td>\n",
+       "        <td>[[[33, 17, 73], [17, 21, 201]], [[5, 222, 1], [118, 148, 66]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[139, 194, 43], [66, 48, 239]], [[159, 52, 84], [240, 220, 232]]]</td>\n",
+       "        <td>[[[194, 61, 116], [168, 187, 124]], [[6, 247, 192], [145, 106, 5]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[183, 253, 187], [144, 168, 194]], [[44, 150, 21], [116, 216, 216]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[170, 44, 68], [245, 256, 207]], [[183, 43, 17], [231, 25, 176]]]</td>\n",
+       "        <td>[[[250, 204, 135], [27, 196, 168]], [[44, 12, 185], [65, 213, 190]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[110, 160, 246], [85, 9, 173]], [[82, 195, 61], [251, 134, 105]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[154, 222, 104], [114, 186, 18]], [[159, 254, 7], [158, 205, 190]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[222, 165, 227], [142, 191, 80]], [[46, 182, 165], [55, 99, 248]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[161, 243, 128], [10, 131, 26]], [[232, 235, 141], [162, 253, 43]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[4, 202, 109], [194, 147, 75]], [[103, 117, 217], [39, 197, 8]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[107, 63, 64], [99, 57, 224]], [[86, 185, 234], [216, 212, 210]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[96, 116, 192], [140, 21, 196]], [[85, 130, 135], [232, 206, 238]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[167, 20, 35], [174, 241, 142]], [[237, 48, 241], [38, 16, 70]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[251, 31, 179], [205, 226, 19]], [[65, 162, 159], [86, 103, 244]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[237, 220, 166], [219, 58, 77]], [[239, 93, 251], [224, 235, 232]]]</td>\n",
+       "        <td>[[[215, 52, 179], [25, 39, 117]], [[86, 155, 29], [16, 24, 35]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[219, 14, 33], [34, 237, 28]], [[64, 160, 232], [34, 180, 41]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[83, 127, 43], [71, 87, 24]], [[35, 253, 243], [93, 74, 227]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[69, 195, 165], [45, 212, 129]], [[59, 245, 162], [40, 16, 226]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[248, 5, 124], [34, 201, 206]], [[161, 244, 21], [248, 13, 57]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[0, 150, 63], [227, 80, 132]], [[166, 245, 176], [121, 118, 235]]]</td>\n",
+       "        <td>[[[215, 180, 113], [220, 61, 107]], [[168, 196, 134], [108, 108, 178]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[104, 42, 37], [143, 227, 111]], [[96, 135, 172], [12, 207, 100]]]</td>\n",
+       "        <td>[[[38, 244, 77], [228, 19, 36]], [[24, 198, 60], [63, 59, 146]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[221, 150, 126], [143, 129, 93]], [[92, 235, 60], [174, 100, 100]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[216, 163, 35], [249, 33, 139]], [[35, 70, 26], [6, 181, 122]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[97, 134, 93], [198, 94, 57]], [[92, 219, 200], [221, 56, 35]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[116, 210, 44], [216, 129, 4]], [[123, 164, 253], [156, 47, 32]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[73, 39, 151], [196, 180, 248]], [[74, 16, 190], [168, 74, 26]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[18, 246, 187], [53, 190, 47]], [[7, 234, 8], [136, 238, 131]]]</td>\n",
+       "        <td>[[[89, 162, 242], [124, 169, 202]], [[48, 26, 166], [109, 134, 78]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[235, 31, 91], [11, 1, 164]], [[49, 152, 103], [229, 144, 177]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[78, 89, 63], [104, 220, 81]], [[94, 151, 134], [28, 199, 141]]]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[206, 21, 244], [81, 65, 223]], [[112, 155, 234], [113, 63, 27]]]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[166, 1, 152], [88, 246, 230]], [[176, 54, 78], [140, 135, 172]]]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[13, 200, 234], [155, 207, 185]], [[176, 195, 10], [240, 162, 122]]]</td>\n",
+       "        <td>[[[12, 185, 157], [191, 49, 195]], [[178, 126, 167], [197, 162, 191]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[140, 235, 202], [167, 244, 113]], [[168, 140, 200], [158, 114, 121]]]</td>\n",
+       "        <td>[[[222, 254, 199], [112, 217, 32]], [[18, 203, 156], [187, 148, 204]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[192, 5, 91], [108, 41, 104]], [[52, 19, 3], [3, 204, 178]]]</td>\n",
+       "        <td>[[[58, 56, 91], [136, 105, 103]], [[65, 6, 38], [114, 201, 216]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[214, 162, 103], [80, 46, 243]], [[60, 248, 154], [47, 105, 65]]]</td>\n",
+       "        <td>[[[111, 157, 147], [46, 41, 113]], [[44, 240, 226], [5, 15, 244]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[49, 223, 45], [170, 179, 237]], [[175, 14, 89], [216, 118, 141]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[121, 144, 183], [43, 86, 141]], [[205, 189, 221], [251, 176, 25]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[74, 72, 92], [139, 3, 141]], [[106, 48, 55], [29, 30, 230]]]</td>\n",
+       "        <td>[[[171, 175, 100], [119, 132, 158]], [[175, 224, 37], [24, 71, 102]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[119, 190, 161], [4, 168, 25]], [[148, 95, 68], [234, 236, 17]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[201, 13, 87], [226, 256, 161]], [[42, 92, 44], [45, 233, 150]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[33, 179, 122], [7, 222, 241]], [[196, 127, 246], [108, 152, 138]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[220, 116, 183], [237, 27, 128]], [[250, 115, 98], [250, 19, 140]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[64, 184, 64], [214, 21, 96]], [[137, 143, 103], [103, 129, 43]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[118, 151, 126], [1, 99, 90]], [[117, 26, 71], [144, 154, 65]]]</td>\n",
+       "        <td>[[[174, 243, 194], [14, 219, 228]], [[86, 254, 177], [214, 92, 119]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[252, 59, 22], [136, 146, 86]], [[64, 209, 43], [85, 49, 181]]]</td>\n",
+       "        <td>[[[24, 120, 130], [256, 167, 172]], [[142, 93, 141], [165, 156, 239]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[81, 253, 127], [77, 53, 45]], [[64, 246, 59], [27, 219, 145]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[140, 103, 118], [4, 127, 142]], [[124, 1, 142], [35, 173, 28]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[58, 193, 28], [41, 201, 109]], [[38, 72, 186], [90, 116, 250]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[176, 21, 44], [65, 47, 184]], [[168, 165, 187], [39, 50, 55]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[192, 90, 212], [220, 218, 14]], [[157, 246, 55], [102, 99, 93]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[152, 28, 101], [195, 2, 220]], [[91, 128, 220], [189, 218, 81]]]</td>\n",
+       "        <td>[[[29, 183, 34], [23, 8, 210]], [[44, 51, 19], [91, 235, 187]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[166, 226, 50], [222, 9, 242]], [[56, 222, 206], [18, 236, 108]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[35, 210, 106], [127, 127, 134]], [[55, 162, 157], [62, 115, 201]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[134, 36, 93], [65, 36, 4]], [[35, 86, 225], [44, 73, 25]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[23, 42, 246], [130, 49, 24]], [[84, 155, 152], [212, 34, 206]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[191, 13, 233], [136, 126, 111]], [[173, 220, 176], [209, 223, 211]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[192, 255, 112], [217, 8, 134]], [[3, 254, 9], [53, 22, 93]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[174, 48, 241], [124, 166, 176]], [[136, 142, 56], [7, 253, 229]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[173, 181, 193], [127, 220, 130]], [[126, 76, 91], [135, 210, 94]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[219, 147, 155], [56, 99, 72]], [[104, 84, 196], [14, 4, 77]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[60, 83, 153], [33, 54, 70]], [[214, 247, 197], [179, 121, 67]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[212, 202, 209], [50, 78, 172]], [[196, 233, 227], [39, 49, 76]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[246, 89, 127], [66, 245, 187]], [[150, 142, 220], [203, 212, 178]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[153, 101, 60], [220, 100, 15]], [[166, 52, 65], [245, 224, 5]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[195, 44, 15], [15, 167, 4]], [[104, 38, 71], [94, 225, 220]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[189, 168, 192], [112, 107, 89]], [[213, 166, 54], [56, 181, 220]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[246, 208, 77], [251, 174, 16]], [[39, 189, 31], [206, 193, 135]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[8, 229, 214], [228, 209, 147]], [[140, 146, 3], [247, 235, 215]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[33, 16, 82], [252, 124, 72]], [[205, 201, 68], [123, 217, 107]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[248, 57, 249], [127, 46, 1]], [[100, 3, 229], [54, 150, 113]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[([[[152, 186, 35], [102, 145, 138]], [[40, 249, 108], [175, 207, 70]]], u'cat'),\n",
-       " ([[[234, 110, 251], [147, 18, 158]], [[55, 79, 14], [140, 50, 143]]], u'cat'),\n",
-       " ([[[179, 202, 20], [219, 198, 173]], [[149, 233, 18], [38, 115, 59]]], u'cat'),\n",
-       " ([[[223, 234, 239], [37, 253, 217]], [[147, 248, 108], [166, 150, 162]]], u'bird'),\n",
-       " ([[[164, 46, 39], [51, 130, 218]], [[253, 150, 181], [195, 66, 75]]], u'bird'),\n",
-       " ([[[85, 113, 32], [144, 145, 255]], [[122, 127, 36], [118, 88, 183]]], u'dog'),\n",
-       " ([[[195, 93, 4], [102, 81, 168]], [[148, 120, 219], [21, 82, 217]]], u'bird'),\n",
-       " ([[[8, 156, 237], [82, 72, 66]], [[196, 104, 210], [84, 103, 75]]], u'bird'),\n",
-       " ([[[139, 194, 43], [66, 48, 239]], [[159, 52, 84], [240, 220, 232]]], u'dog'),\n",
-       " ([[[183, 253, 187], [144, 168, 194]], [[44, 150, 21], [116, 216, 216]]], u'bird'),\n",
-       " ([[[170, 44, 68], [245, 256, 207]], [[183, 43, 17], [231, 25, 176]]], u'cat'),\n",
-       " ([[[110, 160, 246], [85, 9, 173]], [[82, 195, 61], [251, 134, 105]]], u'dog'),\n",
-       " ([[[154, 222, 104], [114, 186, 18]], [[159, 254, 7], [158, 205, 190]]], u'bird'),\n",
-       " ([[[222, 165, 227], [142, 191, 80]], [[46, 182, 165], [55, 99, 248]]], u'bird'),\n",
-       " ([[[161, 243, 128], [10, 131, 26]], [[232, 235, 141], [162, 253, 43]]], u'dog'),\n",
-       " ([[[4, 202, 109], [194, 147, 75]], [[103, 117, 217], [39, 197, 8]]], u'bird'),\n",
-       " ([[[107, 63, 64], [99, 57, 224]], [[86, 185, 234], [216, 212, 210]]], u'bird'),\n",
-       " ([[[96, 116, 192], [140, 21, 196]], [[85, 130, 135], [232, 206, 238]]], u'dog'),\n",
-       " ([[[167, 20, 35], [174, 241, 142]], [[237, 48, 241], [38, 16, 70]]], u'bird'),\n",
-       " ([[[251, 31, 179], [205, 226, 19]], [[65, 162, 159], [86, 103, 244]]], u'bird'),\n",
-       " ([[[237, 220, 166], [219, 58, 77]], [[239, 93, 251], [224, 235, 232]]], u'cat'),\n",
-       " ([[[219, 14, 33], [34, 237, 28]], [[64, 160, 232], [34, 180, 41]]], u'bird'),\n",
-       " ([[[83, 127, 43], [71, 87, 24]], [[35, 253, 243], [93, 74, 227]]], u'bird'),\n",
-       " ([[[69, 195, 165], [45, 212, 129]], [[59, 245, 162], [40, 16, 226]]], u'bird'),\n",
-       " ([[[248, 5, 124], [34, 201, 206]], [[161, 244, 21], [248, 13, 57]]], u'bird'),\n",
-       " ([[[0, 150, 63], [227, 80, 132]], [[166, 245, 176], [121, 118, 235]]], u'dog'),\n",
-       " ([[[104, 42, 37], [143, 227, 111]], [[96, 135, 172], [12, 207, 100]]], u'bird'),\n",
-       " ([[[221, 150, 126], [143, 129, 93]], [[92, 235, 60], [174, 100, 100]]], u'bird'),\n",
-       " ([[[216, 163, 35], [249, 33, 139]], [[35, 70, 26], [6, 181, 122]]], u'dog'),\n",
-       " ([[[97, 134, 93], [198, 94, 57]], [[92, 219, 200], [221, 56, 35]]], u'bird'),\n",
-       " ([[[116, 210, 44], [216, 129, 4]], [[123, 164, 253], [156, 47, 32]]], u'bird'),\n",
-       " ([[[73, 39, 151], [196, 180, 248]], [[74, 16, 190], [168, 74, 26]]], u'dog'),\n",
-       " ([[[18, 246, 187], [53, 190, 47]], [[7, 234, 8], [136, 238, 131]]], u'cat'),\n",
-       " ([[[235, 31, 91], [11, 1, 164]], [[49, 152, 103], [229, 144, 177]]], u'bird'),\n",
-       " ([[[78, 89, 63], [104, 220, 81]], [[94, 151, 134], [28, 199, 141]]], u'cat'),\n",
-       " ([[[206, 21, 244], [81, 65, 223]], [[112, 155, 234], [113, 63, 27]]], u'cat'),\n",
-       " ([[[166, 1, 152], [88, 246, 230]], [[176, 54, 78], [140, 135, 172]]], u'cat'),\n",
-       " ([[[13, 200, 234], [155, 207, 185]], [[176, 195, 10], [240, 162, 122]]], u'dog'),\n",
-       " ([[[140, 235, 202], [167, 244, 113]], [[168, 140, 200], [158, 114, 121]]], u'bird'),\n",
-       " ([[[192, 5, 91], [108, 41, 104]], [[52, 19, 3], [3, 204, 178]]], u'bird'),\n",
-       " ([[[214, 162, 103], [80, 46, 243]], [[60, 248, 154], [47, 105, 65]]], u'bird'),\n",
-       " ([[[49, 223, 45], [170, 179, 237]], [[175, 14, 89], [216, 118, 141]]], u'bird'),\n",
-       " ([[[121, 144, 183], [43, 86, 141]], [[205, 189, 221], [251, 176, 25]]], u'bird'),\n",
-       " ([[[74, 72, 92], [139, 3, 141]], [[106, 48, 55], [29, 30, 230]]], u'cat'),\n",
-       " ([[[119, 190, 161], [4, 168, 25]], [[148, 95, 68], [234, 236, 17]]], u'dog'),\n",
-       " ([[[201, 13, 87], [226, 256, 161]], [[42, 92, 44], [45, 233, 150]]], u'dog'),\n",
-       " ([[[33, 179, 122], [7, 222, 241]], [[196, 127, 246], [108, 152, 138]]], u'bird'),\n",
-       " ([[[220, 116, 183], [237, 27, 128]], [[250, 115, 98], [250, 19, 140]]], u'dog'),\n",
-       " ([[[64, 184, 64], [214, 21, 96]], [[137, 143, 103], [103, 129, 43]]], u'bird'),\n",
-       " ([[[118, 151, 126], [1, 99, 90]], [[117, 26, 71], [144, 154, 65]]], u'cat'),\n",
-       " ([[[252, 59, 22], [136, 146, 86]], [[64, 209, 43], [85, 49, 181]]], u'bird'),\n",
-       " ([[[152, 28, 101], [195, 2, 220]], [[91, 128, 220], [189, 218, 81]]], u'bird')]"
+       "[([[[17, 201, 110], [175, 136, 179]], [[102, 57, 24], [110, 199, 64]]], u'bird'),\n",
+       " ([[[205, 85, 56], [209, 11, 117]], [[86, 82, 41], [226, 192, 132]]], u'cat'),\n",
+       " ([[[209, 227, 160], [86, 88, 177]], [[31, 198, 96], [167, 122, 198]]], u'bird'),\n",
+       " ([[[146, 52, 167], [210, 33, 116]], [[38, 89, 69], [50, 207, 155]]], u'dog'),\n",
+       " ([[[247, 125, 68], [124, 196, 20]], [[95, 100, 107], [183, 21, 138]]], u'dog'),\n",
+       " ([[[117, 49, 248], [59, 18, 137]], [[110, 186, 91], [143, 46, 129]]], u'bird'),\n",
+       " ([[[115, 179, 183], [14, 54, 175]], [[138, 122, 42], [79, 142, 137]]], u'bird'),\n",
+       " ([[[249, 65, 200], [131, 191, 61]], [[180, 182, 119], [199, 63, 230]]], u'dog'),\n",
+       " ([[[154, 117, 174], [27, 94, 33]], [[206, 21, 46], [4, 196, 185]]], u'dog'),\n",
+       " ([[[238, 8, 12], [120, 187, 4]], [[184, 130, 135], [119, 191, 59]]], u'cat'),\n",
+       " ([[[55, 2, 109], [28, 130, 7]], [[146, 48, 34], [240, 81, 240]]], u'cat'),\n",
+       " ([[[128, 244, 200], [57, 113, 182]], [[64, 125, 46], [251, 129, 230]]], u'dog'),\n",
+       " ([[[8, 93, 61], [67, 139, 115]], [[69, 248, 144], [199, 255, 33]]], u'bird'),\n",
+       " ([[[33, 17, 73], [17, 21, 201]], [[5, 222, 1], [118, 148, 66]]], u'bird'),\n",
+       " ([[[194, 61, 116], [168, 187, 124]], [[6, 247, 192], [145, 106, 5]]], u'dog'),\n",
+       " ([[[250, 204, 135], [27, 196, 168]], [[44, 12, 185], [65, 213, 190]]], u'cat'),\n",
+       " ([[[215, 52, 179], [25, 39, 117]], [[86, 155, 29], [16, 24, 35]]], u'cat'),\n",
+       " ([[[215, 180, 113], [220, 61, 107]], [[168, 196, 134], [108, 108, 178]]], u'dog'),\n",
+       " ([[[38, 244, 77], [228, 19, 36]], [[24, 198, 60], [63, 59, 146]]], u'bird'),\n",
+       " ([[[89, 162, 242], [124, 169, 202]], [[48, 26, 166], [109, 134, 78]]], u'cat'),\n",
+       " ([[[12, 185, 157], [191, 49, 195]], [[178, 126, 167], [197, 162, 191]]], u'dog'),\n",
+       " ([[[222, 254, 199], [112, 217, 32]], [[18, 203, 156], [187, 148, 204]]], u'bird'),\n",
+       " ([[[58, 56, 91], [136, 105, 103]], [[65, 6, 38], [114, 201, 216]]], u'bird'),\n",
+       " ([[[111, 157, 147], [46, 41, 113]], [[44, 240, 226], [5, 15, 244]]], u'bird'),\n",
+       " ([[[171, 175, 100], [119, 132, 158]], [[175, 224, 37], [24, 71, 102]]], u'cat'),\n",
+       " ([[[174, 243, 194], [14, 219, 228]], [[86, 254, 177], [214, 92, 119]]], u'cat'),\n",
+       " ([[[24, 120, 130], [256, 167, 172]], [[142, 93, 141], [165, 156, 239]]], u'cat'),\n",
+       " ([[[81, 253, 127], [77, 53, 45]], [[64, 246, 59], [27, 219, 145]]], u'cat'),\n",
+       " ([[[140, 103, 118], [4, 127, 142]], [[124, 1, 142], [35, 173, 28]]], u'dog'),\n",
+       " ([[[58, 193, 28], [41, 201, 109]], [[38, 72, 186], [90, 116, 250]]], u'cat'),\n",
+       " ([[[176, 21, 44], [65, 47, 184]], [[168, 165, 187], [39, 50, 55]]], u'cat'),\n",
+       " ([[[192, 90, 212], [220, 218, 14]], [[157, 246, 55], [102, 99, 93]]], u'bird'),\n",
+       " ([[[29, 183, 34], [23, 8, 210]], [[44, 51, 19], [91, 235, 187]]], u'bird'),\n",
+       " ([[[166, 226, 50], [222, 9, 242]], [[56, 222, 206], [18, 236, 108]]], u'cat'),\n",
+       " ([[[35, 210, 106], [127, 127, 134]], [[55, 162, 157], [62, 115, 201]]], u'dog'),\n",
+       " ([[[134, 36, 93], [65, 36, 4]], [[35, 86, 225], [44, 73, 25]]], u'cat'),\n",
+       " ([[[23, 42, 246], [130, 49, 24]], [[84, 155, 152], [212, 34, 206]]], u'dog'),\n",
+       " ([[[191, 13, 233], [136, 126, 111]], [[173, 220, 176], [209, 223, 211]]], u'cat'),\n",
+       " ([[[192, 255, 112], [217, 8, 134]], [[3, 254, 9], [53, 22, 93]]], u'bird'),\n",
+       " ([[[174, 48, 241], [124, 166, 176]], [[136, 142, 56], [7, 253, 229]]], u'bird'),\n",
+       " ([[[173, 181, 193], [127, 220, 130]], [[126, 76, 91], [135, 210, 94]]], u'dog'),\n",
+       " ([[[219, 147, 155], [56, 99, 72]], [[104, 84, 196], [14, 4, 77]]], u'dog'),\n",
+       " ([[[60, 83, 153], [33, 54, 70]], [[214, 247, 197], [179, 121, 67]]], u'bird'),\n",
+       " ([[[212, 202, 209], [50, 78, 172]], [[196, 233, 227], [39, 49, 76]]], u'dog'),\n",
+       " ([[[246, 89, 127], [66, 245, 187]], [[150, 142, 220], [203, 212, 178]]], u'bird'),\n",
+       " ([[[153, 101, 60], [220, 100, 15]], [[166, 52, 65], [245, 224, 5]]], u'bird'),\n",
+       " ([[[195, 44, 15], [15, 167, 4]], [[104, 38, 71], [94, 225, 220]]], u'bird'),\n",
+       " ([[[189, 168, 192], [112, 107, 89]], [[213, 166, 54], [56, 181, 220]]], u'dog'),\n",
+       " ([[[246, 208, 77], [251, 174, 16]], [[39, 189, 31], [206, 193, 135]]], u'bird'),\n",
+       " ([[[8, 229, 214], [228, 209, 147]], [[140, 146, 3], [247, 235, 215]]], u'dog'),\n",
+       " ([[[33, 16, 82], [252, 124, 72]], [[205, 201, 68], [123, 217, 107]]], u'cat'),\n",
+       " ([[[248, 57, 249], [127, 46, 1]], [[100, 3, 229], [54, 150, 113]]], u'bird')]"
       ]
      },
-     "execution_count": 4,
+     "execution_count": 6,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -463,7 +438,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 5,
+   "execution_count": 8,
    "metadata": {},
    "outputs": [
     {
@@ -480,8 +455,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -500,7 +475,7 @@
        "[([26, 2, 2, 3], [26, 3], 0), ([26, 2, 2, 3], [26, 3], 1)]"
       ]
      },
-     "execution_count": 5,
+     "execution_count": 8,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -517,7 +492,7 @@
     "                                        255                   -- Normalizing constant\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -531,7 +506,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
+   "execution_count": 9,
    "metadata": {},
    "outputs": [
     {
@@ -551,7 +526,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -561,23 +536,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 26, 255.0, 3, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 26, 255.0, [3], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 6,
+     "execution_count": 9,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -599,7 +574,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 10,
    "metadata": {},
    "outputs": [
     {
@@ -616,8 +591,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -636,7 +611,7 @@
        "[([26, 2, 2, 3], [26, 3], 0), ([26, 2, 2, 3], [26, 3], 1)]"
       ]
      },
-     "execution_count": 7,
+     "execution_count": 10,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -644,6 +619,7 @@
    "source": [
     "%%sql\n",
     "DROP TABLE IF EXISTS val_image_data_packed, val_image_data_packed_summary;\n",
+    "\n",
     "SELECT madlib.validation_preprocessor_dl(\n",
     "      'image_data',             -- Source table\n",
     "      'val_image_data_packed',  -- Output table\n",
@@ -652,7 +628,8 @@
     "      'image_data_packed',      -- From training preprocessor step\n",
     "      NULL                      -- Buffer size\n",
     "      ); \n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
+    "\n",
+    "SELECT rgb_shape, species_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -664,7 +641,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 8,
+   "execution_count": 11,
    "metadata": {},
    "outputs": [
     {
@@ -684,7 +661,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -694,23 +671,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>val_image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'val_image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 26, 255.0, 3, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'val_image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 26, 255.0, [3], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 8,
+     "execution_count": 11,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -731,7 +708,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 9,
+   "execution_count": 12,
    "metadata": {},
    "outputs": [
     {
@@ -752,271 +729,271 @@
        "        <th>species</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[19, 126, 250, 219, 119, 255, 86, 152, 200, 36, 57, 188]</td>\n",
+       "        <td>[168, 228, 110, 3, 51, 104, 192, 23, 120, 249, 96, 99]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[20, 145, 109, 135, 149, 100, 39, 66, 124, 102, 77, 140]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[125, 32, 244, 23, 201, 156, 251, 55, 159, 47, 160, 95]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[49, 201, 114, 38, 201, 8, 101, 172, 88, 233, 82, 78]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[203, 196, 132, 57, 220, 151, 183, 214, 113, 46, 213, 200]</td>\n",
+       "        <td>[24, 88, 166, 123, 193, 186, 12, 46, 65, 161, 145, 104]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[157, 236, 255, 90, 38, 48, 35, 152, 86, 236, 160, 187]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[248, 164, 234, 70, 61, 181, 10, 193, 238, 229, 88, 165]</td>\n",
+       "        <td>[14, 206, 47, 154, 85, 172, 186, 73, 196, 131, 229, 191]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[201, 210, 145, 145, 152, 46, 125, 151, 135, 163, 199, 170]</td>\n",
+       "        <td>[131, 238, 90, 227, 51, 114, 59, 217, 237, 252, 147, 248]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[29, 150, 219, 216, 46, 211, 124, 24, 25, 186, 205, 35]</td>\n",
+       "        <td>[211, 153, 187, 59, 123, 200, 10, 171, 98, 95, 87, 28]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[187, 8, 211, 95, 196, 156, 50, 84, 45, 202, 130, 170]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[9, 77, 40, 179, 136, 69, 74, 98, 29, 120, 53, 153]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[78, 83, 93, 113, 206, 23, 121, 160, 119, 61, 60, 168]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[105, 114, 19, 19, 211, 28, 96, 251, 208, 232, 64, 25]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[93, 145, 128, 246, 33, 206, 73, 126, 63, 22, 150, 184]</td>\n",
+       "        <td>[26, 159, 140, 217, 89, 15, 199, 179, 242, 250, 37, 45]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[12, 245, 243, 181, 134, 92, 39, 153, 112, 250, 181, 208]</td>\n",
+       "        <td>[18, 41, 102, 10, 82, 57, 163, 13, 116, 30, 213, 126]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[133, 184, 53, 158, 3, 145, 47, 130, 135, 81, 80, 208]</td>\n",
+       "        <td>[56, 221, 31, 84, 132, 58, 243, 16, 19, 76, 31, 218]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[143, 230, 101, 71, 156, 113, 61, 143, 37, 195, 235, 76]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[91, 70, 17, 43, 59, 150, 227, 111, 53, 229, 0, 100]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[136, 181, 184, 87, 132, 71, 61, 232, 143, 218, 89, 203]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[126, 142, 84, 203, 234, 175, 17, 251, 217, 75, 145, 188]</td>\n",
+       "        <td>[17, 212, 36, 62, 167, 54, 103, 13, 64, 185, 70, 227]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[198, 162, 187, 42, 9, 67, 223, 193, 154, 99, 9, 215]</td>\n",
-       "        <td>cat</td>\n",
+       "        <td>[186, 1, 155, 56, 201, 211, 21, 233, 38, 153, 34, 25]</td>\n",
+       "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[151, 177, 164, 98, 25, 35, 240, 109, 237, 218, 28, 254]</td>\n",
+       "        <td>[53, 101, 200, 15, 101, 217, 227, 137, 23, 138, 191, 126]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[255, 54, 220, 226, 252, 150, 227, 151, 207, 172, 105, 227]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[144, 124, 183, 169, 37, 237, 14, 237, 252, 115, 198, 222]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[246, 73, 102, 178, 4, 45, 84, 191, 87, 93, 2, 54]</td>\n",
+       "        <td>[222, 104, 188, 92, 254, 187, 146, 219, 157, 142, 113, 128]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[156, 153, 39, 115, 228, 190, 35, 136, 32, 61, 171, 16]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[152, 234, 198, 149, 191, 188, 222, 37, 110, 226, 82, 194]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[169, 31, 163, 222, 61, 62, 119, 100, 177, 91, 34, 213]</td>\n",
+       "        <td>[64, 44, 142, 35, 193, 30, 159, 120, 199, 196, 101, 213]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[67, 17, 141, 83, 188, 37, 61, 130, 187, 252, 62, 153]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[172, 123, 115, 110, 28, 28, 140, 191, 250, 202, 253, 113]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[225, 113, 99, 228, 109, 158, 250, 245, 47, 79, 52, 1]</td>\n",
+       "        <td>[96, 72, 120, 63, 69, 86, 167, 0, 177, 165, 187, 67]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[137, 50, 48, 110, 202, 76, 211, 142, 78, 174, 232, 206]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[166, 168, 219, 125, 201, 188, 238, 44, 160, 92, 202, 153]</td>\n",
+       "        <td>[88, 210, 241, 216, 246, 48, 4, 132, 83, 197, 162, 242]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[249, 233, 133, 249, 100, 14, 43, 147, 124, 246, 223, 78]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[45, 253, 108, 251, 135, 18, 163, 98, 143, 108, 30, 126]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[190, 217, 97, 87, 41, 90, 64, 174, 84, 164, 188, 127]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[56, 117, 22, 134, 249, 67, 130, 101, 62, 9, 119, 225]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[6, 78, 138, 132, 230, 72, 93, 71, 159, 134, 161, 223]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[245, 131, 240, 116, 186, 40, 233, 209, 174, 226, 20, 48]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[82, 57, 189, 52, 165, 195, 129, 46, 71, 103, 118, 163]</td>\n",
+       "        <td>[105, 182, 162, 62, 104, 2, 134, 223, 65, 203, 53, 231]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[21, 41, 79, 244, 93, 68, 120, 78, 184, 50, 117, 161]</td>\n",
+       "        <td>[230, 140, 134, 42, 12, 223, 251, 252, 183, 241, 44, 188]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[127, 129, 24, 113, 190, 129, 40, 96, 191, 143, 98, 69]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[162, 16, 163, 137, 219, 137, 21, 97, 179, 33, 64, 174]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[35, 131, 23, 83, 201, 105, 140, 134, 157, 48, 73, 30]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[144, 133, 213, 51, 51, 234, 93, 130, 222, 186, 198, 86]</td>\n",
+       "        <td>[247, 159, 74, 179, 21, 201, 51, 45, 58, 241, 175, 98]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[126, 136, 125, 31, 139, 160, 161, 162, 242, 106, 11, 126]</td>\n",
+       "        <td>[110, 241, 179, 179, 96, 85, 195, 3, 222, 158, 140, 244]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[168, 174, 58, 198, 13, 202, 75, 226, 254, 126, 204, 90]</td>\n",
+       "        <td>[63, 21, 63, 237, 50, 54, 140, 124, 233, 162, 69, 28]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[170, 20, 197, 1, 28, 67, 137, 153, 97, 20, 57, 3]</td>\n",
-       "        <td>bird</td>\n",
+       "        <td>[94, 111, 234, 231, 203, 73, 118, 97, 57, 254, 209, 131]</td>\n",
+       "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[43, 109, 193, 169, 94, 105, 88, 152, 46, 101, 98, 121]</td>\n",
+       "        <td>[246, 73, 151, 78, 201, 43, 59, 1, 215, 155, 138, 63]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[46, 186, 18, 158, 254, 111, 13, 232, 86, 216, 49, 204]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[95, 247, 19, 186, 247, 189, 206, 188, 190, 234, 254, 70]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[96, 90, 188, 98, 16, 231, 207, 209, 145, 45, 58, 232]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[104, 77, 39, 226, 148, 134, 217, 166, 64, 207, 99, 14]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[33, 248, 137, 103, 124, 233, 194, 56, 75, 210, 32, 27]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[176, 72, 221, 152, 12, 70, 229, 51, 39, 121, 185, 0]</td>\n",
+       "        <td>[106, 202, 9, 238, 104, 256, 55, 255, 78, 0, 42, 137]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[249, 207, 131, 7, 90, 164, 255, 228, 11, 123, 205, 205]</td>\n",
+       "        <td>[1, 35, 139, 64, 121, 185, 250, 139, 87, 248, 250, 100]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[25, 160, 211, 51, 67, 131, 123, 33, 28, 135, 102, 1]</td>\n",
+       "        <td>[81, 59, 17, 29, 116, 124, 231, 125, 105, 79, 124, 160]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[202, 160, 119, 83, 161, 120, 118, 44, 183, 239, 230, 177]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[61, 169, 117, 160, 136, 197, 220, 153, 226, 79, 21, 201]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[142, 122, 115, 142, 154, 108, 93, 29, 115, 184, 193, 114]</td>\n",
+       "        <td>[126, 23, 73, 30, 100, 19, 191, 219, 102, 96, 83, 220]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[204, 237, 105, 153, 161, 129, 57, 116, 181, 124, 247, 47]</td>\n",
+       "        <td>[10, 203, 113, 187, 70, 174, 99, 186, 78, 235, 128, 42]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[98, 122, 154, 42, 70, 24, 66, 143, 54, 166, 161, 245]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[7, 84, 211, 227, 224, 221, 174, 82, 152, 244, 255, 251]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[78, 230, 46, 120, 106, 144, 241, 4, 186, 55, 28, 252]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[82, 162, 103, 71, 35, 110, 156, 246, 81, 124, 211, 255]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[106, 243, 205, 101, 161, 26, 75, 207, 146, 181, 94, 132]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[24, 187, 213, 20, 129, 39, 182, 232, 110, 217, 86, 10]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[168, 134, 161, 167, 83, 12, 154, 32, 113, 58, 58, 188]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[205, 113, 103, 80, 42, 128, 11, 255, 148, 140, 39, 74]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[149, 34, 203, 159, 241, 114, 37, 146, 25, 120, 158, 179]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 237, 210, 202, 246, 159, 59, 94, 239, 101, 221, 250]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[113, 134, 139, 187, 250, 32, 222, 197, 192, 206, 55, 229]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[81, 93, 255, 4, 244, 13, 241, 198, 215, 231, 101, 18]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[84, 120, 34, 78, 220, 147, 212, 103, 79, 206, 136, 44]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[71, 251, 203, 44, 91, 28, 136, 90, 31, 124, 103, 16]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[62, 248, 167, 81, 60, 251, 200, 95, 72, 164, 242, 28]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[65, 235, 147, 109, 126, 219, 103, 73, 6, 195, 101, 143]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[([19, 126, 250, 219, 119, 255, 86, 152, 200, 36, 57, 188], u'cat'),\n",
-       " ([49, 201, 114, 38, 201, 8, 101, 172, 88, 233, 82, 78], u'dog'),\n",
-       " ([203, 196, 132, 57, 220, 151, 183, 214, 113, 46, 213, 200], u'bird'),\n",
-       " ([157, 236, 255, 90, 38, 48, 35, 152, 86, 236, 160, 187], u'dog'),\n",
-       " ([248, 164, 234, 70, 61, 181, 10, 193, 238, 229, 88, 165], u'bird'),\n",
-       " ([201, 210, 145, 145, 152, 46, 125, 151, 135, 163, 199, 170], u'cat'),\n",
-       " ([29, 150, 219, 216, 46, 211, 124, 24, 25, 186, 205, 35], u'dog'),\n",
-       " ([187, 8, 211, 95, 196, 156, 50, 84, 45, 202, 130, 170], u'dog'),\n",
-       " ([9, 77, 40, 179, 136, 69, 74, 98, 29, 120, 53, 153], u'dog'),\n",
-       " ([78, 83, 93, 113, 206, 23, 121, 160, 119, 61, 60, 168], u'dog'),\n",
-       " ([105, 114, 19, 19, 211, 28, 96, 251, 208, 232, 64, 25], u'cat'),\n",
-       " ([93, 145, 128, 246, 33, 206, 73, 126, 63, 22, 150, 184], u'bird'),\n",
-       " ([12, 245, 243, 181, 134, 92, 39, 153, 112, 250, 181, 208], u'bird'),\n",
-       " ([133, 184, 53, 158, 3, 145, 47, 130, 135, 81, 80, 208], u'bird'),\n",
-       " ([143, 230, 101, 71, 156, 113, 61, 143, 37, 195, 235, 76], u'dog'),\n",
-       " ([91, 70, 17, 43, 59, 150, 227, 111, 53, 229, 0, 100], u'dog'),\n",
-       " ([136, 181, 184, 87, 132, 71, 61, 232, 143, 218, 89, 203], u'dog'),\n",
-       " ([126, 142, 84, 203, 234, 175, 17, 251, 217, 75, 145, 188], u'bird'),\n",
-       " ([198, 162, 187, 42, 9, 67, 223, 193, 154, 99, 9, 215], u'cat'),\n",
-       " ([151, 177, 164, 98, 25, 35, 240, 109, 237, 218, 28, 254], u'bird'),\n",
-       " ([246, 73, 102, 178, 4, 45, 84, 191, 87, 93, 2, 54], u'cat'),\n",
-       " ([156, 153, 39, 115, 228, 190, 35, 136, 32, 61, 171, 16], u'dog'),\n",
-       " ([152, 234, 198, 149, 191, 188, 222, 37, 110, 226, 82, 194], u'dog'),\n",
-       " ([169, 31, 163, 222, 61, 62, 119, 100, 177, 91, 34, 213], u'bird'),\n",
-       " ([67, 17, 141, 83, 188, 37, 61, 130, 187, 252, 62, 153], u'cat'),\n",
-       " ([172, 123, 115, 110, 28, 28, 140, 191, 250, 202, 253, 113], u'cat'),\n",
-       " ([225, 113, 99, 228, 109, 158, 250, 245, 47, 79, 52, 1], u'dog'),\n",
-       " ([137, 50, 48, 110, 202, 76, 211, 142, 78, 174, 232, 206], u'dog'),\n",
-       " ([166, 168, 219, 125, 201, 188, 238, 44, 160, 92, 202, 153], u'cat'),\n",
-       " ([249, 233, 133, 249, 100, 14, 43, 147, 124, 246, 223, 78], u'dog'),\n",
-       " ([45, 253, 108, 251, 135, 18, 163, 98, 143, 108, 30, 126], u'dog'),\n",
-       " ([190, 217, 97, 87, 41, 90, 64, 174, 84, 164, 188, 127], u'cat'),\n",
-       " ([56, 117, 22, 134, 249, 67, 130, 101, 62, 9, 119, 225], u'dog'),\n",
-       " ([6, 78, 138, 132, 230, 72, 93, 71, 159, 134, 161, 223], u'cat'),\n",
-       " ([245, 131, 240, 116, 186, 40, 233, 209, 174, 226, 20, 48], u'cat'),\n",
-       " ([82, 57, 189, 52, 165, 195, 129, 46, 71, 103, 118, 163], u'bird'),\n",
-       " ([21, 41, 79, 244, 93, 68, 120, 78, 184, 50, 117, 161], u'cat'),\n",
-       " ([35, 131, 23, 83, 201, 105, 140, 134, 157, 48, 73, 30], u'dog'),\n",
-       " ([144, 133, 213, 51, 51, 234, 93, 130, 222, 186, 198, 86], u'cat'),\n",
-       " ([126, 136, 125, 31, 139, 160, 161, 162, 242, 106, 11, 126], u'bird'),\n",
-       " ([168, 174, 58, 198, 13, 202, 75, 226, 254, 126, 204, 90], u'bird'),\n",
-       " ([170, 20, 197, 1, 28, 67, 137, 153, 97, 20, 57, 3], u'bird'),\n",
-       " ([43, 109, 193, 169, 94, 105, 88, 152, 46, 101, 98, 121], u'cat'),\n",
-       " ([95, 247, 19, 186, 247, 189, 206, 188, 190, 234, 254, 70], u'dog'),\n",
-       " ([96, 90, 188, 98, 16, 231, 207, 209, 145, 45, 58, 232], u'bird'),\n",
-       " ([104, 77, 39, 226, 148, 134, 217, 166, 64, 207, 99, 14], u'dog'),\n",
-       " ([33, 248, 137, 103, 124, 233, 194, 56, 75, 210, 32, 27], u'dog'),\n",
-       " ([176, 72, 221, 152, 12, 70, 229, 51, 39, 121, 185, 0], u'cat'),\n",
-       " ([249, 207, 131, 7, 90, 164, 255, 228, 11, 123, 205, 205], u'bird'),\n",
-       " ([25, 160, 211, 51, 67, 131, 123, 33, 28, 135, 102, 1], u'bird'),\n",
-       " ([142, 122, 115, 142, 154, 108, 93, 29, 115, 184, 193, 114], u'dog'),\n",
-       " ([204, 237, 105, 153, 161, 129, 57, 116, 181, 124, 247, 47], u'dog')]"
+       "[([168, 228, 110, 3, 51, 104, 192, 23, 120, 249, 96, 99], u'dog'),\n",
+       " ([20, 145, 109, 135, 149, 100, 39, 66, 124, 102, 77, 140], u'dog'),\n",
+       " ([125, 32, 244, 23, 201, 156, 251, 55, 159, 47, 160, 95], u'cat'),\n",
+       " ([24, 88, 166, 123, 193, 186, 12, 46, 65, 161, 145, 104], u'bird'),\n",
+       " ([14, 206, 47, 154, 85, 172, 186, 73, 196, 131, 229, 191], u'bird'),\n",
+       " ([131, 238, 90, 227, 51, 114, 59, 217, 237, 252, 147, 248], u'cat'),\n",
+       " ([211, 153, 187, 59, 123, 200, 10, 171, 98, 95, 87, 28], u'dog'),\n",
+       " ([26, 159, 140, 217, 89, 15, 199, 179, 242, 250, 37, 45], u'bird'),\n",
+       " ([18, 41, 102, 10, 82, 57, 163, 13, 116, 30, 213, 126], u'bird'),\n",
+       " ([56, 221, 31, 84, 132, 58, 243, 16, 19, 76, 31, 218], u'bird'),\n",
+       " ([17, 212, 36, 62, 167, 54, 103, 13, 64, 185, 70, 227], u'bird'),\n",
+       " ([186, 1, 155, 56, 201, 211, 21, 233, 38, 153, 34, 25], u'dog'),\n",
+       " ([53, 101, 200, 15, 101, 217, 227, 137, 23, 138, 191, 126], u'dog'),\n",
+       " ([255, 54, 220, 226, 252, 150, 227, 151, 207, 172, 105, 227], u'dog'),\n",
+       " ([144, 124, 183, 169, 37, 237, 14, 237, 252, 115, 198, 222], u'bird'),\n",
+       " ([222, 104, 188, 92, 254, 187, 146, 219, 157, 142, 113, 128], u'cat'),\n",
+       " ([64, 44, 142, 35, 193, 30, 159, 120, 199, 196, 101, 213], u'bird'),\n",
+       " ([96, 72, 120, 63, 69, 86, 167, 0, 177, 165, 187, 67], u'dog'),\n",
+       " ([88, 210, 241, 216, 246, 48, 4, 132, 83, 197, 162, 242], u'cat'),\n",
+       " ([105, 182, 162, 62, 104, 2, 134, 223, 65, 203, 53, 231], u'bird'),\n",
+       " ([230, 140, 134, 42, 12, 223, 251, 252, 183, 241, 44, 188], u'dog'),\n",
+       " ([127, 129, 24, 113, 190, 129, 40, 96, 191, 143, 98, 69], u'dog'),\n",
+       " ([162, 16, 163, 137, 219, 137, 21, 97, 179, 33, 64, 174], u'cat'),\n",
+       " ([247, 159, 74, 179, 21, 201, 51, 45, 58, 241, 175, 98], u'cat'),\n",
+       " ([110, 241, 179, 179, 96, 85, 195, 3, 222, 158, 140, 244], u'bird'),\n",
+       " ([63, 21, 63, 237, 50, 54, 140, 124, 233, 162, 69, 28], u'bird'),\n",
+       " ([94, 111, 234, 231, 203, 73, 118, 97, 57, 254, 209, 131], u'dog'),\n",
+       " ([246, 73, 151, 78, 201, 43, 59, 1, 215, 155, 138, 63], u'dog'),\n",
+       " ([46, 186, 18, 158, 254, 111, 13, 232, 86, 216, 49, 204], u'cat'),\n",
+       " ([106, 202, 9, 238, 104, 256, 55, 255, 78, 0, 42, 137], u'cat'),\n",
+       " ([1, 35, 139, 64, 121, 185, 250, 139, 87, 248, 250, 100], u'bird'),\n",
+       " ([81, 59, 17, 29, 116, 124, 231, 125, 105, 79, 124, 160], u'cat'),\n",
+       " ([202, 160, 119, 83, 161, 120, 118, 44, 183, 239, 230, 177], u'dog'),\n",
+       " ([61, 169, 117, 160, 136, 197, 220, 153, 226, 79, 21, 201], u'bird'),\n",
+       " ([126, 23, 73, 30, 100, 19, 191, 219, 102, 96, 83, 220], u'dog'),\n",
+       " ([10, 203, 113, 187, 70, 174, 99, 186, 78, 235, 128, 42], u'dog'),\n",
+       " ([98, 122, 154, 42, 70, 24, 66, 143, 54, 166, 161, 245], u'dog'),\n",
+       " ([7, 84, 211, 227, 224, 221, 174, 82, 152, 244, 255, 251], u'bird'),\n",
+       " ([78, 230, 46, 120, 106, 144, 241, 4, 186, 55, 28, 252], u'bird'),\n",
+       " ([82, 162, 103, 71, 35, 110, 156, 246, 81, 124, 211, 255], u'bird'),\n",
+       " ([106, 243, 205, 101, 161, 26, 75, 207, 146, 181, 94, 132], u'bird'),\n",
+       " ([24, 187, 213, 20, 129, 39, 182, 232, 110, 217, 86, 10], u'bird'),\n",
+       " ([168, 134, 161, 167, 83, 12, 154, 32, 113, 58, 58, 188], u'cat'),\n",
+       " ([205, 113, 103, 80, 42, 128, 11, 255, 148, 140, 39, 74], u'dog'),\n",
+       " ([149, 34, 203, 159, 241, 114, 37, 146, 25, 120, 158, 179], u'dog'),\n",
+       " ([15, 237, 210, 202, 246, 159, 59, 94, 239, 101, 221, 250], u'dog'),\n",
+       " ([113, 134, 139, 187, 250, 32, 222, 197, 192, 206, 55, 229], u'dog'),\n",
+       " ([81, 93, 255, 4, 244, 13, 241, 198, 215, 231, 101, 18], u'cat'),\n",
+       " ([84, 120, 34, 78, 220, 147, 212, 103, 79, 206, 136, 44], u'dog'),\n",
+       " ([71, 251, 203, 44, 91, 28, 136, 90, 31, 124, 103, 16], u'cat'),\n",
+       " ([62, 248, 167, 81, 60, 251, 200, 95, 72, 164, 242, 28], u'cat'),\n",
+       " ([65, 235, 147, 109, 126, 219, 103, 73, 6, 195, 101, 143], u'cat')]"
       ]
      },
-     "execution_count": 9,
+     "execution_count": 12,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1058,7 +1035,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 10,
+   "execution_count": 13,
    "metadata": {},
    "outputs": [
     {
@@ -1075,8 +1052,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1095,7 +1072,7 @@
        "[([26, 12], [26, 3], 0), ([26, 12], [26, 3], 1)]"
       ]
      },
-     "execution_count": 10,
+     "execution_count": 13,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1112,7 +1089,7 @@
     "                                        255                   -- Normalizing constant\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -1127,7 +1104,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 11,
+   "execution_count": 14,
    "metadata": {},
    "outputs": [
     {
@@ -1144,8 +1121,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1164,7 +1141,7 @@
        "[([26, 12], [26, 3], 0), ([26, 12], [26, 3], 1)]"
       ]
      },
-     "execution_count": 11,
+     "execution_count": 14,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1182,7 +1159,7 @@
     "    NULL                      -- Buffer size\n",
     "    );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -1197,7 +1174,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 12,
+   "execution_count": 15,
    "metadata": {},
    "outputs": [
     {
@@ -1214,13 +1191,13 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[8, 12]</td>\n",
-       "        <td>[8, 3]</td>\n",
+       "        <td>[9, 12]</td>\n",
+       "        <td>[9, 3]</td>\n",
        "        <td>0</td>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1244,22 +1221,22 @@
        "        <td>4</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[8, 12]</td>\n",
-       "        <td>[8, 3]</td>\n",
+       "        <td>[7, 12]</td>\n",
+       "        <td>[7, 3]</td>\n",
        "        <td>5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[([8, 12], [8, 3], 0),\n",
+       "[([9, 12], [9, 3], 0),\n",
        " ([9, 12], [9, 3], 1),\n",
        " ([9, 12], [9, 3], 2),\n",
        " ([9, 12], [9, 3], 3),\n",
        " ([9, 12], [9, 3], 4),\n",
-       " ([8, 12], [8, 3], 5)]"
+       " ([7, 12], [7, 3], 5)]"
       ]
      },
-     "execution_count": 12,
+     "execution_count": 15,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1276,7 +1253,7 @@
     "                                        255                   -- Normalizing constant\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -1288,7 +1265,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 13,
+   "execution_count": 16,
    "metadata": {},
    "outputs": [
     {
@@ -1308,7 +1285,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -1318,23 +1295,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
-       "        <td>10</td>\n",
+       "        <td>9</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 10, 255.0, 3, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 9, 255.0, [3], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 13,
+     "execution_count": 16,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1356,7 +1333,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 14,
+   "execution_count": 18,
    "metadata": {},
    "outputs": [
     {
@@ -1373,8 +1350,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1393,7 +1370,7 @@
        "[([26, 12], [26, 5], 0), ([26, 12], [26, 5], 1)]"
       ]
      },
-     "execution_count": 14,
+     "execution_count": 18,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1408,15 +1385,15 @@
     "                                        'rgb',                -- Independent variable\n",
     "                                        NULL,                 -- Buffer size\n",
     "                                        255,                  -- Normalizing constant\n",
-    "                                        5                     -- Number of desired class values\n",
+    "                                        ARRAY[5]              -- Number of desired class values\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 15,
+   "execution_count": 19,
    "metadata": {},
    "outputs": [
     {
@@ -1436,7 +1413,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -1446,23 +1423,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog', None, None]</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>5</td>\n",
+       "        <td>[5]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog', None, None], 26, 255.0, 5, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog', None, None], 26, 255.0, [5], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 15,
+     "execution_count": 19,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1513,7 +1490,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 17,
+   "execution_count": 20,
    "metadata": {},
    "outputs": [
     {
@@ -1538,7 +1515,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -1548,23 +1525,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>[2, 3]</td>\n",
        "        <td>[0, 1]</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 26, 255.0, 3, [2, 3], [0, 1])]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 26, 255.0, [3], [2, 3], [0, 1])]"
       ]
      },
-     "execution_count": 17,
+     "execution_count": 20,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1610,7 +1587,7 @@
    "name": "python",
    "nbconvert_exporter": "python",
    "pygments_lexer": "ipython2",
-   "version": "2.7.10"
+   "version": "2.7.16"
   }
  },
  "nbformat": 4,
diff --git a/community-artifacts/Deep-learning/Load-model-selection-table-v1.ipynb b/community-artifacts/Deep-learning/Load-model-selection-table-v1.ipynb
deleted file mode 100644
index 778d988..0000000
--- a/community-artifacts/Deep-learning/Load-model-selection-table-v1.ipynb
+++ /dev/null
@@ -1,955 +0,0 @@
-{
- "cells": [
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "# Load model selection table\n",
-    "This utility function generates model selection tuples (model architecture, compile parameters, fit parameters) for both hyper-parameter search and model architecture search.  The model selection table and associated summary table are used by the multiple model fit feature of MADlib.  \n",
-    "\n",
-    "This utility was added in MADlib 1.17.\n",
-    "\n",
-    "## Table of contents\n",
-    "\n",
-    "<a href=\"#define_model_arch\">1. Define model architecture table</a>\n",
-    "\n",
-    "<a href=\"#load_model_arch\">2. Load model architecture</a>\n",
-    "\n",
-    "<a href=\"#load_model_selection\">3. Load model selection table</a>\n",
-    "\n",
-    "<a href=\"#load_model_selection_manual\">4. Create model selection table manually</a>\n",
-    "\n",
-    "<a href=\"#load_model_selection_auto\">5. Generate hyperparameters automatically</a>"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 1,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
-   "source": [
-    "%load_ext sql"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 2,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
-    "# Greenplum Database 5.x on GCP - via tunnel\n",
-    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
-    "        \n",
-    "# PostgreSQL local\n",
-    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 17,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>version</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
-      ]
-     },
-     "execution_count": 17,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql select madlib.version();\n",
-    "#%sql select version();"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"define_model_arch\"></a>\n",
-    "# 1. Define model architecture table\n",
-    "The model selection loader works in conjunction with the model architecture table, so we first create a model architecture table with two different models.  See http://madlib.apache.org/docs/latest/group__grp__keras__model__arch.html for more details on the model architecture table.\n",
-    "\n",
-    "Import Keras libraries"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 18,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "import keras\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Define model architecture with 1 hidden layer:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 19,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "dense_8 (Dense)              (None, 10)                50        \n",
-      "_________________________________________________________________\n",
-      "dense_9 (Dense)              (None, 10)                110       \n",
-      "_________________________________________________________________\n",
-      "dense_10 (Dense)             (None, 3)                 33        \n",
-      "=================================================================\n",
-      "Total params: 193\n",
-      "Trainable params: 193\n",
-      "Non-trainable params: 0\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "model1 = Sequential()\n",
-    "model1.add(Dense(10, activation='relu', input_shape=(4,)))\n",
-    "model1.add(Dense(10, activation='relu'))\n",
-    "model1.add(Dense(3, activation='softmax'))\n",
-    "    \n",
-    "model1.summary()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 20,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_8\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_9\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_10\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
-      ]
-     },
-     "execution_count": 20,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "model1.to_json()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 2, \"batch_input_shape\": [null, 3], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"new_dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'\n",
-    "        "
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Define model architecture with 2 hidden layers:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 21,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "dense_11 (Dense)             (None, 10)                50        \n",
-      "_________________________________________________________________\n",
-      "dense_12 (Dense)             (None, 10)                110       \n",
-      "_________________________________________________________________\n",
-      "dense_13 (Dense)             (None, 10)                110       \n",
-      "_________________________________________________________________\n",
-      "dense_14 (Dense)             (None, 3)                 33        \n",
-      "=================================================================\n",
-      "Total params: 303\n",
-      "Trainable params: 303\n",
-      "Non-trainable params: 0\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "model2 = Sequential()\n",
-    "model2.add(Dense(10, activation='relu', input_shape=(4,)))\n",
-    "model2.add(Dense(10, activation='relu'))\n",
-    "model2.add(Dense(10, activation='relu'))\n",
-    "model2.add(Dense(3, activation='softmax'))\n",
-    "    \n",
-    "model2.summary()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 22,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_11\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_12\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_13\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_14\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
-      ]
-     },
-     "execution_count": 22,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "model2.to_json()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load_model_arch\"></a>\n",
-    "# 2. Load model architecture\n",
-    "\n",
-    "Load both into model architecture table:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 23,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "2 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_id</th>\n",
-       "        <th>model_arch</th>\n",
-       "        <th>model_weights</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>__internal_madlib_id__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
-       "        <td>None</td>\n",
-       "        <td>Sophie</td>\n",
-       "        <td>MLP with 1 hidden layer</td>\n",
-       "        <td>__madlib_temp_80732521_1576707528_41018934__</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
-       "        <td>None</td>\n",
-       "        <td>Maria</td>\n",
-       "        <td>MLP with 2 hidden layers</td>\n",
-       "        <td>__madlib_temp_54900499_1576707528_11190984__</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'MLP with 1 hidden layer', u'__madlib_temp_80732521_1576707528_41018934__'),\n",
-       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'MLP with 2 hidden layers', u'__madlib_temp_54900499_1576707528_11190984__')]"
-      ]
-     },
-     "execution_count": 23,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS model_arch_library;\n",
-    "\n",
-    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
-    "                               \n",
-    "$$\n",
-    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
-    "$$\n",
-    "::json,         -- JSON blob\n",
-    "                               NULL,                  -- Weights\n",
-    "                               'Sophie',              -- Name\n",
-    "                               'MLP with 1 hidden layer'       -- Descr\n",
-    ");\n",
-    "\n",
-    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
-    "                               \n",
-    "$$\n",
-    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
-    "$$\n",
-    "::json,         -- JSON blob\n",
-    "                               NULL,                  -- Weights\n",
-    "                               'Maria',               -- Name\n",
-    "                               'MLP with 2 hidden layers'       -- Descr\n",
-    ");\n",
-    "\n",
-    "SELECT * FROM model_arch_library ORDER BY model_id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load_model_selection\"></a>\n",
-    "# 3.  Load model selection table\n",
-    "\n",
-    "Select the model(s) from the model architecture table that you want to run, along with the compile and fit parameters.  Unique combinations will be created for the set of model selection parameters:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 24,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "12 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>mst_key</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>11</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>12</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (3, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (4, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (5, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (6, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (7, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (8, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (9, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (10, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (11, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (12, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1')]"
-      ]
-     },
-     "execution_count": 24,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
-    "\n",
-    "SELECT madlib.load_model_selection_table('model_arch_library', -- model architecture table\n",
-    "                                         'mst_table',          -- model selection table output\n",
-    "                                          ARRAY[1,2],              -- model ids from model architecture table\n",
-    "                                          ARRAY[                   -- compile params\n",
-    "                                              $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$,\n",
-    "                                              $$loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']$$,\n",
-    "                                              $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$\n",
-    "                                          ],\n",
-    "                                          ARRAY[                    -- fit params\n",
-    "                                              $$batch_size=4,epochs=1$$,\n",
-    "                                              $$batch_size=8,epochs=1$$\n",
-    "                                          ]\n",
-    "                                         );\n",
-    "                                  \n",
-    "SELECT * FROM mst_table ORDER BY mst_key;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "The name of the model architecture table is stored in the summary table:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 25,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_arch_table</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>model_arch_library</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'model_arch_library',)]"
-      ]
-     },
-     "execution_count": 25,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM mst_table_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load_model_selection_manual\"></a>\n",
-    "# 4.  Create model selection table manually\n",
-    "\n",
-    "If you would like to have more control over the set of model selection parameters to run, you can manually create the model selection table and the associated summary table.  Both must be created since they are needed by the multiple model fit module.\n",
-    "\n",
-    "For example, let's say we don't want all combinations but only want batch_size=4 for model_id=1 and batch_size=8 for model_id=2:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 26,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "Done.\n",
-      "6 rows affected.\n",
-      "6 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>mst_key</th>\n",
-       "        <th>model_arch_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (3, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (4, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (5, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (6, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1')]"
-      ]
-     },
-     "execution_count": 26,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS mst_table_manual;\n",
-    "\n",
-    "CREATE TABLE mst_table_manual(\n",
-    "    mst_key serial,\n",
-    "    model_arch_id integer,\n",
-    "    compile_params varchar,\n",
-    "    fit_params varchar\n",
-    ");\n",
-    "\n",
-    "INSERT INTO mst_table_manual(model_arch_id, compile_params, fit_params) VALUES\n",
-    "(1, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$, 'batch_size=4,epochs=1'),\n",
-    "(1, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']$$, 'batch_size=4,epochs=1'),\n",
-    "(1, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$, 'batch_size=4,epochs=1'),\n",
-    "(2, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$, 'batch_size=8,epochs=1'),\n",
-    "(2, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']$$, 'batch_size=8,epochs=1'),\n",
-    "(2, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$, 'batch_size=8,epochs=1');\n",
-    "\n",
-    "SELECT * FROM mst_table_manual ORDER BY mst_key; "
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Create the summary table which must be named with the model selection output table appended by \"_summary\":"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 27,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_arch_table</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>model_arch_library</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'model_arch_library',)]"
-      ]
-     },
-     "execution_count": 27,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS mst_table_manual_summary;\n",
-    "\n",
-    "CREATE TABLE mst_table_manual_summary (\n",
-    "    model_arch_table varchar\n",
-    ");\n",
-    "\n",
-    "INSERT INTO mst_table_manual_summary(model_arch_table) VALUES\n",
-    "('model_arch_library');\n",
-    "\n",
-    "SELECT * FROM mst_table_manual_summary; "
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load_model_selection_auto\"></a>\n",
-    "# 5. Generate hyperparameters automatically\n",
-    "\n",
-    "You can use other libraries or methods to generate hyperparameters according to the tests that you want to run.  For example, let's randomly generate batch size from powers of 2 and learning rate on a log scale.\n",
-    "\n",
-    "We use psycopg which is a PostgreSQL database adapter for the Python programming language."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 28,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "12 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>mst_key</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=32,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=64,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=32,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=64,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=32,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=64,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=32,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=64,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=32,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=64,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>11</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=32,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>12</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=64,epochs=1</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=32,epochs=1'),\n",
-       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=64,epochs=1'),\n",
-       " (3, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=32,epochs=1'),\n",
-       " (4, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=64,epochs=1'),\n",
-       " (5, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=32,epochs=1'),\n",
-       " (6, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=64,epochs=1'),\n",
-       " (7, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=32,epochs=1'),\n",
-       " (8, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=64,epochs=1'),\n",
-       " (9, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=32,epochs=1'),\n",
-       " (10, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=64,epochs=1'),\n",
-       " (11, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=32,epochs=1'),\n",
-       " (12, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=64,epochs=1')]"
-      ]
-     },
-     "execution_count": 28,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "import numpy as np\n",
-    "import psycopg2 as p2\n",
-    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
-    "#conn = p2.connect('postgresql://fmcquillan@localhost:5432/madlib')\n",
-    "cur = conn.cursor()\n",
-    "\n",
-    "%sql DROP TABLE IF EXISTS mst_table_auto, mst_table_auto_summary;\n",
-    "\n",
-    "#compile params\n",
-    "learning_rate = np.random.permutation([0.1,0.01,0.001,0.0001])[:3]\n",
-    "compile_param1 = \"loss='categorical_crossentropy',optimizer='Adam(lr=\" + str(learning_rate[0]) + \")',metrics=['accuracy']\"\n",
-    "compile_param2 = \"loss='categorical_crossentropy',optimizer='Adam(lr=\" + str(learning_rate[1]) + \")',metrics=['accuracy']\"\n",
-    "compile_param3 = \"loss='categorical_crossentropy',optimizer='Adam(lr=\" + str(learning_rate[2]) + \")',metrics=['accuracy']\"\n",
-    "compile_params = [compile_param1,compile_param2,compile_param3]\n",
-    "\n",
-    "#fit params\n",
-    "batch_size = np.random.permutation([4,8,16,32,64])[:2]\n",
-    "fit_param1 = \"batch_size=\" + str(batch_size[0]) + \",epochs=1\"\n",
-    "fit_param2 = \"batch_size=\" + str(batch_size[1]) + \",epochs=1\"\n",
-    "fit_params = [fit_param1,fit_param2]\n",
-    "\n",
-    "query = \"SELECT madlib.load_model_selection_table('model_arch_library', 'mst_table_auto', ARRAY[1,2], %s, %s);\"\n",
-    "\n",
-    "cur.execute(query,[compile_params, fit_params])\n",
-    "conn.commit()\n",
-    "\n",
-    "#review model selection table\n",
-    "%sql SELECT * FROM mst_table_auto ORDER BY mst_key;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "The name of the model architecture table is stored in the summary table:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 29,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_arch_table</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>model_arch_library</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'model_arch_library',)]"
-      ]
-     },
-     "execution_count": 29,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM mst_table_auto_summary;"
-   ]
-  }
- ],
- "metadata": {
-  "kernelspec": {
-   "display_name": "Python 2",
-   "language": "python",
-   "name": "python2"
-  },
-  "language_info": {
-   "codemirror_mode": {
-    "name": "ipython",
-    "version": 2
-   },
-   "file_extension": ".py",
-   "mimetype": "text/x-python",
-   "name": "python",
-   "nbconvert_exporter": "python",
-   "pygments_lexer": "ipython2",
-   "version": "2.7.10"
-  }
- },
- "nbformat": 4,
- "nbformat_minor": 1
-}
diff --git a/community-artifacts/Deep-learning/MADlib-Keras-MLP-v2.ipynb b/community-artifacts/Deep-learning/MADlib-Keras-MLP-v2.ipynb
deleted file mode 100644
index c86fb75..0000000
--- a/community-artifacts/Deep-learning/MADlib-Keras-MLP-v2.ipynb
+++ /dev/null
@@ -1,4057 +0,0 @@
-{
- "cells": [
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "# Multilayer Perceptron Using Keras and MADlib\n",
-    "\n",
-    "E2E classification example using MADlib calling a Keras MLP.\n",
-    "\n",
-    "Deep learning works best on very large datasets, but that is not convenient for a quick introduction to the syntax.  So in this workbook we use the well known iris data set from https://archive.ics.uci.edu/ml/datasets/iris to help get you started.  It is similar to the example in user docs http://madlib.apache.org/docs/latest/index.html\n",
-    "\n",
-    "For more realistic examples with images please refer to the deep learning notebooks at\n",
-    "https://github.com/apache/madlib-site/tree/asf-site/community-artifacts\n",
-    "\n",
-    "## Table of contents\n",
-    "\n",
-    "<a href=\"#class\">Classification</a>\n",
-    "\n",
-    "* <a href=\"#create_input_data\">1. Create input data</a>\n",
-    "\n",
-    "* <a href=\"#pp\">2. Call preprocessor for deep learning</a>\n",
-    "\n",
-    "* <a href=\"#load\">3. Define and load model architecture</a>\n",
-    "\n",
-    "* <a href=\"#train\">4. Train</a>\n",
-    "\n",
-    "* <a href=\"#eval\">5. Evaluate</a>\n",
-    "\n",
-    "* <a href=\"#pred\">6. Predict</a>\n",
-    "\n",
-    "* <a href=\"#pred_byom\">7. Predict BYOM</a>\n",
-    "\n",
-    "<a href=\"#class2\">Classification with Other Parameters</a>\n",
-    "\n",
-    "* <a href=\"#val_dataset\">1. Validation dataset</a>\n",
-    "\n",
-    "* <a href=\"#pred_prob\">2. Predict probabilities</a>\n",
-    "\n",
-    "* <a href=\"#warm_start\">3. Warm start</a>\n",
-    "\n",
-    "<a href=\"#transfer_learn\">Transfer learning</a>\n",
-    "\n",
-    "* <a href=\"#load2\">1. Define and load model architecture with some layers frozen</a>\n",
-    "\n",
-    "* <a href=\"#train2\">2. Train transfer model</a>"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 1,
-   "metadata": {
-    "scrolled": false
-   },
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
-   "source": [
-    "%load_ext sql"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 2,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
-    "# Greenplum Database 5.x on GCP - via tunnel\n",
-    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
-    "        \n",
-    "# PostgreSQL local\n",
-    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 3,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>version</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
-      ]
-     },
-     "execution_count": 3,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql select madlib.version();\n",
-    "#%sql select version();"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"class\"></a>\n",
-    "# Classification\n",
-    "\n",
-    "<a id=\"create_input_data\"></a>\n",
-    "# 1.  Create input data\n",
-    "\n",
-    "Load iris data set."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 4,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "Done.\n",
-      "150 rows affected.\n",
-      "150 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>attributes</th>\n",
-       "        <th>class_text</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>[Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>[Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>[Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>11</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>12</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>13</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>14</td>\n",
-       "        <td>[Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>15</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>16</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>17</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>18</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>19</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>20</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>21</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>22</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>23</td>\n",
-       "        <td>[Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>24</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>25</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>26</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>27</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>28</td>\n",
-       "        <td>[Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>29</td>\n",
-       "        <td>[Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>30</td>\n",
-       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>31</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>32</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>33</td>\n",
-       "        <td>[Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>34</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>35</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>36</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>37</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>38</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>39</td>\n",
-       "        <td>[Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>40</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>41</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>42</td>\n",
-       "        <td>[Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>43</td>\n",
-       "        <td>[Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>44</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>45</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>46</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>47</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>48</td>\n",
-       "        <td>[Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>49</td>\n",
-       "        <td>[Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>50</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>51</td>\n",
-       "        <td>[Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>52</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>53</td>\n",
-       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>54</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>55</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>56</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>57</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>58</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>59</td>\n",
-       "        <td>[Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>60</td>\n",
-       "        <td>[Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>61</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>62</td>\n",
-       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>63</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>64</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>65</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>66</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>67</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>68</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>69</td>\n",
-       "        <td>[Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>70</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>71</td>\n",
-       "        <td>[Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>72</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>73</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>74</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>75</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>76</td>\n",
-       "        <td>[Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>77</td>\n",
-       "        <td>[Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>78</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>79</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>80</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>81</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>82</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>83</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>84</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>85</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>86</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>87</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>88</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>89</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>90</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>91</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>92</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>93</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>94</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>95</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>96</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>97</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>98</td>\n",
-       "        <td>[Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>99</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>100</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>101</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>102</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>103</td>\n",
-       "        <td>[Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>104</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>105</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>106</td>\n",
-       "        <td>[Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>107</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>108</td>\n",
-       "        <td>[Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>109</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>110</td>\n",
-       "        <td>[Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>111</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>112</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>113</td>\n",
-       "        <td>[Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>114</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>115</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>116</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>117</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>118</td>\n",
-       "        <td>[Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>119</td>\n",
-       "        <td>[Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>120</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>121</td>\n",
-       "        <td>[Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>122</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>123</td>\n",
-       "        <td>[Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>124</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>125</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>126</td>\n",
-       "        <td>[Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>127</td>\n",
-       "        <td>[Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>128</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>129</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>130</td>\n",
-       "        <td>[Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>131</td>\n",
-       "        <td>[Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>132</td>\n",
-       "        <td>[Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>133</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>134</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>135</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>136</td>\n",
-       "        <td>[Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>137</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>138</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>139</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>140</td>\n",
-       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>141</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>142</td>\n",
-       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>143</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>144</td>\n",
-       "        <td>[Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>145</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>146</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>147</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>148</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>149</td>\n",
-       "        <td>[Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>150</td>\n",
-       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (2, [Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (3, [Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (4, [Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (5, [Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (6, [Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (7, [Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (8, [Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (9, [Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (10, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (11, [Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (12, [Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (13, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (14, [Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (15, [Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (16, [Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (17, [Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (18, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (19, [Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (20, [Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (21, [Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (22, [Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (23, [Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (24, [Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')], u'Iris-setosa'),\n",
-       " (25, [Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (26, [Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (27, [Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (28, [Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (29, [Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (30, [Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (31, [Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (32, [Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (33, [Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (34, [Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (35, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (36, [Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (37, [Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (38, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (39, [Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (40, [Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (41, [Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (42, [Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (43, [Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (44, [Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')], u'Iris-setosa'),\n",
-       " (45, [Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (46, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (47, [Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (48, [Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (49, [Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (50, [Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (51, [Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (52, [Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (53, [Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (54, [Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (55, [Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (56, [Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (57, [Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')], u'Iris-versicolor'),\n",
-       " (58, [Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (59, [Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (60, [Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (61, [Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (62, [Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (63, [Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (64, [Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (65, [Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (66, [Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (67, [Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (68, [Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (69, [Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (70, [Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')], u'Iris-versicolor'),\n",
-       " (71, [Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')], u'Iris-versicolor'),\n",
-       " (72, [Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (73, [Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (74, [Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (75, [Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (76, [Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (77, [Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (78, [Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')], u'Iris-versicolor'),\n",
-       " (79, [Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (80, [Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (81, [Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')], u'Iris-versicolor'),\n",
-       " (82, [Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (83, [Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (84, [Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')], u'Iris-versicolor'),\n",
-       " (85, [Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (86, [Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')], u'Iris-versicolor'),\n",
-       " (87, [Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (88, [Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (89, [Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (90, [Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (91, [Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (92, [Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (93, [Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (94, [Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (95, [Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (96, [Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (97, [Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (98, [Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (99, [Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')], u'Iris-versicolor'),\n",
-       " (100, [Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (101, [Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')], u'Iris-virginica'),\n",
-       " (102, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (103, [Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (104, [Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (105, [Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')], u'Iris-virginica'),\n",
-       " (106, [Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (107, [Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')], u'Iris-virginica'),\n",
-       " (108, [Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (109, [Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (110, [Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')], u'Iris-virginica'),\n",
-       " (111, [Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (112, [Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (113, [Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (114, [Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (115, [Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')], u'Iris-virginica'),\n",
-       " (116, [Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (117, [Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (118, [Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')], u'Iris-virginica'),\n",
-       " (119, [Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (120, [Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')], u'Iris-virginica'),\n",
-       " (121, [Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (122, [Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (123, [Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (124, [Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (125, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (126, [Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (127, [Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (128, [Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (129, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (130, [Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')], u'Iris-virginica'),\n",
-       " (131, [Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (132, [Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (133, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')], u'Iris-virginica'),\n",
-       " (134, [Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')], u'Iris-virginica'),\n",
-       " (135, [Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')], u'Iris-virginica'),\n",
-       " (136, [Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (137, [Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
-       " (138, [Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (139, [Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (140, [Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (141, [Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
-       " (142, [Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (143, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (144, [Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (145, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')], u'Iris-virginica'),\n",
-       " (146, [Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (147, [Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (148, [Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (149, [Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (150, [Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')], u'Iris-virginica')]"
-      ]
-     },
-     "execution_count": 4,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql \n",
-    "DROP TABLE IF EXISTS iris_data;\n",
-    "\n",
-    "CREATE TABLE iris_data(\n",
-    "    id serial,\n",
-    "    attributes numeric[],\n",
-    "    class_text varchar\n",
-    ");\n",
-    "\n",
-    "INSERT INTO iris_data(id, attributes, class_text) VALUES\n",
-    "(1,ARRAY[5.1,3.5,1.4,0.2],'Iris-setosa'),\n",
-    "(2,ARRAY[4.9,3.0,1.4,0.2],'Iris-setosa'),\n",
-    "(3,ARRAY[4.7,3.2,1.3,0.2],'Iris-setosa'),\n",
-    "(4,ARRAY[4.6,3.1,1.5,0.2],'Iris-setosa'),\n",
-    "(5,ARRAY[5.0,3.6,1.4,0.2],'Iris-setosa'),\n",
-    "(6,ARRAY[5.4,3.9,1.7,0.4],'Iris-setosa'),\n",
-    "(7,ARRAY[4.6,3.4,1.4,0.3],'Iris-setosa'),\n",
-    "(8,ARRAY[5.0,3.4,1.5,0.2],'Iris-setosa'),\n",
-    "(9,ARRAY[4.4,2.9,1.4,0.2],'Iris-setosa'),\n",
-    "(10,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
-    "(11,ARRAY[5.4,3.7,1.5,0.2],'Iris-setosa'),\n",
-    "(12,ARRAY[4.8,3.4,1.6,0.2],'Iris-setosa'),\n",
-    "(13,ARRAY[4.8,3.0,1.4,0.1],'Iris-setosa'),\n",
-    "(14,ARRAY[4.3,3.0,1.1,0.1],'Iris-setosa'),\n",
-    "(15,ARRAY[5.8,4.0,1.2,0.2],'Iris-setosa'),\n",
-    "(16,ARRAY[5.7,4.4,1.5,0.4],'Iris-setosa'),\n",
-    "(17,ARRAY[5.4,3.9,1.3,0.4],'Iris-setosa'),\n",
-    "(18,ARRAY[5.1,3.5,1.4,0.3],'Iris-setosa'),\n",
-    "(19,ARRAY[5.7,3.8,1.7,0.3],'Iris-setosa'),\n",
-    "(20,ARRAY[5.1,3.8,1.5,0.3],'Iris-setosa'),\n",
-    "(21,ARRAY[5.4,3.4,1.7,0.2],'Iris-setosa'),\n",
-    "(22,ARRAY[5.1,3.7,1.5,0.4],'Iris-setosa'),\n",
-    "(23,ARRAY[4.6,3.6,1.0,0.2],'Iris-setosa'),\n",
-    "(24,ARRAY[5.1,3.3,1.7,0.5],'Iris-setosa'),\n",
-    "(25,ARRAY[4.8,3.4,1.9,0.2],'Iris-setosa'),\n",
-    "(26,ARRAY[5.0,3.0,1.6,0.2],'Iris-setosa'),\n",
-    "(27,ARRAY[5.0,3.4,1.6,0.4],'Iris-setosa'),\n",
-    "(28,ARRAY[5.2,3.5,1.5,0.2],'Iris-setosa'),\n",
-    "(29,ARRAY[5.2,3.4,1.4,0.2],'Iris-setosa'),\n",
-    "(30,ARRAY[4.7,3.2,1.6,0.2],'Iris-setosa'),\n",
-    "(31,ARRAY[4.8,3.1,1.6,0.2],'Iris-setosa'),\n",
-    "(32,ARRAY[5.4,3.4,1.5,0.4],'Iris-setosa'),\n",
-    "(33,ARRAY[5.2,4.1,1.5,0.1],'Iris-setosa'),\n",
-    "(34,ARRAY[5.5,4.2,1.4,0.2],'Iris-setosa'),\n",
-    "(35,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
-    "(36,ARRAY[5.0,3.2,1.2,0.2],'Iris-setosa'),\n",
-    "(37,ARRAY[5.5,3.5,1.3,0.2],'Iris-setosa'),\n",
-    "(38,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
-    "(39,ARRAY[4.4,3.0,1.3,0.2],'Iris-setosa'),\n",
-    "(40,ARRAY[5.1,3.4,1.5,0.2],'Iris-setosa'),\n",
-    "(41,ARRAY[5.0,3.5,1.3,0.3],'Iris-setosa'),\n",
-    "(42,ARRAY[4.5,2.3,1.3,0.3],'Iris-setosa'),\n",
-    "(43,ARRAY[4.4,3.2,1.3,0.2],'Iris-setosa'),\n",
-    "(44,ARRAY[5.0,3.5,1.6,0.6],'Iris-setosa'),\n",
-    "(45,ARRAY[5.1,3.8,1.9,0.4],'Iris-setosa'),\n",
-    "(46,ARRAY[4.8,3.0,1.4,0.3],'Iris-setosa'),\n",
-    "(47,ARRAY[5.1,3.8,1.6,0.2],'Iris-setosa'),\n",
-    "(48,ARRAY[4.6,3.2,1.4,0.2],'Iris-setosa'),\n",
-    "(49,ARRAY[5.3,3.7,1.5,0.2],'Iris-setosa'),\n",
-    "(50,ARRAY[5.0,3.3,1.4,0.2],'Iris-setosa'),\n",
-    "(51,ARRAY[7.0,3.2,4.7,1.4],'Iris-versicolor'),\n",
-    "(52,ARRAY[6.4,3.2,4.5,1.5],'Iris-versicolor'),\n",
-    "(53,ARRAY[6.9,3.1,4.9,1.5],'Iris-versicolor'),\n",
-    "(54,ARRAY[5.5,2.3,4.0,1.3],'Iris-versicolor'),\n",
-    "(55,ARRAY[6.5,2.8,4.6,1.5],'Iris-versicolor'),\n",
-    "(56,ARRAY[5.7,2.8,4.5,1.3],'Iris-versicolor'),\n",
-    "(57,ARRAY[6.3,3.3,4.7,1.6],'Iris-versicolor'),\n",
-    "(58,ARRAY[4.9,2.4,3.3,1.0],'Iris-versicolor'),\n",
-    "(59,ARRAY[6.6,2.9,4.6,1.3],'Iris-versicolor'),\n",
-    "(60,ARRAY[5.2,2.7,3.9,1.4],'Iris-versicolor'),\n",
-    "(61,ARRAY[5.0,2.0,3.5,1.0],'Iris-versicolor'),\n",
-    "(62,ARRAY[5.9,3.0,4.2,1.5],'Iris-versicolor'),\n",
-    "(63,ARRAY[6.0,2.2,4.0,1.0],'Iris-versicolor'),\n",
-    "(64,ARRAY[6.1,2.9,4.7,1.4],'Iris-versicolor'),\n",
-    "(65,ARRAY[5.6,2.9,3.6,1.3],'Iris-versicolor'),\n",
-    "(66,ARRAY[6.7,3.1,4.4,1.4],'Iris-versicolor'),\n",
-    "(67,ARRAY[5.6,3.0,4.5,1.5],'Iris-versicolor'),\n",
-    "(68,ARRAY[5.8,2.7,4.1,1.0],'Iris-versicolor'),\n",
-    "(69,ARRAY[6.2,2.2,4.5,1.5],'Iris-versicolor'),\n",
-    "(70,ARRAY[5.6,2.5,3.9,1.1],'Iris-versicolor'),\n",
-    "(71,ARRAY[5.9,3.2,4.8,1.8],'Iris-versicolor'),\n",
-    "(72,ARRAY[6.1,2.8,4.0,1.3],'Iris-versicolor'),\n",
-    "(73,ARRAY[6.3,2.5,4.9,1.5],'Iris-versicolor'),\n",
-    "(74,ARRAY[6.1,2.8,4.7,1.2],'Iris-versicolor'),\n",
-    "(75,ARRAY[6.4,2.9,4.3,1.3],'Iris-versicolor'),\n",
-    "(76,ARRAY[6.6,3.0,4.4,1.4],'Iris-versicolor'),\n",
-    "(77,ARRAY[6.8,2.8,4.8,1.4],'Iris-versicolor'),\n",
-    "(78,ARRAY[6.7,3.0,5.0,1.7],'Iris-versicolor'),\n",
-    "(79,ARRAY[6.0,2.9,4.5,1.5],'Iris-versicolor'),\n",
-    "(80,ARRAY[5.7,2.6,3.5,1.0],'Iris-versicolor'),\n",
-    "(81,ARRAY[5.5,2.4,3.8,1.1],'Iris-versicolor'),\n",
-    "(82,ARRAY[5.5,2.4,3.7,1.0],'Iris-versicolor'),\n",
-    "(83,ARRAY[5.8,2.7,3.9,1.2],'Iris-versicolor'),\n",
-    "(84,ARRAY[6.0,2.7,5.1,1.6],'Iris-versicolor'),\n",
-    "(85,ARRAY[5.4,3.0,4.5,1.5],'Iris-versicolor'),\n",
-    "(86,ARRAY[6.0,3.4,4.5,1.6],'Iris-versicolor'),\n",
-    "(87,ARRAY[6.7,3.1,4.7,1.5],'Iris-versicolor'),\n",
-    "(88,ARRAY[6.3,2.3,4.4,1.3],'Iris-versicolor'),\n",
-    "(89,ARRAY[5.6,3.0,4.1,1.3],'Iris-versicolor'),\n",
-    "(90,ARRAY[5.5,2.5,4.0,1.3],'Iris-versicolor'),\n",
-    "(91,ARRAY[5.5,2.6,4.4,1.2],'Iris-versicolor'),\n",
-    "(92,ARRAY[6.1,3.0,4.6,1.4],'Iris-versicolor'),\n",
-    "(93,ARRAY[5.8,2.6,4.0,1.2],'Iris-versicolor'),\n",
-    "(94,ARRAY[5.0,2.3,3.3,1.0],'Iris-versicolor'),\n",
-    "(95,ARRAY[5.6,2.7,4.2,1.3],'Iris-versicolor'),\n",
-    "(96,ARRAY[5.7,3.0,4.2,1.2],'Iris-versicolor'),\n",
-    "(97,ARRAY[5.7,2.9,4.2,1.3],'Iris-versicolor'),\n",
-    "(98,ARRAY[6.2,2.9,4.3,1.3],'Iris-versicolor'),\n",
-    "(99,ARRAY[5.1,2.5,3.0,1.1],'Iris-versicolor'),\n",
-    "(100,ARRAY[5.7,2.8,4.1,1.3],'Iris-versicolor'),\n",
-    "(101,ARRAY[6.3,3.3,6.0,2.5],'Iris-virginica'),\n",
-    "(102,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
-    "(103,ARRAY[7.1,3.0,5.9,2.1],'Iris-virginica'),\n",
-    "(104,ARRAY[6.3,2.9,5.6,1.8],'Iris-virginica'),\n",
-    "(105,ARRAY[6.5,3.0,5.8,2.2],'Iris-virginica'),\n",
-    "(106,ARRAY[7.6,3.0,6.6,2.1],'Iris-virginica'),\n",
-    "(107,ARRAY[4.9,2.5,4.5,1.7],'Iris-virginica'),\n",
-    "(108,ARRAY[7.3,2.9,6.3,1.8],'Iris-virginica'),\n",
-    "(109,ARRAY[6.7,2.5,5.8,1.8],'Iris-virginica'),\n",
-    "(110,ARRAY[7.2,3.6,6.1,2.5],'Iris-virginica'),\n",
-    "(111,ARRAY[6.5,3.2,5.1,2.0],'Iris-virginica'),\n",
-    "(112,ARRAY[6.4,2.7,5.3,1.9],'Iris-virginica'),\n",
-    "(113,ARRAY[6.8,3.0,5.5,2.1],'Iris-virginica'),\n",
-    "(114,ARRAY[5.7,2.5,5.0,2.0],'Iris-virginica'),\n",
-    "(115,ARRAY[5.8,2.8,5.1,2.4],'Iris-virginica'),\n",
-    "(116,ARRAY[6.4,3.2,5.3,2.3],'Iris-virginica'),\n",
-    "(117,ARRAY[6.5,3.0,5.5,1.8],'Iris-virginica'),\n",
-    "(118,ARRAY[7.7,3.8,6.7,2.2],'Iris-virginica'),\n",
-    "(119,ARRAY[7.7,2.6,6.9,2.3],'Iris-virginica'),\n",
-    "(120,ARRAY[6.0,2.2,5.0,1.5],'Iris-virginica'),\n",
-    "(121,ARRAY[6.9,3.2,5.7,2.3],'Iris-virginica'),\n",
-    "(122,ARRAY[5.6,2.8,4.9,2.0],'Iris-virginica'),\n",
-    "(123,ARRAY[7.7,2.8,6.7,2.0],'Iris-virginica'),\n",
-    "(124,ARRAY[6.3,2.7,4.9,1.8],'Iris-virginica'),\n",
-    "(125,ARRAY[6.7,3.3,5.7,2.1],'Iris-virginica'),\n",
-    "(126,ARRAY[7.2,3.2,6.0,1.8],'Iris-virginica'),\n",
-    "(127,ARRAY[6.2,2.8,4.8,1.8],'Iris-virginica'),\n",
-    "(128,ARRAY[6.1,3.0,4.9,1.8],'Iris-virginica'),\n",
-    "(129,ARRAY[6.4,2.8,5.6,2.1],'Iris-virginica'),\n",
-    "(130,ARRAY[7.2,3.0,5.8,1.6],'Iris-virginica'),\n",
-    "(131,ARRAY[7.4,2.8,6.1,1.9],'Iris-virginica'),\n",
-    "(132,ARRAY[7.9,3.8,6.4,2.0],'Iris-virginica'),\n",
-    "(133,ARRAY[6.4,2.8,5.6,2.2],'Iris-virginica'),\n",
-    "(134,ARRAY[6.3,2.8,5.1,1.5],'Iris-virginica'),\n",
-    "(135,ARRAY[6.1,2.6,5.6,1.4],'Iris-virginica'),\n",
-    "(136,ARRAY[7.7,3.0,6.1,2.3],'Iris-virginica'),\n",
-    "(137,ARRAY[6.3,3.4,5.6,2.4],'Iris-virginica'),\n",
-    "(138,ARRAY[6.4,3.1,5.5,1.8],'Iris-virginica'),\n",
-    "(139,ARRAY[6.0,3.0,4.8,1.8],'Iris-virginica'),\n",
-    "(140,ARRAY[6.9,3.1,5.4,2.1],'Iris-virginica'),\n",
-    "(141,ARRAY[6.7,3.1,5.6,2.4],'Iris-virginica'),\n",
-    "(142,ARRAY[6.9,3.1,5.1,2.3],'Iris-virginica'),\n",
-    "(143,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
-    "(144,ARRAY[6.8,3.2,5.9,2.3],'Iris-virginica'),\n",
-    "(145,ARRAY[6.7,3.3,5.7,2.5],'Iris-virginica'),\n",
-    "(146,ARRAY[6.7,3.0,5.2,2.3],'Iris-virginica'),\n",
-    "(147,ARRAY[6.3,2.5,5.0,1.9],'Iris-virginica'),\n",
-    "(148,ARRAY[6.5,3.0,5.2,2.0],'Iris-virginica'),\n",
-    "(149,ARRAY[6.2,3.4,5.4,2.3],'Iris-virginica'),\n",
-    "(150,ARRAY[5.9,3.0,5.1,1.8],'Iris-virginica');\n",
-    "\n",
-    "SELECT * FROM iris_data ORDER BY id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Create a test/validation dataset from the training data"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 5,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>120</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(120L,)]"
-      ]
-     },
-     "execution_count": 5,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_train, iris_test;\n",
-    "\n",
-    "-- Set seed so results are reproducible\n",
-    "SELECT setseed(0);\n",
-    "\n",
-    "SELECT madlib.train_test_split('iris_data',     -- Source table\n",
-    "                               'iris',          -- Output table root name\n",
-    "                                0.8,            -- Train proportion\n",
-    "                                NULL,           -- Test proportion (0.2)\n",
-    "                                NULL,           -- Strata definition\n",
-    "                                NULL,           -- Output all columns\n",
-    "                                NULL,           -- Sample without replacement\n",
-    "                                TRUE            -- Separate output tables\n",
-    "                              );\n",
-    "\n",
-    "SELECT COUNT(*) FROM iris_train;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"pp\"></a>\n",
-    "# 2. Call preprocessor for deep learning\n",
-    "Training dataset (uses training preprocessor):"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 6,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "2 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
-       "        <th>buffer_id</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[60, 4]</td>\n",
-       "        <td>[60, 3]</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[60, 4]</td>\n",
-       "        <td>[60, 3]</td>\n",
-       "        <td>1</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[([60, 4], [60, 3], 0), ([60, 4], [60, 3], 1)]"
-      ]
-     },
-     "execution_count": 6,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_train_packed, iris_train_packed_summary;\n",
-    "\n",
-    "SELECT madlib.training_preprocessor_dl('iris_train',         -- Source table\n",
-    "                                       'iris_train_packed',  -- Output table\n",
-    "                                       'class_text',        -- Dependent variable\n",
-    "                                       'attributes'         -- Independent variable\n",
-    "                                        ); \n",
-    "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM iris_train_packed ORDER BY buffer_id;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 7,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_train</td>\n",
-       "        <td>iris_train_packed</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>60</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>3</td>\n",
-       "        <td>all_segments</td>\n",
-       "        <td>all_segments</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_train', u'iris_train_packed', u'class_text', u'attributes', u'character varying', [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 60, 1.0, 3, 'all_segments', 'all_segments')]"
-      ]
-     },
-     "execution_count": 7,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_train_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Validation dataset (uses validation preprocessor):"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 8,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "2 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
-       "        <th>buffer_id</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[15, 4]</td>\n",
-       "        <td>[15, 3]</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[15, 4]</td>\n",
-       "        <td>[15, 3]</td>\n",
-       "        <td>1</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[([15, 4], [15, 3], 0), ([15, 4], [15, 3], 1)]"
-      ]
-     },
-     "execution_count": 8,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_test_packed, iris_test_packed_summary;\n",
-    "\n",
-    "SELECT madlib.validation_preprocessor_dl('iris_test',          -- Source table\n",
-    "                                         'iris_test_packed',   -- Output table\n",
-    "                                         'class_text',         -- Dependent variable\n",
-    "                                         'attributes',         -- Independent variable\n",
-    "                                         'iris_train_packed'   -- From training preprocessor step\n",
-    "                                          ); \n",
-    "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM iris_test_packed ORDER BY buffer_id;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 9,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_test</td>\n",
-       "        <td>iris_test_packed</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>15</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>3</td>\n",
-       "        <td>all_segments</td>\n",
-       "        <td>all_segments</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_test', u'iris_test_packed', u'class_text', u'attributes', u'character varying', [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 15, 1.0, 3, 'all_segments', 'all_segments')]"
-      ]
-     },
-     "execution_count": 9,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_test_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load\"></a>\n",
-    "# 3. Define and load model architecture\n",
-    "Import Keras libraries"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 10,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "Using TensorFlow backend.\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Couldn't import dot_parser, loading of dot files will not be possible.\n"
-     ]
-    }
-   ],
-   "source": [
-    "import keras\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Define model architecture"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 11,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "dense_1 (Dense)              (None, 10)                50        \n",
-      "_________________________________________________________________\n",
-      "dense_2 (Dense)              (None, 10)                110       \n",
-      "_________________________________________________________________\n",
-      "dense_3 (Dense)              (None, 3)                 33        \n",
-      "=================================================================\n",
-      "Total params: 193\n",
-      "Trainable params: 193\n",
-      "Non-trainable params: 0\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "model_simple = Sequential()\n",
-    "model_simple.add(Dense(10, activation='relu', input_shape=(4,)))\n",
-    "model_simple.add(Dense(10, activation='relu'))\n",
-    "model_simple.add(Dense(3, activation='softmax'))\n",
-    "    \n",
-    "model_simple.summary()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 12,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
-      ]
-     },
-     "execution_count": 12,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "model_simple.to_json()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Load into model architecture table"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 13,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_id</th>\n",
-       "        <th>model_arch</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
-       "        <td>Sophie</td>\n",
-       "        <td>A simple model</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Sophie', u'A simple model')]"
-      ]
-     },
-     "execution_count": 13,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS model_arch_library;\n",
-    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
-    "                               \n",
-    "$$\n",
-    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
-    "$$\n",
-    "::json,         -- JSON blob\n",
-    "                               NULL,                  -- Weights\n",
-    "                               'Sophie',              -- Name\n",
-    "                               'A simple model'       -- Descr\n",
-    ");\n",
-    "\n",
-    "SELECT model_id, model_arch, name, description FROM model_arch_library ORDER BY model_id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"train\"></a>\n",
-    "# 4.  Train\n",
-    "Train the model:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 14,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>madlib_keras_fit</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td></td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[('',)]"
-      ]
-     },
-     "execution_count": 14,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
-    "                               'iris_model',          -- model output table\n",
-    "                               'model_arch_library',  -- model arch table\n",
-    "                                1,                    -- model arch id\n",
-    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
-    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
-    "                                10                    -- num_iterations\n",
-    "                              );"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View the model summary:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 15,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>model</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>validation_table</th>\n",
-       "        <th>metrics_compute_frequency</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>madlib_version</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "        <th>metrics_iters</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_train_packed</td>\n",
-       "        <td>iris_model</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>model_arch_library</td>\n",
-       "        <td>1</td>\n",
-       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
-       "        <td> batch_size=5, epochs=3 </td>\n",
-       "        <td>10</td>\n",
-       "        <td>None</td>\n",
-       "        <td>10</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>2019-12-18 18:09:06.678020</td>\n",
-       "        <td>2019-12-18 18:09:09.703493</td>\n",
-       "        <td>[3.02539992332458]</td>\n",
-       "        <td>1.17-dev</td>\n",
-       "        <td>3</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.958333313465</td>\n",
-       "        <td>0.619696557522</td>\n",
-       "        <td>[0.958333313465118]</td>\n",
-       "        <td>[0.61969655752182]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>[10]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_train_packed', u'iris_model', u'class_text', u'attributes', u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, None, 10, None, None, u'madlib_keras', 0.7900390625, datetime.datetime(2019, 12, 18, 18, 9, 6, 678020), datetime.datetime(2019, 12, 18, 18, 9, 9, 703493), [3.02539992332458], u'1.17-dev', 3, [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], u'character varying', 1.0, [u'accuracy'], 0.958333313465, 0.619696557522, [0.958333313465118], [0.61969655752182], None, None, None, None, [10])]"
-      ]
-     },
-     "execution_count": 15,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_model_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"eval\"></a>\n",
-    "# 5. Evaluate\n",
-    "\n",
-    "Now run evaluate using model we built above:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 16,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>loss</th>\n",
-       "        <th>metric</th>\n",
-       "        <th>metrics_type</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>0.627631247044</td>\n",
-       "        <td>0.899999976158</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(0.62763124704361, 0.899999976158142, [u'accuracy'])]"
-      ]
-     },
-     "execution_count": 16,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_validate;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_evaluate('iris_model',       -- model\n",
-    "                                   'iris_test_packed',  -- test table\n",
-    "                                   'iris_validate'      -- output table\n",
-    "                                   );\n",
-    "\n",
-    "SELECT * FROM iris_validate;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"pred\"></a>\n",
-    "# 6. Predict\n",
-    "\n",
-    "Now predict using model we built.  We will use the validation data set for prediction as well, which is not usual but serves to show the syntax. The prediction is in the estimated_class_text column:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 17,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "30 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>estimated_class_text</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>14</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>18</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>28</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>44</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>48</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>54</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>56</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>69</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>80</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>83</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>85</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>88</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>89</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>90</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>94</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>97</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>103</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>105</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>111</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>120</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>128</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>131</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>132</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>133</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>136</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>138</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>149</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(7, u'Iris-setosa'),\n",
-       " (8, u'Iris-setosa'),\n",
-       " (9, u'Iris-setosa'),\n",
-       " (14, u'Iris-setosa'),\n",
-       " (18, u'Iris-setosa'),\n",
-       " (28, u'Iris-setosa'),\n",
-       " (44, u'Iris-setosa'),\n",
-       " (48, u'Iris-setosa'),\n",
-       " (54, u'Iris-virginica'),\n",
-       " (56, u'Iris-versicolor'),\n",
-       " (69, u'Iris-virginica'),\n",
-       " (80, u'Iris-versicolor'),\n",
-       " (83, u'Iris-versicolor'),\n",
-       " (85, u'Iris-versicolor'),\n",
-       " (88, u'Iris-virginica'),\n",
-       " (89, u'Iris-versicolor'),\n",
-       " (90, u'Iris-versicolor'),\n",
-       " (94, u'Iris-versicolor'),\n",
-       " (97, u'Iris-versicolor'),\n",
-       " (103, u'Iris-virginica'),\n",
-       " (105, u'Iris-virginica'),\n",
-       " (111, u'Iris-virginica'),\n",
-       " (120, u'Iris-virginica'),\n",
-       " (128, u'Iris-virginica'),\n",
-       " (131, u'Iris-virginica'),\n",
-       " (132, u'Iris-virginica'),\n",
-       " (133, u'Iris-virginica'),\n",
-       " (136, u'Iris-virginica'),\n",
-       " (138, u'Iris-virginica'),\n",
-       " (149, u'Iris-virginica')]"
-      ]
-     },
-     "execution_count": 17,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_predict;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_predict('iris_model', -- model\n",
-    "                                   'iris_test',  -- test_table\n",
-    "                                   'id',  -- id column\n",
-    "                                   'attributes', -- independent var\n",
-    "                                   'iris_predict'  -- output table\n",
-    "                                   );\n",
-    "\n",
-    "SELECT * FROM iris_predict ORDER BY id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Count missclassifications"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 18,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(3L,)]"
-      ]
-     },
-     "execution_count": 18,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT COUNT(*) FROM iris_predict JOIN iris_test USING (id) \n",
-    "WHERE iris_predict.estimated_class_text != iris_test.class_text;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Percent missclassifications"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 19,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>test_accuracy_percent</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>90.00</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(Decimal('90.00'),)]"
-      ]
-     },
-     "execution_count": 19,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
-    "    (select iris_test.class_text as actual, iris_predict.estimated_class_text as estimated\n",
-    "     from iris_predict inner join iris_test\n",
-    "     on iris_test.id=iris_predict.id) q\n",
-    "WHERE q.actual=q.estimated;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"pred_byom\"></a>\n",
-    "# 7. Predict BYOM\n",
-    "The predict BYOM function allows you to do inference on models that have not been trained on MADlib, but rather imported from elsewhere.  \n",
-    "\n",
-    "We will use the validation dataset for prediction as well, which is not usual but serves to show the syntax.\n",
-    "\n",
-    "See load_keras_model()\n",
-    "http://madlib.apache.org/docs/latest/group__grp__keras__model__arch.html\n",
-    "for details on how to load the model architecture and weights.  In this example we will use weights we already have:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 20,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "[]"
-      ]
-     },
-     "execution_count": 20,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "UPDATE model_arch_library \n",
-    "SET model_weights = iris_model.model_weights \n",
-    "FROM iris_model \n",
-    "WHERE model_arch_library.model_id = 1;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Now train using a model from the model architecture table directly without referencing the model table from the MADlib training.  \n",
-    "\n",
-    "Note that if you specify the class values parameter as we do below, it must reflect how the dependent variable was 1-hot encoded for training.  In this example the 'training_preprocessor_dl()' in Step 2 above encoded in the order {'Iris-setosa', 'Iris-versicolor', 'Iris-virginica'} so this is the order we pass in the parameter.  If we accidently picked another order that did not match the 1-hot encoding, the predictions would be wrong."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 21,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "30 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>estimated_dependent_var</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>14</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>18</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>28</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>44</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>48</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>54</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>56</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>69</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>80</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>83</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>85</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>88</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>89</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>90</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>94</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>97</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>103</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>105</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>111</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>120</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>128</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>131</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>132</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>133</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>136</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>138</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>149</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(7, u'Iris-setosa'),\n",
-       " (8, u'Iris-setosa'),\n",
-       " (9, u'Iris-setosa'),\n",
-       " (14, u'Iris-setosa'),\n",
-       " (18, u'Iris-setosa'),\n",
-       " (28, u'Iris-setosa'),\n",
-       " (44, u'Iris-setosa'),\n",
-       " (48, u'Iris-setosa'),\n",
-       " (54, u'Iris-virginica'),\n",
-       " (56, u'Iris-versicolor'),\n",
-       " (69, u'Iris-virginica'),\n",
-       " (80, u'Iris-versicolor'),\n",
-       " (83, u'Iris-versicolor'),\n",
-       " (85, u'Iris-versicolor'),\n",
-       " (88, u'Iris-virginica'),\n",
-       " (89, u'Iris-versicolor'),\n",
-       " (90, u'Iris-versicolor'),\n",
-       " (94, u'Iris-versicolor'),\n",
-       " (97, u'Iris-versicolor'),\n",
-       " (103, u'Iris-virginica'),\n",
-       " (105, u'Iris-virginica'),\n",
-       " (111, u'Iris-virginica'),\n",
-       " (120, u'Iris-virginica'),\n",
-       " (128, u'Iris-virginica'),\n",
-       " (131, u'Iris-virginica'),\n",
-       " (132, u'Iris-virginica'),\n",
-       " (133, u'Iris-virginica'),\n",
-       " (136, u'Iris-virginica'),\n",
-       " (138, u'Iris-virginica'),\n",
-       " (149, u'Iris-virginica')]"
-      ]
-     },
-     "execution_count": 21,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_predict_byom;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_predict_byom('model_arch_library',  -- model arch table\n",
-    "                                         1,                    -- model arch id\n",
-    "                                        'iris_test',           -- test_table\n",
-    "                                        'id',                  -- id column\n",
-    "                                        'attributes',          -- independent var\n",
-    "                                        'iris_predict_byom',   -- output table\n",
-    "                                        'response',            -- prediction type\n",
-    "                                         FALSE,                -- use GPUs\n",
-    "                                         ARRAY['Iris-setosa', 'Iris-versicolor', 'Iris-virginica'], -- class values\n",
-    "                                         1.0                   -- normalizing const\n",
-    "                                   );\n",
-    "SELECT * FROM iris_predict_byom ORDER BY id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Count missclassifications:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 22,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(3L,)]"
-      ]
-     },
-     "execution_count": 22,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT COUNT(*) FROM iris_predict_byom JOIN iris_test USING (id)\n",
-    "WHERE iris_predict_byom.estimated_dependent_var != iris_test.class_text;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Percent missclassifications:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 23,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>test_accuracy_percent</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>90.00</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(Decimal('90.00'),)]"
-      ]
-     },
-     "execution_count": 23,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
-    "    (select iris_test.class_text as actual, iris_predict_byom.estimated_dependent_var as estimated\n",
-    "     from iris_predict_byom inner join iris_test\n",
-    "     on iris_test.id=iris_predict_byom.id) q\n",
-    "WHERE q.actual=q.estimated;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"class2\"></a>\n",
-    "# Classification with Other Parameters\n",
-    "\n",
-    "<a id=\"val_dataset\"></a>\n",
-    "# 1.  Validation dataset\n",
-    "Now use a validation dataset and compute metrics every 2nd iteration using the 'metrics_compute_frequency' parameter.  This can help reduce run time if you do not need metrics computed at every iteration."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 24,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>madlib_keras_fit</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td></td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[('',)]"
-      ]
-     },
-     "execution_count": 24,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
-    "                               'iris_model',          -- model output table\n",
-    "                               'model_arch_library',  -- model arch table\n",
-    "                                1,                    -- model arch id\n",
-    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
-    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
-    "                                10,                   -- num_iterations\n",
-    "                                FALSE,                -- use GPUs\n",
-    "                                'iris_test_packed',   -- validation dataset\n",
-    "                                2,                    -- metrics compute frequency\n",
-    "                                FALSE,                -- warm start\n",
-    "                               'Sophie L.',           -- name\n",
-    "                               'Simple MLP for iris dataset'  -- description\n",
-    "                              );"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View the model summary:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 25,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>model</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>validation_table</th>\n",
-       "        <th>metrics_compute_frequency</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>madlib_version</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "        <th>metrics_iters</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_train_packed</td>\n",
-       "        <td>iris_model</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>model_arch_library</td>\n",
-       "        <td>1</td>\n",
-       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
-       "        <td> batch_size=5, epochs=3 </td>\n",
-       "        <td>10</td>\n",
-       "        <td>iris_test_packed</td>\n",
-       "        <td>2</td>\n",
-       "        <td>Sophie L.</td>\n",
-       "        <td>Simple MLP for iris dataset</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>2019-12-18 18:09:19.330964</td>\n",
-       "        <td>2019-12-18 18:09:21.010635</td>\n",
-       "        <td>[0.915475130081177, 1.10240316390991, 1.24091100692749, 1.37801814079285, 1.67959213256836]</td>\n",
-       "        <td>1.17-dev</td>\n",
-       "        <td>3</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.983333349228</td>\n",
-       "        <td>0.308444350958</td>\n",
-       "        <td>[0.949999988079071, 0.975000023841858, 0.975000023841858, 0.983333349227905, 0.983333349227905]</td>\n",
-       "        <td>[0.5235316157341, 0.450434356927872, 0.391158282756805, 0.344655215740204, 0.30844435095787]</td>\n",
-       "        <td>0.933333337307</td>\n",
-       "        <td>0.363271415234</td>\n",
-       "        <td>[0.866666674613953, 0.933333337306976, 0.933333337306976, 0.933333337306976, 0.933333337306976]</td>\n",
-       "        <td>[0.549321353435516, 0.490176409482956, 0.438665509223938, 0.397390514612198, 0.363271415233612]</td>\n",
-       "        <td>[2, 4, 6, 8, 10]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_train_packed', u'iris_model', u'class_text', u'attributes', u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', 2, u'Sophie L.', u'Simple MLP for iris dataset', u'madlib_keras', 0.7900390625, datetime.datetime(2019, 12, 18, 18, 9, 19, 330964), datetime.datetime(2019, 12, 18, 18, 9, 21, 10635), [0.915475130081177, 1.10240316390991, 1.24091100692749, 1.37801814079285, 1.67959213256836], u'1.17-dev', 3, [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], u'character varying', 1.0, [u'accuracy'], 0.983333349228, 0.308444350958, [0.949999988079071, 0.975000023841858, 0.975000023841858, 0.983333349227905, 0.983333349227905], [0.5235316157341, 0.450434356927872, 0.391158282756805, 0.344655215740204, 0.30844435095787], 0.933333337307, 0.363271415234, [0.866666674613953, 0.933333337306976, 0.933333337306976, 0.933333337306976, 0.933333337306976], [0.549321353435516, 0.490176409482956, 0.438665509223938, 0.397390514612198, 0.363271415233612], [2, 4, 6, 8, 10])]"
-      ]
-     },
-     "execution_count": 25,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_model_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Accuracy by iteration"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 26,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x12c58eb50>"
-      ]
-     },
-     "execution_count": 26,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "import pandas as pd\n",
-    "import numpy as np\n",
-    "import sys\n",
-    "import os\n",
-    "from matplotlib import pyplot as plt\n",
-    "\n",
-    "# get accuracy and iteration number\n",
-    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
-    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
-    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
-    "\n",
-    "# get number of points\n",
-    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
-    "num_points = num_points_proxy[0]\n",
-    "\n",
-    "# reshape to np arrays\n",
-    "iters = np.array(iters_proxy).reshape(num_points)\n",
-    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
-    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
-    "\n",
-    "#plot\n",
-    "plt.title('Iris validation accuracy by iteration')\n",
-    "plt.xlabel('Iteration number')\n",
-    "plt.ylabel('Accuracy')\n",
-    "plt.grid(True)\n",
-    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
-    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
-    "plt.legend()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Loss by iteration"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 27,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x12f1b1c10>"
-      ]
-     },
-     "execution_count": 27,
-     "metadata": {},
-     "output_type": "execute_result"
-    },
-    {
-     "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": [
-    "# get loss\n",
-    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
-    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
-    "\n",
-    "# reshape to np arrays\n",
-    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
-    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
-    "\n",
-    "#plot\n",
-    "plt.title('Iris validation loss by iteration')\n",
-    "plt.xlabel('Iteration number')\n",
-    "plt.ylabel('Loss')\n",
-    "plt.grid(True)\n",
-    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
-    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
-    "plt.legend()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Accuracy by time"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 28,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x12f24bbd0>"
-      ]
-     },
-     "execution_count": 28,
-     "metadata": {},
-     "output_type": "execute_result"
-    },
-    {
-     "data": {
-      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYsAAAEWCAYAAACXGLsWAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADl0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uIDIuMi4zLCBodHRwOi8vbWF0cGxvdGxpYi5vcmcvIxREBQAAIABJREFUeJzt3Xd8FHX+x/HXh0DoRUA5KQKC5bBSRKOnBlAPznZnuVNPPAtiOSwo1p8ioqdiL3AqKthQ7O0ORQRijQoIoiIoIAiIjR6kJp/fHzPh1phkF5LJZJP38/HYR2ZnvjP73t3sfna+08zdERERKU2NuAOIiEjlp2IhIiJJqViIiEhSKhYiIpKUioWIiCSlYiEiIkmpWMivmNnfzezNCny8dmbmZlYzvP+6mf0jlbbb8FhXm9nDZclbnZlZtpktjnD5X5hZdlTLl7IxHWdRvZjZAqCfu78VdxYICgDwDVDL3TeXY9ts4El3b10eOaV8X1MzexRY7O7XlHVZUjG0ZiFbbOsvdomX3jepCCoW1ZiZnW5m75vZXWa2DBgSjnsvnG7htB/NbLWZfWZmexaznL+Z2dQi4waa2avh8JFmNj1cxiIzG1JKphwz6xcOZ5jZ7Wb2s5nNB44s0vYMM/vSzNaY2XwzOyccXx94HWhpZnnhraWZDTGzJxPmPybs+lgZPu7vE6YtMLNBZjbTzFaZ2TNmVqeEzB3MbJKZLQuzjjGzJgnT25jZi2b2U9hmeMK0sxOewywz6xKOdzPrmNDuUTO7MRzONrPFZnaFmX0PjDaz7czsP+FjrAiHWyfM39TMRpvZd+H0l8Pxn5vZ0QntaoXPoXMp79HVYZsFZvb3cNx+ZvaDmWUktDvOzD4tZv7+wN+By8P35rWE1/ywcHiImT1nZk+Gr81nZrarmV0V/j8uMrMjEpbZ2MweMbOlZrbEzG5MzCJlp2Ih+wPzgRbAv4pMOwI4BNgVaAz8FVhWzDJeA3Yzs10Sxp0CPBUOrwVOA5oQfOGfZ2Z/TiHb2cBRQGegG3BCkek/htMbAWcAd5lZF3dfC/QBvnP3BuHtu8QZzWxX4GngYmB7YBzwmpllJjT7K9AbaA/sDZxeQk4DbgZaAr8H2gBDwsfJAP4DLATaAa2AseG0E8N2p4XP4RiKf32L8zugKdAW6E/wWR4d3t8JWAcMT2j/BFAP2APYAbgrHP84cGpCuz8BS919eimP2zx8Hv8ARprZbu4+Jcx+RELbvuHyf8XdRwJjgFvD9+boom1CR4e5twOmA+PD59kKGAo8mND2UWAz0JHg/+UIoF8Jy5Vt4e66VaMbsAA4LBw+Hfi2yPTTgffC4Z7AV8ABQI0ky30SGBwO7wKsAeqV0PZu4K5wuB3gQM3wfg7BNhWAScC5CfMdkdi2mOW+DFwUDmcT9IknTh9C0OcOcC3wbMK0GsASIDvhdTo1YfqtwAMpvsZ/BqaHw1nAT8VlJvjyu6iEZTjQMeH+o8CNCc9tI1CnlAz7AivC4R2BAmC7Ytq1DN+rRuH954HLS1hmNsEXcv2Ecc8C14bDVwBjwuGmwC/AjiUsa8vzKeF/cwgwIWHa0UAekBHebxi+Rk0IfuhsAOomtD8ZmBz3560q3bRmIYtKmuDukwh+nY4AfjSzkWbWqITmTxF8QCFYq3jZ3X8BMLP9zWxy2EWyCjiX4NdpMi2L5FuYONHM+pjZh2a23MxWEvwqTmW5hcvesjx3Lwgfq1VCm+8Thn8BGhS3IDNrYWZjw+6P1QSFszBHG2ChF79Bvg0wL8W8Rf3k7usTMtQzswfNbGGY4R2gSbhm0wZY7u4rii7EgzWu94Hjw66zPgS/+kuywoM1t0ILCV5LCJ730WE34F+Bd9196TY+P4AfEobXAT+7e37CfQjek7ZALWBp2KW4kmCtY4cyPLYUoWIhpe4O5+73untXoBNBd9RlJTSdAGxvZvsSFI2nEqY9BbwKtHH3xsADBF03ySwl+KIrtFPhgJnVBl4AbgdauHsTgq6kwuUm283vO4IvmcLlWfhYS1LIVdRN4ePt5e6NCLp1CnMsAnay4jdCLwI6lLDMXwi6jQr9rsj0os/vUmA3YP8wwyHheAsfp2nidpQiHgsznwjkuntpr8F2YTEotBPBa0k4Xy5wHEEX1BOlLKc8d8NcRLBm0dzdm4S3Ru6+Rzk+RrWnYiElCjda7m9mtQi2O6wn6M74DXffBDwH3EbQBTEhYXJDgl+2682sO8GaRyqeBS40s9Zmth1wZcK0TKA2QRfPZjPrw6/7y38AmplZ41KWfaSZ9Qqf36UEXzgfpJgtUUOCLpJVZtaKXxfUjwmK3i1mVt/M6pjZQeG0h4FBZtbVAh3NrLCAzQBOsWAjf2/g0BQyrANWmllT4LrCCeGv+9eBf4cbwmuZ2SEJ874MdAEuophtDMW43swyzexggm1GzyVMexy4HNgLeLGUZfwA7JzCYyUVPr83gTvMrJGZ1bBgp4Nkr5lsBRULKU0j4CFgBUF3wzKCYlCSp4DDgOeKdLucDww1szXAYIIv6lQ8RNCv/ynwCQlfPu6+BrgwXNYKggL0asL02QQbsOeHXRMtE5aLu88h+DV9H/AzQZ/40e6+McVsia4n+LJdBfy3SM78cNkdgW+BxcDfwmnPEexU8BTBdoOXCQotBF/cRwMrCfYcejlJhruBuuFz+RB4o8j0vsAmYDbBjgEXJ2RcR7CW1p7Sv+Ah6JpbQbA2MYZgm9LshOkvEayxvVTYDVmCR4BO4XuT7Lml4jSCHxCzwnzPE2yrkXKig/JEBDMbDOzq7qcmbZx8WfOAc7ySHPgp5UMH84hUc2G31VkEax9lXdbxBNsjJpV1WVK5qBtKpBozs7MJNhC/7u7vlHFZOcD9wD/DvcukClE3lIiIJKU1CxERSarKbLNo3ry5t2vXLtYMa9eupX79+skbVlLKHy/lj086Z4ey5Z82bdrP7r59snZVpli0a9eOqVOnJm8YoZycHLKzs2PNUBbKHy/lj086Z4ey5TezhclbqRtKRERSoGIhIiJJqViIiEhSKhYiIpKUioWIiCSlYiEiIkmpWIhUcbmLcrn53ZvJXZQbdxSJQO6iXMZ8Oyby97fKHGchIr/m7oz5bAxnvnImmws2k1Ejg75796V1o9aRPN6CBQuY5Ol5/sB0zb549WKemPkE+QX5jFk0homnTSSrTVYkj6ViIVJFuDvzV8xn0jeTmLRgEpO+mcSPa3/cMn1zwWZGzxiNpXSRwm30bXSLjlwaZveECw5uzN9IzoIcFQsR+a0lq5cwecFkJn4zkUnfTOLbVcE33o4NduSIDkewU6OduPPDO9mUv4nMjMxIf3mm81HQ6Zo9d1EuvR7vxYbNG8jMyCS7XXZkj6ViIZJGlv2yjMkLJgdrD99MYs6yOQA0rduUHu16cMVBV9CzfU92a7YbwWXF4ahdjyJnQQ7Z7bIjKxQSj6w2WUw8bSKjJo/izB5nRvr+RloswmsH3wNkAA+7+y1FprcFRgHbA8uBU919cTjtVuBIgo3wE4CLXOdTl2pm9YbVvLvw3S1dSzO+nwFAg8wGHNL2EM7ucja9du7F3i32poYVv79KVpssFYkqLKtNFht22hD5exxZsTCzDGAEcDjBdYenmNmr7j4rodntwOPu/piZ9QRuBvqa2YHAQcDeYbv3CC5YnxNVXpHKYN2mdeQuzt2y5vDxko/J93xqZ9TmwDYHcmOPG+nZvifdWnajVkatuONKNRLlmkV3YK67zwcws7HAsQQXVC/UCbgkHJ7M/y5K70AdgguwG1AL+CHCrCKx2JS/ianfTWXiNxN58dMXmfXeLDbkbyDDMujeqjtX/uFKerbvSVbrLOrWqht3XKnGIrtSnpmdAPR2937h/b7A/u4+IKHNU8BH7n6PmR0HvAA0d/dlZnY70I+gWAx39/8r5jH6A/0BWrRo0XXs2LGRPJdU5eXl0aBBg1gzlIXyR6/AC5iXN4/pK6fzycpPmLlqJuvy1wGwc92d6dqsK52bdGbvxntTv2Z6XV8hHV7/kqRzdihb/h49ekxz927J2sW9gXsQMNzMTgfeAZYA+WbWEfg9ULhD+AQzO9jd302c2d1HAiMBunXr5nHvzZCue1QUUv7y5+7MWTZnS7fS5AWTWb5uOQC7NduN0zufTs/2Pclul83nH39e6fJvjcr4+qcqnbNDxeSPslgsAdok3G8djtvC3b8DjgMwswbA8e6+MryI/IfunhdOex3IAn5VLEQqo4UrFzLpm0lbdmddmrcUgDaN2nDMbsfQq30verTrQatGrWJOKpK6KIvFFGAXM2tPUCROAk5JbGBmzYHl7l4AXEWwZxQEh8ecbWY3E3RDHQrcHWFWkW32Q94PW9YcJi2YxPwV8wHYof4O9Gzfk57tetKzfU923m7nLbuziqSbyIqFu282swHAeIJdZ0e5+xdmNhSY6u6vAtnAzWbmBN1Q/wxnfx7oCXxGsLH7DXd/LaqsIltjxboVvL3w7S0F4oufvgCgce3GZLfL5qL9L6Jn+57ssf0eKg5SZUS6zcLdxwHjiowbnDD8PEFhKDpfPnBOlNlEUrV241re+/a9LWsOnyz9hAIvoG7Nuhzc9mD67t2Xnu170mXHLmTUyIg7rkgk4t7ALWkkd1FulT0SOPG5ddmxCx8t+WjLdoePFn/EpoJN1KpRiwNaH8DgQwbTs31PurfqTu2ateOOLlIhVCwkJbmLcun5eE82bN5AzRo1GZQ1iA5NO8Qd61dmL53NvE/mbfV885bP4/bc29lcsBkzo2aNmmzM30gNq0HXHbtySdYl9Gzfk4PaHET9zPTanVWkvKhYSFI///IzA8cPZP3m9QBsKtjEze/fHHOqEnxVttndna47duWKg67g0HaH0qROk/LJJZLmVCykRBvzNzLi4xFc//b1rNmwhppWE8eplVGLp497mq4tu8Yd8Vdyc3PJytr67rFp303j5BdP3nJm1juOuKPKdbOJlJWKhfyGu/PaV68x6M1BfL38a47ocAR3HnEnqzesrtTbLObVmUebxm2SNyyiTeM2TDptUqV+biJxU7GQX5n5w0wuGX8JE7+ZyO7Nd+e/p/yXPh37bNkFtKp+kerMrCKlU7EQAH5c+yN3fHUH494ZR+Pajbm3972c2+1cndlURAAVi2pvw+YN3PvRvdz47o2s3biWC7pfwOBDB9O0btO4o4lIJaJiUU25Oy/NfonLJlzG/BXzOWrXozix8Ymc1vu0uKOJSCVU/KW1pEqbvnQ6PR7rwfHPHk/dmnUZf+p4Xjv5NXaqt1Pc0USkktKaRTWydM1Srpl0DaNnjKZZvWbcf+T99OvSj5o19G8gIqXTt0Q1sG7TOu768C5ufu9mNmzewCVZl3DNIdfogDMRSZmKRRXm7jz7xbNc8dYVLFy1kD/v/mduO/w2OjbtGHc0EUkzKhZV1JQlUxg4fiDvL3qffVrsw+hjR9OjfY+4Y4lImlKxqGKWrF7CVROv4omZT7BD/R146OiHOGPfM3TqbBEpExWLKuKXTb9w+we3M+z9YWwu2MyVB13JVQdfRaPajeKOJiJVgIpFmivwAp7+7GmunHgli1cv5sROJzLssGG036593NFEpApRsUhjuYtyGTh+IB8t+YguO3bhqeOe4uC2B8cdS0SqIBWLNPTtqm+58q0refrzp9mxwY48euyj9N2nLzVMx1iKSDRULNJI3sY8hr03jNtzbwfgmoOv4Yo/XEGDzAYxJxORqk7FIg0UeAGPf/o4V0+8mqV5Szl5z5O55bBb2KmxTs8hIhVDxaKSe3fhuwwcP5BpS6exf6v9eeGvL+i6CyJS4VQsKqlvVnzD5W9dzvOznqd1o9Y8+ZcnOXmvk7VdQkRioWJRyazesJqb3r2Juz68i5o1anJ99vUMOnAQ9WrVizuaiFRjKhaVRH5BPqNnjOb/Jv0fP679kdP2OY2bet5Eq0at4o4mIhLt9SzMrLeZzTGzuWZ2ZTHT25rZRDObaWY5ZtY6YdpOZvammX1pZrPMrF2UWeM0+ZvJdB3ZlbNfO5uOTTvycb+PeezPj6lQiEilEVmxMLMMYATQB+gEnGxmnYo0ux143N33BoYCNydMexy4zd1/D3QHfowqa1zmLp/LX575Cz0f78nK9St55oRneO+M99iv1X5xRxMR+ZUou6G6A3PdfT6AmY0FjgVmJbTpBFwSDk8GXg7bdgJquvsEAHfPizBnhVu5fiU3vnMj9350L7Vr1uamnjdx8QEXU7dW3bijiYgUy9w9mgWbnQD0dvd+4f2+wP7uPiChzVPAR+5+j5kdB7wANAcOBvoBG4H2wFvAle6eX+Qx+gP9AVq0aNF17NixkTyXVOXl5dGgQckHyOV7Pv9Z+h9GLxjN6k2r6f273pzV7iya1W5WgSlLlix/Zaf88Urn/OmcHcqWv0ePHtPcvVvShu4eyQ04AXg44X5fYHiRNi2BF4HpwD3AYqBJOO8qYGeCtZ8XgLNKe7yuXbt63CZPnlzitPFzx/seI/ZwhuCHjj7UP/nuk4oLlqLS8qcD5Y9XOudP5+zuZcsPTPUUvtOj7IZaArRJuN86HLeFu38HHAdgZg2A4919pZktBmb4/7qwXgYOAB6JMG8kZv88m0FvDuK/X/+XnbfbmRf++gJ/2f0vmFnc0UREUhZlsZgC7GJm7QmKxEnAKYkNzKw5sNzdC4CrgFEJ8zYxs+3d/SegJzA1wqzlbvm65Vyfcz3/nvpv6tasy62H3cqF+19I7Zq1444mIrLVIisW7r7ZzAYA44EMYJS7f2FmQwlWe14FsoGbzcyBd4B/hvPmm9kgYKIFP8GnAQ9FlbU8bcrfxANTH+C6nOtYtWEVZ3c5m6E9hrJD/R3ijiYiss0iPSjP3ccB44qMG5ww/DzwfAnzTgD2jjJfeXJ3Plz2Iec9cB6zf55Nr/a9uOuPd7FXi73ijiYiUmY6grscfPHjF1z65qWMnzeeXZruwqsnvcpRux6l7RIiUmWoWJTBz7/8zHWTr+PBaQ/SsHZD/tnhn9x58p1kZmTGHU1EpFypWGyDjfkbGf7xcIa+PZS8jXmc2+1chmQP4fOPP1ehEJEqScViK7g7r855lUETBjF3+Vx6d+zNHUfcQafti57FRESkalGxSNHMH2YycPxAJn0zid83/z3jThlHn136xB1LRKRCqFgk8UPeD1w7+Voemf4ITeo04b4+93FO13OolVEr7mgiIhVGxaIEGzZv4J6P7uHGd25k3eZ1XNj9QgYfOpjt6m4XdzQRkQqnYlGEu/Pily9y2YTL+GblNxy969Hcdvht7NZ8t7ijiYjERsUiwSdLP2Hg+IG8s/Ad9txhT9489U0O73B43LFERGKnYgG8Nuc1rn/7eqYtnUbzes154MgHOKvLWdSsoZdHRARULHjm82c46YWTAKhZoyZPHfeU1iZERIqI9Brc6WDeinkYwWk53J2p36XVyW1FRCpEtS8WPdr1oE7NOmRYBpkZmWS3y447kohIpVPtu6Gy2mQx8bSJ5CzIIbtdNlltsuKOJCJS6VT7YgFBwVCREBEpWbXvhhIRkeRULEREJCkVCxERSUrFQkREklKxEBGRpFQsREQkKRULERFJSsVCRESSUrEQEZGkVCxERCSpSIuFmfU2szlmNtfMrixmelszm2hmM80sx8xaF5neyMwWm9nwKHOKiEjpIisWZpYBjAD6AJ2Ak82sU5FmtwOPu/vewFDg5iLTbwDeiSqjiIikJso1i+7AXHef7+4bgbHAsUXadAImhcOTE6ebWVegBfBmhBlFRCQF5u7RLNjsBKC3u/cL7/cF9nf3AQltngI+cvd7zOw44AWgObCCoIicChwGdEucL2H+/kB/gBYtWnQdO3ZsJM8lVXl5eTRo0CDWDGWh/PFS/vikc3YoW/4ePXpMc/duydrFfYryQcBwMzudoLtpCZAPnA+Mc/fFZlbizO4+EhgJ0K1bN8/Ozo46b6lycnKIO0NZKH+8lD8+6ZwdKiZ/lMViCdAm4X7rcNwW7v4dcByAmTUAjnf3lWaWBRxsZucDDYBMM8tz999sJBcRkehFWSymALuYWXuCInEScEpiAzNrDix39wLgKmAUgLv/PaHN6QTdUCoUIiIxiWwDt7tvBgYA44EvgWfd/QszG2pmx4TNsoE5ZvYVwcbsf0WVR0REtl3SNQszuwB40t1XbO3C3X0cMK7IuMEJw88DzydZxqPAo1v72CIiUn5SWbNoAUwxs2fDg+xK3uIsIiJVUtJi4e7XALsAjwCnA1+b2U1m1iHibCIiUkmktM3Cg4Mxvg9vm4HtgOfN7NYIs4mISCWRtFiY2UVmNg24FXgf2MvdzwO6AsdHnE8qk9xcuPnm4G9Vo+cm6So3l53GjIn8/U1l19mmwHHuvjBxpLsXmNlR0cSSSic3F7KzYeNGMIPdd4dKdsRrlzVroGHDrZ8xLw9mzwb3WJ/bNucvTQU+t0jyV5C0zR6+v+3dYcwYmDgRsrIieahUisXrwPLCO2bWCPi9u3/k7l9Gkkoqn2efDQoFBF88+fnQvHm8mYrYVKMGNG269TOuWhU8J4j1uW1z/tJU4HOLJH8FSdvs4ftrEHw+c3JiLRb3A10S7ucVM06qsvx8eDM8n2NGBmRmwqOPRvZPua0+29ZTHuTmQq9ewYctxue2zflLU4HPLZL8FSRts4fvb8GGDdTIzAzW/iOSSrEwTzjbYNj9FPc5paQi3X03zJoFQ4YEXzjZ2ZWuUJRJVlaw+p6To+cm6SV8fxeMGsXOZ54Z6fubypf+fDO7kGBtAoKT/M2PLJFULl99BddcA8ccA4MHB/3eVVFWVtX9Iq3Kz00gK4tvN2xg54jf41R2nT0XOJDg/E6Lgf0JTwsuVVxBAZx1FtSpA/ffX3ULhYgklXTNwt1/JDgJoFQ3I0bAe+/B6NHQsmXcaUQkRqmcG6oOcBawB1CncLy7nxlhLonb/Plw5ZXQpw/84x9xpxGRmKXSDfUE8Dvgj8DbBNelWBNlKIlZQQH06xfs+fTgg+p+EpGUikVHd78WWOvujwFHEmy3kKpq5EiYPBnuuAPatEneXkSqvFSKxabw70oz2xNoDOwQXSSJ1bffwmWXBfvm9+sXdxoRqSRS2XV2pJltB1wDvEpwmdNrI00l8XCHs88O/j78sLqfRGSLUouFmdUAVocXPnoH2LlCUkksfvf668GR2sOHQ7t2cccRkUqk1G6o8NrYl1dQFonTkiV0/Pe/4ZBD4Lzz4k4jIpVMKtss3jKzQWbWxsyaFt4iTyYVxx3OPRfbvBkeeQRqRHZpdhFJU6lss/hb+PefCeMcdUlVHWPGwH/+wzfnn0/Hjh3jTiMilVAqR3C3r4ggEpPvv4cLL4SsLBYfdxwqFSJSnFSO4D6tuPHu/nj5x5EK5Q7nnw+//AKjRgWFQ0SkGKl0Q+2XMFwH6AV8AqhYpLvnnoOXXoJbbgmuoKZiISIlSKUb6oLE+2bWBBgbWSKpGD/9BP/8J+y3H1x6adxpRKSS25bdXtYCKW3HMLPeZjbHzOaa2ZXFTG9rZhPNbKaZ5ZhZ63D8vmaWa2ZfhNP+9tulS5lccEFwScZRo6CmrmUlIqVLZZvFawR7P0FQXDoBz6YwXwYwAjic4DoYU8zsVXefldDsduBxd3/MzHoCNwN9gV+A09z9azNrCUwzs/HuvnIrnpuU5KWX4JlnYOhQ2HPPuNOISBpI5Sfl7QnDm4GF7r44hfm6A3PdfT6AmY0FjgUSi0Un4JJweDLwMoC7f1XYwN2/M7Mfge0BFYuyWr48OOhu332DU5CLiKQglWLxLbDU3dcDmFldM2vn7guSzNcKWJRwv/Aqe4k+BY4D7gH+AjQ0s2buvqywgZl1BzKBeSlklWQuvhiWLYM33oBateJOIyJpwty99AZmU4ED3X1jeD8TeN/d90sy3wlAb3fvF97vC+zv7gMS2rQEhhNsA3kHOB7Ys7C7ycx2BHKAf7j7h8U8Rn/CS7y2aNGi69ix8W53z8vLo0GDBrFmKE3T3Fz2vvpqFvTty4Izf3vtqsqePxnlj1c650/n7FC2/D169Jjm7t2SNnT3Um/AjGLGfZrCfFnA+IT7VwFXldK+AbA44X4jgl10T0j2WO5O165dPW6TJ0+OO0LJVqxwb9nSfY893NevL7ZJpc6fAuWPVzrnT+fs7mXLD0z1FL5jU9kb6iczO6bwjpkdC/ycwnxTgF3MrH24NnISwSnOtzCz5uGZbQuLyahwfCbwEsHG7+dTeCxJZtCg4DiK0aOhdu2404hImkmlWJwLXG1m35rZt8AVwDnJZnL3zcAAYDzwJfCsu39hZkMTik82MMfMvgJaAP8Kx/8VOAQ43cxmhLd9t+aJSYI33wxOEHjZZcFxFSIiWymVg/LmAQeYWYPwfl6qC3f3ccC4IuMGJww/D/xmzcHdnwSeTPVxpBRr1gQXNNptNxgyJO40IpKmkq5ZmNlNZtbE3fPcPc/MtjOzGysinJSDyy+HRYuCg+/q1Ik7jYikqVS6ofp4wsFwHlw170/RRZJyM3kyPPBAsLvsgQfGnUZE0lgqxSLDzLZsETWzuoC2kFZ2a9dCv37QsSPcqBVBESmbVA7KGwNMNLPRgAGnA49FGUrKwdVXw/z58PbbUK9e3GlEJM2lsoF7mJl9ChxGcI6o8UDbqINJGbz3Htx3X3BW2UMOiTuNiFQBqZ519geCQnEi0JNgV1ipjNatgzPPhLZtg+tUiIiUgxLXLMxsV+Dk8PYz8AzB6UF6VFA22RaDB8PXX8Nbb0Ean75ARCqX0rqhZgPvAke5+1wAMxtYIalk23z0Edx5J/TvD716xZ1GRKqQ0rqhjgOWApPN7CEz60WwgVsqo/Xr4YwzoGVLuPXWuNOISBVTYrFw95fd/SRgd4JrTVwM7GBm95vZERUVUFJ0ww3w5Zfw0EPQuHHcaUSkikm6gdvd17r7U+5+NNAamE5wfiipLKZNg2HD4PTToXfvuNOISBW0VdfgdvcV7j7S3dUhXlls3Bh0P+2wQ7C9QkQkAqkclCeV2U03wWefwSuvwHbbxZ1GRKqorVqzkEpm5kz417/glFPgmGOStxcR2UYqFulq06ag+6lpU7j33rhm7Q3GAAAUFUlEQVTTiEgVp26odHXbbfDJJ/D889CsWdxpRKSK05pFOvriC7j+ejjhBDj++LjTiEg1oGKRbvLzg3M/NWwII0bEnUZEqgl1Q6Wbu+6Cjz+Gp54KdpcVEakAWrNIJ199BddeC8ceCyedFHcaEalGVCzSRWH3U506cP/9YDpNl4hUHHVDpYsRI+D99+HRR2HHHeNOIyLVjNYs0sG8eXDVVdCnD5x2WtxpRKQaUrGo7AoKoF8/qFkTRo5U95OIxELdUJXdgw9CTk5QKFq3jjuNiFRTka5ZmFlvM5tjZnPN7Mpiprc1s4lmNtPMcsysdcK0f5jZ1+HtH1HmrLQWLoTLL4fDDgvWLkREYhJZsTCzDGAE0AfoBJxsZp2KNLsdeNzd9waGAjeH8zYFrgP2B7oD15lZ9TqlqjucfXbw96GH1P0kIrGKcs2iOzDX3ee7+0ZgLHBskTadgEnh8OSE6X8EJrj7cndfAUwAqtdVfUaNggkTgkuktmsXdxoRqeai3GbRCliUcH8xwZpCok8JrvV9D/AXoKGZNSth3lZFH8DM+gP9AVq0aEFOTk55Zd8meXl55ZKh9k8/sd9FF5G3zz7M2H33YJtFBSiv/HFR/nilc/50zg4Vkz/uDdyDgOFmdjrwDrAEyE91ZncfCYwE6Natm2dnZ0cQMXU5OTmUOYM7HHUUFBTQ5Pnnye7YsVyypaJc8sdI+eOVzvnTOTtUTP4oi8USoE3C/dbhuC3c/TuCNQvMrAFwvLuvNLMlQHaReXMizFp5PPkkjBsXnAOqAguFiEhpotxmMQXYxczam1kmcBLwamIDM2tuZoUZrgJGhcPjgSPMbLtww/YR4biqbelSuOgiOPBAuOCCuNOIiGwRWbFw983AAIIv+S+BZ939CzMbamaF1wDNBuaY2VdAC+Bf4bzLgRsICs4UYGg4rupyh/PPh19+gUcegYyMuBOJiGwR6TYLdx8HjCsybnDC8PPA8yXMO4r/rWlUfc8+Cy+/DMOGwe67x51GRORXdLqPyuCnn2DAANhvP7jkkrjTiIj8hopFZTBgAKxaBaNHB+eAEhGpZPTNFLcXXwy6oG64AfbYI+40IiLF0ppFnJYtCzZqd+4MV1wRdxoRkRJpzSJOF18cFIzx46FWrbjTiIiUSGsWcfnPf4ID8K6+GvbZJ+40IiKlUrGIw8qVcM45sOee8H//F3caEZGk1A0Vh0svhR9+gFdegczMuNOIiCSlNYuKNn58cPrxyy6Dbt3iTiMikhIVi4q0enVwQaPdd4frros7jYhIytQNVZEuvxwWL4b334c6deJOIyKSMq1ZVJRJk+DBB2HgQMjKijuNiMhWUbGoCHl50K9fcH2KG26IO42IyFZTN1RFuPpq+OYbeOcdqFcv7jQiIltNaxZRe/dduO++4GSBBx8cdxoRkW2iYhGlX36BM8+E9u3h5pvjTiMiss3UDRWlwYNh7lyYOBEaNIg7jYjINtOaRVQ+/BDuuis4rUfPnnGnEREpExWLKKxfD2ecAa1awa23xp1GRKTM1A0VhaFDYfZseOMNaNQo7jQiImWmNYvyNm1asDZxxhnwxz/GnUZEpFyoWJQj27QpKBI77AB33hl3HBGRcqNuqHLU9skn4bPP4NVXoUmTuOOIiJQbFYvy8umn7DRmDPz973D00XGnEZEkNm3axOLFi1m/fj2NGzfmyy+/jDvSNkslf506dWjdujW1tvESzioW5SHsftrcqBGZ99wTdxoRScHixYtp2LAh7dq1Iy8vj4YNG8YdaZutWbOm1PzuzrJly1i8eDHt27ffpseIdJuFmfU2szlmNtfMrixm+k5mNtnMppvZTDP7Uzi+lpk9ZmafmdmXZnZVlDnL7NZbYfp0vrr4YmjWLO40IpKC9evX06xZM8ws7iiRMzOaNWvG+vXrt3kZkRULM8sARgB9gE7AyWbWqUiza4Bn3b0zcBLw73D8iUBtd98L6AqcY2btospaJl98Eewqe+KJ/HzIIXGnEZGtUB0KRaGyPtco1yy6A3Pdfb67bwTGAscWaeNA4YEIjYHvEsbXN7OaQF1gI7A6wqzbZvPmYO+nRo1g+PC404iIRCbKbRatgEUJ9xcD+xdpMwR408wuAOoDh4XjnycoLEuBesBAd19e9AHMrD/QH6BFixbk5OSUY/zk2owdS4cpU5h17bX8OGsWeXl5FZ6hPCl/vJS/YjVu3Jg1a9YAkJ+fv2W4oixbtoxjjjkGgB9++IGMjAyaN28OwOTJk8nMzEy6jPPOO49LLrmEnXfeOaX869ev3/b3yN0juQEnAA8n3O8LDC/S5hLg0nA4C5hFsLZzEDAGqAXsAMwBdi7t8bp27eoVavZs99q13f/8Z/eCAnd3nzx5csVmKGfKHy/lr1izZs3aMrx69eqU5vng2w/8pndu8g++/aBcs1x33XV+2223/WZ8QUGB5+fnJ50/1fyJz7kQMNVT+E6Pcs1iCdAm4X7rcFyis4DeAO6ea2Z1gObAKcAb7r4J+NHM3ge6AfMjzJu6/Pzg1OP16sG//w3VqN9TpCq6YvIVzFo+q9Q2qzasYuYPMynwAmpYDfZusTeNazcusf2+v9uXu3vfvdVZ5s6dyzHHHEPnzp2ZPn06EyZM4Prrr+eTTz5h3bp1/O1vf2Pw4MEA/OEPf2D48OG0bduWJk2acO655/L6669Tr149XnnlFXbYYYetfvySRLnNYgqwi5m1N7NMgg3YrxZp8y3QC8DMfg/UAX4Kx/cMx9cHDgBmR5h16wwfDh98AHffDTvuGHcaEakAq9avosALACjwAlatXxXZY82ePZuBAwcya9YsWrVqxS233MLUqVP59NNPmTBhArNm/bawrVq1ikMPPZRPP/2UrKwsRo0aVa6ZIluzcPfNZjYAGA9kAKPc/QszG0qw2vMqcCnwkJkNJNiofbq7u5mNAEab2ReAAaPdfWZUWbfKvHlw1VXwpz9B375xpxGRcjCsx7Ckx1nkLsql1+O92Ji/kcyMTMYcN4asNlmR5OnQoQPdunXbcv/pp5/mkUceYfPmzXz33XfMmjWLTp1+vXNp3bp16dOnDwBdu3bl3XffLddMkR6U5+7jgHFFxg1OGJ5FsH2i6Hx5BLvPVi4FBXDWWVCrFjz4oLqfRKqRrDZZTDxtIjkLcshulx1ZoQCoX7/+luGvv/6ae+65h48//pgmTZpw6qmnFnu8ROIG8YyMDDZv3lyumXQE99Z44AF4+2146CFo3TruNCJSwbLaZEVaJIqzevVqGjZsSKNGjVi6dCnjx4+nd+/eFZoBVCxSt2ABXH45HH54sHYhIlIBunTpQqdOndh9991p27YtBx30m86YCqFikQp36N8/6HZ66CF1P4lIuRoyZMiW4Y4dOzJjxowt982MJ554otj53nvvPSA4N9TKlSu3jD/ppJM46aSTyjWjikUqHnkEJkwIdpNt2zbuNCIiFU4XP0pm8WK49FLIzoZzzok7jYhILFQsSuMeFIjNm+Hhh6GGXi4RqZ7UDVWaJ56AceOCg+86dIg7jYhIbPRTuSRLl8JFF8FBB8EFF8SdRkQkVioWxXGH886D9euDjdvqfhKRak7dUMV55hl45ZXgCni77RZ3GhGpgpYtW0avXr0A+P7778nIyGD77bcH4OOPP07pFOUAo0aN4pBDDon8srAqFkX9+CMMGADdu8Mll8SdRkQqk9xcyMkJ9o7MKtuR3M2aNdtyPMWQIUNo0KABgwYN2urljBo1it12242OHTuWKU8yKhZFDRgAa9bAqFGQkRF3GhGpALWvuAKKOZPrr6xaBTNnBueIq1ED9t4bGpd8inL23TfYOWYbPPbYY4wYMYKNGzdy4IEHMnz4cAoKCjjjjDOYMWMG7k7//v1p0aIFM2bM4PTTT6d+/fpbtUaytVQsEr3wAjz3HNx4I+yxR9xpRKQyWbUqKBQQ/F21qvRisY0+//xzXnrpJT744ANq1qxJ//79GTt2LB06dODnn3/ms88+A2DlypU0adKE++67j2HDhkV+GhAVi0LLlsH550PnzsE5oESk2tgwbBiZyfr8c3OhVy/YuBEyM2HMmDJ3RRXnrbfeYsqUKVtOUb5u3TratGnDH//4R+bMmcOFF17IkUceyRFHHFHuj10aFYtCF10Ey5fDm28GpyAXEUmUlQUTJ5bbNouSuDtnnnkmN9xww2+mzZw5k9dff50RI0bwwgsvMHLkyEgyFEfFAuC224JfCWeeCfvsE3caEamssrIiKxKFDjvsME444QQuuugimjdvzrJly1i7di1169alTp06nHjiieyyyy7069cPgIYNG5KXlxdpJlCxgPHj/9ft9PTT0K9f5P8MIiIl2Wuvvbjuuus47LDDKCgooFatWjzwwANkZGRw1lln4e6YGcOGDQPgjDPOYMCAAdrAHbl33vnf8MaNwSqmioWIVKDEU5QDnHLKKZxyyim/aTd9+vTfjPvrX/9Knz59Ij/OQocmH3UU1K0b7CabmRn0RYqIyK9ozaKCNlqJiKQzFQuokI1WIlL5FPb/VwfuXqb51Q0lItVSnTp1WLZsWZm/RNOBu7Ns2TLq1KmzzcvQmoWIVEutW7dm8eLF/PTTT6xfv75MX6RxSyV/nTp1aN269TY/hoqFiFRLtWrVon379gDk5OTQuXPnmBNtu4rIr24oERFJSsVCRESSUrEQEZGkrKrsCWBmPwELY47RHPg55gxlofzxUv74pHN2KFv+tu6+fbJGVaZYVAZmNtXdu8WdY1spf7yUPz7pnB0qJr+6oUREJCkVCxERSUrFonxV3JVIoqH88VL++KRzdqiA/NpmISIiSWnNQkREklKxEBGRpFQsSmFmvc1sjpnNNbMri5le28yeCad/ZGbtwvHNzGyymeWZ2fAi82Sa2Ugz+8rMZpvZ8WmW/2Qz+8zMZprZG2bWvJJlP9zMpoUZp5lZz4R5uobj55rZvRbhuanLO7+Z1TOz/4b/M1+Y2S1RZY8if5F5XzWzz9Mtf5p8dkvLX7bPrrvrVswNyADmATsDmcCnQKcibc4HHgiHTwKeCYfrA38AzgWGF5nneuDGcLgG0Dxd8hOcePLHwszArcCQSpa9M9AyHN4TWJIwz8fAAYABrwN9KuFrX2x+oB7QIxzOBN5Np/wJ8x0HPAV8HkX2iP9/0uGzW9L/T5k/u5G8WVXhBmQB4xPuXwVcVaTNeCAr4c34mXCngXDc6fy2WCwC6qdjfqAW8BPQluAL9wGgf2XMHo43YDlQG9gRmJ0w7WTgwcr62hfNX8xj3AOcnU75gQbAe0Anoi0WUeVPm89u0fzl8dlVN1TJWhH8cxRaHI4rto27bwZWAc1KWqCZNQkHbzCzT8zsOTNrUX6Ri88WKnN+d98EnAd8BnxH8KF/pPwi/zZXaFuzHw984u4bwvaLkyyzvESRf4vw/+hoYGI5Zi42W6i88t8A3AH8Ut6BS8oWKnP+NP3sbslfHp9dFYuKVRNoDXzg7l2AXOD2eCOlzsxqEfzDdQZaAjMJfvVUOma2BzAMOCfuLNuipPxmVhN4GrjX3efHkS0VRfOb2b5AB3d/KdZgKSrm9U+rz24xr3+ZP7sqFiVbArRJuN86HFdsm/BD3BhYVsoylxH8qnoxvP8c0KU8whYjivz7Arj7PA/Wc58FDiyvwMXlCm1VdjNrDbwEnObu8xLaJ14mrLhllpco8hcaCXzt7ndHkPs32ULlkT8L6GZmCwi6onY1s5w0yp82n90S8pf5s6tiUbIpwC5m1t7MMgk2Ir1apM2rwD/C4ROASeEbUaxw2mtAdjiqFzCrPEMnKPf8BP+gncys8AyVhwNflmPmQtucPewu+C9wpbu/X9jY3ZcCq83sgHAvqNOAVyLIHkl+ADO7keBL4eKIcheK4vW/391buns7gp0nvnL37DTKnxaf3VL+f8r+2Y16Y00634A/AV8R7Jnwf+G4ocAx4XAdgl8Ycwn2tNk5Yd4FBBuX8gj6HDuF49sC7xCsBk4Edkqz/OeG/2QzCT48zSpTduAaYC0wI+G2QzitG/B5uMzhFNkgWJnzE/y69PC1LxzfL13yF1l2OyLcwB3h/0+l/+wmyV+mz65O9yEiIkmpG0pERJJSsRARkaRULEREJCkVCxERSUrFQkREklKxkGrLgrPrzghv35vZkoT7H0T0mJ3NbKtOs2BmD5tZpyRtBpjZmWVLJ1Iy7TorApjZECDP3SM9hYOZPUdw5tJPy3m59YD33b1zeS5XpJDWLESKYWZ54d9sM3vbzF4xs/lmdouZ/d3MPg6vDdAhbLe9mb1gZlPC20HFLLMhsHdhoTCzIWb2mJm9a2YLzew4M7s1XO4b4fl8MLMcM+tWmMvM/mVmn5rZh4Uns3P3X4AFZta9Yl4hqW5ULESS24fg6NffA32BXd29O/AwcEHY5h7gLnffj+Bsnw8Xs5zCI8gTdQB6AscATwKT3X0vYB1wZDHLqA986O77EBxNfHbCtKnAwVv97ERSUDPuACJpYIoH55bCzOYBb4bjPwN6hMOHEZx7p3CeRmbWwN3zEpazI8E1BRK97u6bzOwzgovevJGw7HbFZNkI/CccnkZwjp9CPwK7b8XzEkmZioVIconXkyhIuF/A/z5DNYAD3H19KctZR3BOn98s290LzGyT/28jYuKyEyW2yS/Spk74GCLlTt1QIuXjTf7XJVV4/YaivgQ6RphhV37bzSVSLlQsRMrHhQTXa5hpZrMItnH8irvPBhqHG7qjcBAwIaJlSzWnXWdFKpCZDQTWuHtxG8DLstzOwCXu3rc8lytSSGsWIhXrfn69DaS8NAeujWC5IoDWLEREJAVasxARkaRULEREJCkVCxERSUrFQkREklKxEBGRpP4fpmvRDGDMKMMAAAAASUVORK5CYII=\n",
-      "text/plain": [
-       "<Figure size 432x288 with 1 Axes>"
-      ]
-     },
-     "metadata": {
-      "needs_background": "light"
-     },
-     "output_type": "display_data"
-    }
-   ],
-   "source": [
-    "# get time\n",
-    "time_proxy = %sql SELECT metrics_elapsed_time FROM iris_model_summary;\n",
-    "\n",
-    "# reshape to np arrays\n",
-    "time = np.array(time_proxy).reshape(num_points)/60.0\n",
-    "\n",
-    "#plot\n",
-    "plt.title('Iris validation accuracy by time')\n",
-    "plt.xlabel('Time (min)')\n",
-    "plt.ylabel('Accuracy')\n",
-    "plt.grid(True)\n",
-    "plt.plot(time, train_accuracy, 'g.-', label='Train')\n",
-    "plt.plot(time, test_accuracy, 'r.-', label='Test')\n",
-    "plt.legend()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Time to achieve a given accuracy"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 29,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x12f344210>"
-      ]
-     },
-     "execution_count": 29,
-     "metadata": {},
-     "output_type": "execute_result"
-    },
-    {
-     "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": [
-    "#plot\n",
-    "plt.title('Iris time by validation accuracy')\n",
-    "plt.xlabel('Accuracy')\n",
-    "plt.ylabel('Time (min)')\n",
-    "plt.grid(True)\n",
-    "plt.plot(train_accuracy, time, 'g.-', label='Train')\n",
-    "plt.plot(test_accuracy, time, 'r.-', label='Test')\n",
-    "plt.legend()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"pred_prob\"></a>\n",
-    "# 2. Predict probabilities\n",
-    "Predict with probabilities for each class:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 30,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "30 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>prob_Iris-setosa</th>\n",
-       "        <th>prob_Iris-versicolor</th>\n",
-       "        <th>prob_Iris-virginica</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>0.89789814</td>\n",
-       "        <td>0.0880069</td>\n",
-       "        <td>0.014094983</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>0.90666765</td>\n",
-       "        <td>0.081442654</td>\n",
-       "        <td>0.011889744</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>0.8795763</td>\n",
-       "        <td>0.1017618</td>\n",
-       "        <td>0.018661851</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>14</td>\n",
-       "        <td>0.8874597</td>\n",
-       "        <td>0.095630445</td>\n",
-       "        <td>0.016909808</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>18</td>\n",
-       "        <td>0.9102227</td>\n",
-       "        <td>0.078691445</td>\n",
-       "        <td>0.011085836</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>28</td>\n",
-       "        <td>0.9124432</td>\n",
-       "        <td>0.077006854</td>\n",
-       "        <td>0.010549883</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>44</td>\n",
-       "        <td>0.90255314</td>\n",
-       "        <td>0.08451119</td>\n",
-       "        <td>0.012935703</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>48</td>\n",
-       "        <td>0.89486533</td>\n",
-       "        <td>0.09027297</td>\n",
-       "        <td>0.014861753</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>54</td>\n",
-       "        <td>0.026524143</td>\n",
-       "        <td>0.51825184</td>\n",
-       "        <td>0.45522407</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>56</td>\n",
-       "        <td>0.020466398</td>\n",
-       "        <td>0.538594</td>\n",
-       "        <td>0.4409396</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>69</td>\n",
-       "        <td>0.009856132</td>\n",
-       "        <td>0.38160574</td>\n",
-       "        <td>0.60853815</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>80</td>\n",
-       "        <td>0.088389054</td>\n",
-       "        <td>0.68402624</td>\n",
-       "        <td>0.22758465</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>83</td>\n",
-       "        <td>0.04700892</td>\n",
-       "        <td>0.6974011</td>\n",
-       "        <td>0.25559002</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>85</td>\n",
-       "        <td>0.02379873</td>\n",
-       "        <td>0.53655416</td>\n",
-       "        <td>0.4396471</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>88</td>\n",
-       "        <td>0.014446292</td>\n",
-       "        <td>0.48625696</td>\n",
-       "        <td>0.4992967</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>89</td>\n",
-       "        <td>0.045492876</td>\n",
-       "        <td>0.6929876</td>\n",
-       "        <td>0.26151955</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>90</td>\n",
-       "        <td>0.032893542</td>\n",
-       "        <td>0.57819253</td>\n",
-       "        <td>0.38891393</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>94</td>\n",
-       "        <td>0.078232706</td>\n",
-       "        <td>0.6571468</td>\n",
-       "        <td>0.26462048</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>97</td>\n",
-       "        <td>0.036127776</td>\n",
-       "        <td>0.6550248</td>\n",
-       "        <td>0.30884746</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>103</td>\n",
-       "        <td>0.0017510698</td>\n",
-       "        <td>0.222155</td>\n",
-       "        <td>0.7760939</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>105</td>\n",
-       "        <td>0.001789655</td>\n",
-       "        <td>0.1864191</td>\n",
-       "        <td>0.81179124</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>111</td>\n",
-       "        <td>0.010464086</td>\n",
-       "        <td>0.46730253</td>\n",
-       "        <td>0.52223337</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>120</td>\n",
-       "        <td>0.0034266405</td>\n",
-       "        <td>0.21947922</td>\n",
-       "        <td>0.7770942</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>128</td>\n",
-       "        <td>0.011816905</td>\n",
-       "        <td>0.4490503</td>\n",
-       "        <td>0.5391328</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>131</td>\n",
-       "        <td>0.0009711725</td>\n",
-       "        <td>0.17797765</td>\n",
-       "        <td>0.82105124</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>132</td>\n",
-       "        <td>0.0024771853</td>\n",
-       "        <td>0.395437</td>\n",
-       "        <td>0.6020858</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>133</td>\n",
-       "        <td>0.0019595844</td>\n",
-       "        <td>0.18066718</td>\n",
-       "        <td>0.8173732</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>136</td>\n",
-       "        <td>0.0012088934</td>\n",
-       "        <td>0.20632832</td>\n",
-       "        <td>0.7924628</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>138</td>\n",
-       "        <td>0.0045178244</td>\n",
-       "        <td>0.3225387</td>\n",
-       "        <td>0.6729434</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>149</td>\n",
-       "        <td>0.006746102</td>\n",
-       "        <td>0.3544818</td>\n",
-       "        <td>0.6387721</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(7, 0.89789814, 0.0880069, 0.014094983),\n",
-       " (8, 0.90666765, 0.081442654, 0.011889744),\n",
-       " (9, 0.8795763, 0.1017618, 0.018661851),\n",
-       " (14, 0.8874597, 0.095630445, 0.016909808),\n",
-       " (18, 0.9102227, 0.078691445, 0.011085836),\n",
-       " (28, 0.9124432, 0.077006854, 0.010549883),\n",
-       " (44, 0.90255314, 0.08451119, 0.012935703),\n",
-       " (48, 0.89486533, 0.09027297, 0.014861753),\n",
-       " (54, 0.026524143, 0.51825184, 0.45522407),\n",
-       " (56, 0.020466398, 0.538594, 0.4409396),\n",
-       " (69, 0.009856132, 0.38160574, 0.60853815),\n",
-       " (80, 0.088389054, 0.68402624, 0.22758465),\n",
-       " (83, 0.04700892, 0.6974011, 0.25559002),\n",
-       " (85, 0.02379873, 0.53655416, 0.4396471),\n",
-       " (88, 0.014446292, 0.48625696, 0.4992967),\n",
-       " (89, 0.045492876, 0.6929876, 0.26151955),\n",
-       " (90, 0.032893542, 0.57819253, 0.38891393),\n",
-       " (94, 0.078232706, 0.6571468, 0.26462048),\n",
-       " (97, 0.036127776, 0.6550248, 0.30884746),\n",
-       " (103, 0.0017510698, 0.222155, 0.7760939),\n",
-       " (105, 0.001789655, 0.1864191, 0.81179124),\n",
-       " (111, 0.010464086, 0.46730253, 0.52223337),\n",
-       " (120, 0.0034266405, 0.21947922, 0.7770942),\n",
-       " (128, 0.011816905, 0.4490503, 0.5391328),\n",
-       " (131, 0.0009711725, 0.17797765, 0.82105124),\n",
-       " (132, 0.0024771853, 0.395437, 0.6020858),\n",
-       " (133, 0.0019595844, 0.18066718, 0.8173732),\n",
-       " (136, 0.0012088934, 0.20632832, 0.7924628),\n",
-       " (138, 0.0045178244, 0.3225387, 0.6729434),\n",
-       " (149, 0.006746102, 0.3544818, 0.6387721)]"
-      ]
-     },
-     "execution_count": 30,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_predict;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_predict('iris_model',      -- model\n",
-    "                                   'iris_test',       -- test_table\n",
-    "                                   'id',              -- id column\n",
-    "                                   'attributes',      -- independent var\n",
-    "                                   'iris_predict',    -- output table\n",
-    "                                   'prob'             -- response type\n",
-    "                                   );\n",
-    "\n",
-    "SELECT * FROM iris_predict ORDER BY id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"warm_start\"></a>\n",
-    "# 3. Warm start\n",
-    "Next, use the warm_start parameter to continue learning, using the coefficients from the run above. Note that we don't drop the model table or model summary table:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 31,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>madlib_keras_fit</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td></td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[('',)]"
-      ]
-     },
-     "execution_count": 31,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
-    "                               'iris_model',          -- model output table\n",
-    "                               'model_arch_library',  -- model arch table\n",
-    "                                1,                    -- model arch id\n",
-    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
-    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
-    "                                10,                   -- num_iterations\n",
-    "                                FALSE,                -- use GPUs\n",
-    "                                'iris_test_packed',   -- validation dataset\n",
-    "                                2,                    -- metrics compute frequency\n",
-    "                                TRUE,                 -- warm start\n",
-    "                               'Sophie L.',           -- name \n",
-    "                               'Simple MLP for iris dataset'  -- description\n",
-    "                              );"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "In the summary table and plots below note that the loss and accuracy values pick up from where the previous run left off:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 32,
-   "metadata": {
-    "scrolled": true
-   },
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>model</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>validation_table</th>\n",
-       "        <th>metrics_compute_frequency</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>madlib_version</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "        <th>metrics_iters</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_train_packed</td>\n",
-       "        <td>iris_model</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>model_arch_library</td>\n",
-       "        <td>1</td>\n",
-       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
-       "        <td> batch_size=5, epochs=3 </td>\n",
-       "        <td>10</td>\n",
-       "        <td>iris_test_packed</td>\n",
-       "        <td>2</td>\n",
-       "        <td>Sophie L.</td>\n",
-       "        <td>Simple MLP for iris dataset</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>2019-12-18 18:09:27.128581</td>\n",
-       "        <td>2019-12-18 18:09:28.838569</td>\n",
-       "        <td>[0.982600927352905, 1.11963605880737, 1.24473285675049, 1.41093587875366, 1.70990204811096]</td>\n",
-       "        <td>1.17-dev</td>\n",
-       "        <td>3</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.983333349228</td>\n",
-       "        <td>0.198354303837</td>\n",
-       "        <td>[0.966666638851166, 0.983333349227905, 0.975000023841858, 0.983333349227905, 0.983333349227905]</td>\n",
-       "        <td>[0.27795821428299, 0.251547634601593, 0.231610581278801, 0.213408783078194, 0.198354303836823]</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.255444854498</td>\n",
-       "        <td>[0.933333337306976, 0.966666638851166, 0.933333337306976, 0.966666638851166, 0.966666638851166]</td>\n",
-       "        <td>[0.333956837654114, 0.309911340475082, 0.291009396314621, 0.271284729242325, 0.25544485449791]</td>\n",
-       "        <td>[2, 4, 6, 8, 10]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_train_packed', u'iris_model', u'class_text', u'attributes', u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', 2, u'Sophie L.', u'Simple MLP for iris dataset', u'madlib_keras', 0.7900390625, datetime.datetime(2019, 12, 18, 18, 9, 27, 128581), datetime.datetime(2019, 12, 18, 18, 9, 28, 838569), [0.982600927352905, 1.11963605880737, 1.24473285675049, 1.41093587875366, 1.70990204811096], u'1.17-dev', 3, [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], u'character varying', 1.0, [u'accuracy'], 0.983333349228, 0.198354303837, [0.966666638851166, 0.983333349227905, 0.975000023841858, 0.983333349227905, 0.983333349227905], [0.27795821428299, 0.251547634601593, 0.231610581278801, 0.213408783078194, 0.198354303836823], 0.966666638851, 0.255444854498, [0.933333337306976, 0.966666638851166, 0.933333337306976, 0.966666638851166, 0.966666638851166], [0.333956837654114, 0.309911340475082, 0.291009396314621, 0.271284729242325, 0.25544485449791], [2, 4, 6, 8, 10])]"
-      ]
-     },
-     "execution_count": 32,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_model_summary;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 33,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x12f4c0110>"
-      ]
-     },
-     "execution_count": 33,
-     "metadata": {},
-     "output_type": "execute_result"
-    },
-    {
-     "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": [
-    "import pandas as pd\n",
-    "import numpy as np\n",
-    "import sys\n",
-    "import os\n",
-    "from matplotlib import pyplot as plt\n",
-    "\n",
-    "# get accuracy and iteration number\n",
-    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
-    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
-    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
-    "\n",
-    "# get number of points\n",
-    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
-    "num_points = num_points_proxy[0]\n",
-    "\n",
-    "# reshape to np arrays\n",
-    "iters = np.array(iters_proxy).reshape(num_points)\n",
-    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
-    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
-    "\n",
-    "#plot\n",
-    "plt.title('Iris validation accuracy by iteration - warm start')\n",
-    "plt.xlabel('Iteration number')\n",
-    "plt.ylabel('Accuracy')\n",
-    "plt.grid(True)\n",
-    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
-    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
-    "plt.legend()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 34,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x12f560f10>"
-      ]
-     },
-     "execution_count": 34,
-     "metadata": {},
-     "output_type": "execute_result"
-    },
-    {
-     "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": [
-    "# get loss\n",
-    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
-    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
-    "\n",
-    "# reshape to np arrays\n",
-    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
-    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
-    "\n",
-    "#plot\n",
-    "plt.title('Iris validation loss by iteration - warm start')\n",
-    "plt.xlabel('Iteration number')\n",
-    "plt.ylabel('Loss')\n",
-    "plt.grid(True)\n",
-    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
-    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
-    "plt.legend()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"transfer_learn\"></a>\n",
-    "# Transfer learning\n",
-    "\n",
-    "<a id=\"load2\"></a>\n",
-    "# 1. Define and load model architecture with some layers frozen\n",
-    "Here we want to start with initial weights from a pre-trained model rather than training from scratch.  We also want to use a model architecture with the earlier feature layer(s) frozen to save on training time.  The example below is somewhat contrived but gives you the idea of the steps.\n",
-    "\n",
-    "First define a model architecture with the 1st hidden layer frozen:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 35,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "dense_4 (Dense)              (None, 10)                50        \n",
-      "_________________________________________________________________\n",
-      "dense_5 (Dense)              (None, 10)                110       \n",
-      "_________________________________________________________________\n",
-      "dense_6 (Dense)              (None, 3)                 33        \n",
-      "=================================================================\n",
-      "Total params: 193\n",
-      "Trainable params: 143\n",
-      "Non-trainable params: 50\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "model_transfer = Sequential()\n",
-    "model_transfer.add(Dense(10, activation='relu', input_shape=(4,), trainable=False))\n",
-    "model_transfer.add(Dense(10, activation='relu'))\n",
-    "model_transfer.add(Dense(3, activation='softmax'))\n",
-    "    \n",
-    "model_transfer.summary()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 36,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": false, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
-      ]
-     },
-     "execution_count": 36,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "model_transfer.to_json()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Load transfer model into model architecture table"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 37,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "2 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_id</th>\n",
-       "        <th>model_arch</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
-       "        <td>Sophie</td>\n",
-       "        <td>A simple model</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': False, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
-       "        <td>Maria</td>\n",
-       "        <td>A transfer model</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Sophie', u'A simple model'),\n",
-       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': False, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Maria', u'A transfer model')]"
-      ]
-     },
-     "execution_count": 37,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,                      \n",
-    "$$\n",
-    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": false, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
-    "$$\n",
-    "::json,         -- JSON blob\n",
-    "                               NULL,                  -- Weights\n",
-    "                               'Maria',               -- Name\n",
-    "                               'A transfer model'     -- Descr\n",
-    ");\n",
-    "\n",
-    "SELECT model_id, model_arch, name, description FROM model_arch_library ORDER BY model_id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"train2\"></a>\n",
-    "# 2. Train transfer model\n",
-    "\n",
-    "Fetch the weights from a previous MADlib run.  (Normally these would be downloaded from a source that trained the same model architecture on a related dataset.)"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 38,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "[]"
-      ]
-     },
-     "execution_count": 38,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "UPDATE model_arch_library \n",
-    "SET model_weights = iris_model.model_weights \n",
-    "FROM iris_model \n",
-    "WHERE model_arch_library.model_id = 2;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Now train the model using the transfer model and the pre-trained weights:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 39,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>madlib_keras_fit</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td></td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[('',)]"
-      ]
-     },
-     "execution_count": 39,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
-    "                               'iris_model',          -- model output table\n",
-    "                               'model_arch_library',  -- model arch table\n",
-    "                                2,                    -- model arch id\n",
-    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
-    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
-    "                                10,                   -- num_iterations\n",
-    "                                FALSE,                -- use GPUs\n",
-    "                                'iris_test_packed',   -- validation dataset\n",
-    "                                2                     -- metrics compute frequency\n",
-    "                              );"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 40,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>model</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>validation_table</th>\n",
-       "        <th>metrics_compute_frequency</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>madlib_version</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "        <th>metrics_iters</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_train_packed</td>\n",
-       "        <td>iris_model</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>model_arch_library</td>\n",
-       "        <td>2</td>\n",
-       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
-       "        <td> batch_size=5, epochs=3 </td>\n",
-       "        <td>10</td>\n",
-       "        <td>iris_test_packed</td>\n",
-       "        <td>2</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>2019-12-18 18:09:32.439417</td>\n",
-       "        <td>2019-12-18 18:09:34.068824</td>\n",
-       "        <td>[0.853152990341187, 0.990938901901245, 1.11821985244751, 1.24195981025696, 1.62932586669922]</td>\n",
-       "        <td>1.17-dev</td>\n",
-       "        <td>3</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.983333349228</td>\n",
-       "        <td>0.155750438571</td>\n",
-       "        <td>[0.983333349227905, 0.983333349227905, 0.975000023841858, 0.975000023841858, 0.983333349227905]</td>\n",
-       "        <td>[0.187174424529076, 0.17763115465641, 0.169175431132317, 0.161857321858406, 0.155750438570976]</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.211615949869</td>\n",
-       "        <td>[0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
-       "        <td>[0.244408145546913, 0.234545931220055, 0.225818797945976, 0.218266576528549, 0.211615949869156]</td>\n",
-       "        <td>[2, 4, 6, 8, 10]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_train_packed', u'iris_model', u'class_text', u'attributes', u'model_arch_library', 2, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', 2, None, None, u'madlib_keras', 0.7900390625, datetime.datetime(2019, 12, 18, 18, 9, 32, 439417), datetime.datetime(2019, 12, 18, 18, 9, 34, 68824), [0.853152990341187, 0.990938901901245, 1.11821985244751, 1.24195981025696, 1.62932586669922], u'1.17-dev', 3, [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], u'character varying', 1.0, [u'accuracy'], 0.983333349228, 0.155750438571, [0.983333349227905, 0.983333349227905, 0.975000023841858, 0.975000023841858, 0.983333349227905], [0.187174424529076, 0.17763115465641, 0.169175431132317, 0.161857321858406, 0.155750438570976], 0.966666638851, 0.211615949869, [0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166], [0.244408145546913, 0.234545931220055, 0.225818797945976, 0.218266576528549, 0.211615949869156], [2, 4, 6, 8, 10])]"
-      ]
-     },
-     "execution_count": 40,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_model_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Note loss picks up from where the last training left off:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 41,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x12f85fdd0>"
-      ]
-     },
-     "execution_count": 41,
-     "metadata": {},
-     "output_type": "execute_result"
-    },
-    {
-     "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": [
-    "import pandas as pd\n",
-    "import numpy as np\n",
-    "import sys\n",
-    "import os\n",
-    "from matplotlib import pyplot as plt\n",
-    "\n",
-    "# get accuracy and iteration number\n",
-    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
-    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
-    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
-    "\n",
-    "# get number of points\n",
-    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
-    "num_points = num_points_proxy[0]\n",
-    "\n",
-    "# reshape to np arrays\n",
-    "iters = np.array(iters_proxy).reshape(num_points)\n",
-    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
-    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
-    "\n",
-    "#plot\n",
-    "plt.title('Iris validation accuracy by iteration - transfer learn')\n",
-    "plt.xlabel('Iteration number')\n",
-    "plt.ylabel('Accuracy')\n",
-    "plt.grid(True)\n",
-    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
-    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
-    "plt.legend()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 42,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x12f8d5990>"
-      ]
-     },
-     "execution_count": 42,
-     "metadata": {},
-     "output_type": "execute_result"
-    },
-    {
-     "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": [
-    "# get loss\n",
-    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
-    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
-    "\n",
-    "# reshape to np arrays\n",
-    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
-    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
-    "\n",
-    "#plot\n",
-    "plt.title('Iris validation loss by iteration - transfer learn')\n",
-    "plt.xlabel('Iteration number')\n",
-    "plt.ylabel('Loss')\n",
-    "plt.grid(True)\n",
-    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
-    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
-    "plt.legend()"
-   ]
-  }
- ],
- "metadata": {
-  "kernelspec": {
-   "display_name": "Python 2",
-   "language": "python",
-   "name": "python2"
-  },
-  "language_info": {
-   "codemirror_mode": {
-    "name": "ipython",
-    "version": 2
-   },
-   "file_extension": ".py",
-   "mimetype": "text/x-python",
-   "name": "python",
-   "nbconvert_exporter": "python",
-   "pygments_lexer": "ipython2",
-   "version": "2.7.10"
-  }
- },
- "nbformat": 4,
- "nbformat_minor": 1
-}
diff --git a/community-artifacts/Deep-learning/MADlib-Keras-cifar10-inference-v1.ipynb b/community-artifacts/Deep-learning/MADlib-Keras-cifar10-inference-v1.ipynb
deleted file mode 100644
index c5de290..0000000
--- a/community-artifacts/Deep-learning/MADlib-Keras-cifar10-inference-v1.ipynb
+++ /dev/null
@@ -1,601 +0,0 @@
-{
- "cells": [
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "# Inference for CIFAR-10 dataset using predict BYOM\n",
-    "The predict BYOM function allows you to do inference using models that have not been trained with MADlib, but rather imported or created elsewhere. It was added in MADlib 1.17.\n",
-    "\n",
-    "In this workbook we train a model in Python using\n",
-    "https://keras.io/examples/cifar10_cnn/\n",
-    "and run inference on the validation set.\n",
-    "\n",
-    "## Table of contents\n",
-    "\n",
-    "<a href=\"#setup\">1. Setup</a>\n",
-    "\n",
-    "<a href=\"#train_model\">2. Train model in Python</a>\n",
-    "\n",
-    "<a href=\"#load_model\">3. Load model into table</a>\n",
-    "\n",
-    "<a href=\"#load_images\">4. Get validation data set and load into table</a>\n",
-    "\n",
-    "<a href=\"#inference\">5. Inference</a>"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"setup\"></a>\n",
-    "# 1. Setup"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 1,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
-   "source": [
-    "%load_ext sql"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 2,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
-    "# Greenplum Database 5.x on GCP - via tunnel\n",
-    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
-    "        \n",
-    "# PostgreSQL local\n",
-    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 3,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>version</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
-      ]
-     },
-     "execution_count": 3,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql select madlib.version();\n",
-    "#%sql select version();"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"train_model\"></a>\n",
-    "# 2. Train model in Python\n",
-    "\n",
-    "Train a model in Python using https://keras.io/examples/cifar10_cnn/"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "from __future__ import print_function\n",
-    "import keras\n",
-    "from keras.datasets import cifar10\n",
-    "from keras.preprocessing.image import ImageDataGenerator\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense, Dropout, Activation, Flatten\n",
-    "from keras.layers import Conv2D, MaxPooling2D\n",
-    "import os\n",
-    "\n",
-    "batch_size = 32\n",
-    "num_classes = 10\n",
-    "epochs = 25\n",
-    "data_augmentation = True\n",
-    "num_predictions = 20\n",
-    "#save_dir = os.path.join(os.getcwd(), 'saved_models')\n",
-    "#model_name = 'keras_cifar10_trained_model.h5'\n",
-    "\n",
-    "# The data, split between train and test sets:\n",
-    "(x_train, y_train), (x_test, y_test) = cifar10.load_data()\n",
-    "print('x_train shape:', x_train.shape)\n",
-    "print(x_train.shape[0], 'train samples')\n",
-    "print(x_test.shape[0], 'test samples')\n",
-    "\n",
-    "# Convert class vectors to binary class matrices.\n",
-    "y_train = keras.utils.to_categorical(y_train, num_classes)\n",
-    "y_test = keras.utils.to_categorical(y_test, num_classes)\n",
-    "\n",
-    "model = Sequential()\n",
-    "model.add(Conv2D(32, (3, 3), padding='same',\n",
-    "                 input_shape=x_train.shape[1:]))\n",
-    "model.add(Activation('relu'))\n",
-    "model.add(Conv2D(32, (3, 3)))\n",
-    "model.add(Activation('relu'))\n",
-    "model.add(MaxPooling2D(pool_size=(2, 2)))\n",
-    "model.add(Dropout(0.25))\n",
-    "\n",
-    "model.add(Conv2D(64, (3, 3), padding='same'))\n",
-    "model.add(Activation('relu'))\n",
-    "model.add(Conv2D(64, (3, 3)))\n",
-    "model.add(Activation('relu'))\n",
-    "model.add(MaxPooling2D(pool_size=(2, 2)))\n",
-    "model.add(Dropout(0.25))\n",
-    "\n",
-    "model.add(Flatten())\n",
-    "model.add(Dense(512))\n",
-    "model.add(Activation('relu'))\n",
-    "model.add(Dropout(0.5))\n",
-    "model.add(Dense(num_classes))\n",
-    "model.add(Activation('softmax'))\n",
-    "\n",
-    "# initiate RMSprop optimizer\n",
-    "opt = keras.optimizers.rmsprop(lr=0.0001, decay=1e-6)\n",
-    "\n",
-    "# Let's train the model using RMSprop\n",
-    "model.compile(loss='categorical_crossentropy',\n",
-    "              optimizer=opt,\n",
-    "              metrics=['accuracy'])\n",
-    "\n",
-    "x_train = x_train.astype('float32')\n",
-    "x_test = x_test.astype('float32')\n",
-    "x_train /= 255\n",
-    "x_test /= 255\n",
-    "\n",
-    "if not data_augmentation:\n",
-    "    print('Not using data augmentation.')\n",
-    "    model.fit(x_train, y_train,\n",
-    "              batch_size=batch_size,\n",
-    "              epochs=epochs,\n",
-    "              validation_data=(x_test, y_test),\n",
-    "              shuffle=True)\n",
-    "else:\n",
-    "    print('Using real-time data augmentation.')\n",
-    "    # This will do preprocessing and realtime data augmentation:\n",
-    "    datagen = ImageDataGenerator(\n",
-    "        featurewise_center=False,  # set input mean to 0 over the dataset\n",
-    "        samplewise_center=False,  # set each sample mean to 0\n",
-    "        featurewise_std_normalization=False,  # divide inputs by std of the dataset\n",
-    "        samplewise_std_normalization=False,  # divide each input by its std\n",
-    "        zca_whitening=False,  # apply ZCA whitening\n",
-    "        zca_epsilon=1e-06,  # epsilon for ZCA whitening\n",
-    "        rotation_range=0,  # randomly rotate images in the range (degrees, 0 to 180)\n",
-    "        # randomly shift images horizontally (fraction of total width)\n",
-    "        width_shift_range=0.1,\n",
-    "        # randomly shift images vertically (fraction of total height)\n",
-    "        height_shift_range=0.1,\n",
-    "        shear_range=0.,  # set range for random shear\n",
-    "        zoom_range=0.,  # set range for random zoom\n",
-    "        channel_shift_range=0.,  # set range for random channel shifts\n",
-    "        # set mode for filling points outside the input boundaries\n",
-    "        fill_mode='nearest',\n",
-    "        cval=0.,  # value used for fill_mode = \"constant\"\n",
-    "        horizontal_flip=True,  # randomly flip images\n",
-    "        vertical_flip=False,  # randomly flip images\n",
-    "        # set rescaling factor (applied before any other transformation)\n",
-    "        rescale=None,\n",
-    "        # set function that will be applied on each input\n",
-    "        preprocessing_function=None,\n",
-    "        # image data format, either \"channels_first\" or \"channels_last\"\n",
-    "        data_format=None,\n",
-    "        # fraction of images reserved for validation (strictly between 0 and 1)\n",
-    "        validation_split=0.0)\n",
-    "\n",
-    "    # Compute quantities required for feature-wise normalization\n",
-    "    # (std, mean, and principal components if ZCA whitening is applied).\n",
-    "    datagen.fit(x_train)\n",
-    "\n",
-    "    # Fit the model on the batches generated by datagen.flow().\n",
-    "    model.fit_generator(datagen.flow(x_train, y_train,\n",
-    "                                     batch_size=batch_size),\n",
-    "                        epochs=epochs,\n",
-    "                        validation_data=(x_test, y_test),\n",
-    "                        workers=4)\n",
-    "\n",
-    "# Save model and weights\n",
-    "#if not os.path.isdir(save_dir):\n",
-    "#    os.makedirs(save_dir)\n",
-    "#model_path = os.path.join(save_dir, model_name)\n",
-    "#model.save(model_path)\n",
-    "#print('Saved trained model at %s ' % model_path)\n",
-    "\n",
-    "# Score trained model.\n",
-    "scores = model.evaluate(x_test, y_test, verbose=1)\n",
-    "print('Test loss:', scores[0])\n",
-    "print('Test accuracy:', scores[1])"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "model.to_json()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load_model\"></a>\n",
-    "# 3.  Load model into table\n",
-    "\n",
-    "Load the model architecture and weights into the model architecture table"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "import psycopg2 as p2\n",
-    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
-    "#conn = p2.connect('postgresql://fmcquillan@localhost:5432/madlib')\n",
-    "cur = conn.cursor()\n",
-    "\n",
-    "from keras.layers import *\n",
-    "from keras import Sequential\n",
-    "import numpy as np\n",
-    "\n",
-    "# get weights, flatten and serialize\n",
-    "weights = model.get_weights()\n",
-    "weights_flat = [w.flatten() for w in weights]\n",
-    "weights1d =  np.concatenate(weights_flat).ravel()\n",
-    "weights_bytea = p2.Binary(weights1d.tostring())\n",
-    "\n",
-    "query = \"SELECT madlib.load_keras_model('model_arch_library_cifar10', %s,%s,%s,%s)\"\n",
-    "cur.execute(query,[model.to_json(), weights_bytea, \"CIFAR10 model\", \"CNN model with weights trained on CIFAR10.\"])\n",
-    "conn.commit()\n",
-    "\n",
-    "# check weights loaded OK\n",
-    "%sql SELECT model_id, name, description FROM model_arch_library_cifar10;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load_images\"></a>\n",
-    "# 4. Get validation data set and load into table\n",
-    "\n",
-    "First set up image loader using the script called <em>madlib_image_loader.py</em> located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "import sys\n",
-    "import os\n",
-    "madlib_site_dir = '/Users/fmcquillan/Documents/Product/MADlib/Demos/data'\n",
-    "sys.path.append(madlib_site_dir)\n",
-    "\n",
-    "# Import image loader module\n",
-    "from madlib_image_loader import ImageLoader, DbCredentials\n",
-    "\n",
-    "# Specify database credentials, for connecting to db\n",
-    "#db_creds = DbCredentials(user='fmcquillan',\n",
-    "#                         host='localhost',\n",
-    "#                         port='5432',\n",
-    "#                         password='')\n",
-    "\n",
-    "# Specify database credentials, for connecting to db\n",
-    "db_creds = DbCredentials(user='gpadmin', \n",
-    "                         db_name='madlib',\n",
-    "                         host='localhost',\n",
-    "                         port='8000',\n",
-    "                         password='')\n",
-    "\n",
-    "# Initialize ImageLoader (increase num_workers to run faster)\n",
-    "iloader = ImageLoader(num_workers=5, db_creds=db_creds)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Next load CIFAR-10 data from Keras consisting of 50,000 32x32 color training images, labeled over 10 categories, and 10,000 test images."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "from keras.datasets import cifar10\n",
-    "\n",
-    "# Load dataset into np array\n",
-    "(x_train, y_train), (x_test, y_test) = cifar10.load_data()\n",
-    "\n",
-    "%sql DROP TABLE IF EXISTS cifar_10_test_data;\n",
-    "\n",
-    "# Save images to temporary directories and load into database\n",
-    "#iloader.load_dataset_from_np(x_train, y_train, 'cifar_10_train_data', append=False)\n",
-    "iloader.load_dataset_from_np(x_test, y_test, 'cifar_10_test_data', append=False)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"inference\"></a>\n",
-    "# 5. Inference"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 4,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "10 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>estimated_dependent_var</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>3</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>8</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>6</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>5</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>4</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>5</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>5</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>8</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, u'3'),\n",
-       " (2, u'8'),\n",
-       " (3, u'1'),\n",
-       " (4, u'6'),\n",
-       " (5, u'5'),\n",
-       " (6, u'4'),\n",
-       " (7, u'5'),\n",
-       " (8, u'5'),\n",
-       " (9, u'0'),\n",
-       " (10, u'8')]"
-      ]
-     },
-     "execution_count": 4,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS cifar10_predict_byom;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_predict_byom('model_arch_library_cifar10',  -- model arch table\n",
-    "                                         1,                            -- model arch id\n",
-    "                                        'cifar_10_test_data',          -- test_table\n",
-    "                                        'id',                          -- id column\n",
-    "                                        'x',                           -- independent var\n",
-    "                                        'cifar10_predict_byom',        -- output table\n",
-    "                                        'response',                    -- prediction type\n",
-    "                                         FALSE,                        -- use gpus\n",
-    "                                         NULL,                         -- class values\n",
-    "                                         255.0                         -- normalizing const\n",
-    "                                   );\n",
-    "SELECT * FROM cifar10_predict_byom ORDER BY id LIMIT 10;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Number of missclassifications:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 5,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2551</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(2551L,)]"
-      ]
-     },
-     "execution_count": 5,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT COUNT(*) FROM cifar10_predict_byom JOIN cifar_10_test_data USING (id)\n",
-    "WHERE cifar10_predict_byom.estimated_dependent_var != cifar_10_test_data.y;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Predict accuracy. From https://keras.io/examples/cifar10_cnn/ accuracy claim is 75% on validation set after 25 epochs.  From run above test accuracy: 0.7449.  MADlib predict BYOM accuracy matches:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 6,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>test_accuracy_percent</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>74.49</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(Decimal('74.49'),)]"
-      ]
-     },
-     "execution_count": 6,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT round(count(*)*100.0/10000.0, 2) as test_accuracy_percent from\n",
-    "    (select cifar_10_test_data.y as actual, cifar10_predict_byom.estimated_dependent_var as estimated\n",
-    "     from cifar10_predict_byom inner join cifar_10_test_data\n",
-    "     on cifar_10_test_data.id=cifar10_predict_byom.id) q\n",
-    "WHERE q.actual=q.estimated;"
-   ]
-  }
- ],
- "metadata": {
-  "kernelspec": {
-   "display_name": "Python 2",
-   "language": "python",
-   "name": "python2"
-  },
-  "language_info": {
-   "codemirror_mode": {
-    "name": "ipython",
-    "version": 2
-   },
-   "file_extension": ".py",
-   "mimetype": "text/x-python",
-   "name": "python",
-   "nbconvert_exporter": "python",
-   "pygments_lexer": "ipython2",
-   "version": "2.7.10"
-  }
- },
- "nbformat": 4,
- "nbformat_minor": 2
-}
diff --git a/community-artifacts/Deep-learning/MADlib-Keras-model-selection-MLP-v1.ipynb b/community-artifacts/Deep-learning/MADlib-Keras-model-selection-MLP-v1.ipynb
deleted file mode 100644
index cfe8c97..0000000
--- a/community-artifacts/Deep-learning/MADlib-Keras-model-selection-MLP-v1.ipynb
+++ /dev/null
@@ -1,5709 +0,0 @@
-{
- "cells": [
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "# Model Selection for Multilayer Perceptron Using Keras and MADlib\n",
-    "\n",
-    "E2E classification example using MADlib calling a Keras MLP for different hyperparameters and model architectures.\n",
-    "\n",
-    "Deep learning works best on very large datasets, but that is not convenient for a quick introduction to the syntax.  So in this workbook we use the well known iris data set from https://archive.ics.uci.edu/ml/datasets/iris to help get you started.  It is similar to the example in user docs http://madlib.apache.org/docs/latest/index.html\n",
-    "\n",
-    "For more realistic examples please refer to the deep learning notebooks at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts\n",
-    "\n",
-    "## Table of contents\n",
-    "\n",
-    "<a href=\"#class\">Classification</a>\n",
-    "\n",
-    "* <a href=\"#create_input_data\">1. Create input data</a>\n",
-    "\n",
-    "* <a href=\"#pp\">2. Call preprocessor for deep learning</a>\n",
-    "\n",
-    "* <a href=\"#load\">3. Define and load model architecture</a>\n",
-    "\n",
-    "* <a href=\"#def_mst\">4. Define and load model selection tuples</a>\n",
-    "\n",
-    "* <a href=\"#train\">5. Train</a>\n",
-    "\n",
-    "* <a href=\"#eval\">6. Evaluate</a>\n",
-    "\n",
-    "* <a href=\"#pred\">7. Predict</a>\n",
-    "\n",
-    "<a href=\"#class2\">Classification with Other Parameters</a>\n",
-    "\n",
-    "* <a href=\"#val_dataset\">1. Validation dataset</a>\n",
-    "\n",
-    "* <a href=\"#pred_prob\">2. Predict probabilities</a>\n",
-    "\n",
-    "* <a href=\"#warm_start\">3. Warm start</a>"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 1,
-   "metadata": {
-    "scrolled": false
-   },
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
-   "source": [
-    "%load_ext sql"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 2,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
-    "# Greenplum Database 5.x on GCP - via tunnel\n",
-    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
-    "        \n",
-    "# PostgreSQL local\n",
-    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 3,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>version</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
-      ]
-     },
-     "execution_count": 3,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql select madlib.version();\n",
-    "#%sql select version();"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"class\"></a>\n",
-    "# Classification\n",
-    "\n",
-    "<a id=\"create_input_data\"></a>\n",
-    "# 1.  Create input data\n",
-    "\n",
-    "Load iris data set."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 4,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "Done.\n",
-      "150 rows affected.\n",
-      "150 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>attributes</th>\n",
-       "        <th>class_text</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>[Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>[Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>[Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>11</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>12</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>13</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>14</td>\n",
-       "        <td>[Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>15</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>16</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>17</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>18</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>19</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>20</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>21</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>22</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>23</td>\n",
-       "        <td>[Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>24</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>25</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>26</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>27</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>28</td>\n",
-       "        <td>[Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>29</td>\n",
-       "        <td>[Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>30</td>\n",
-       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>31</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>32</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>33</td>\n",
-       "        <td>[Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>34</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>35</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>36</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>37</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>38</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>39</td>\n",
-       "        <td>[Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>40</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>41</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>42</td>\n",
-       "        <td>[Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>43</td>\n",
-       "        <td>[Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>44</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>45</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>46</td>\n",
-       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>47</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>48</td>\n",
-       "        <td>[Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>49</td>\n",
-       "        <td>[Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>50</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')]</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>51</td>\n",
-       "        <td>[Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>52</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>53</td>\n",
-       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>54</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>55</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>56</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>57</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>58</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>59</td>\n",
-       "        <td>[Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>60</td>\n",
-       "        <td>[Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>61</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>62</td>\n",
-       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>63</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>64</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>65</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>66</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>67</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>68</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>69</td>\n",
-       "        <td>[Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>70</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>71</td>\n",
-       "        <td>[Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>72</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>73</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>74</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>75</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>76</td>\n",
-       "        <td>[Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>77</td>\n",
-       "        <td>[Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>78</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>79</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>80</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>81</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>82</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>83</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>84</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>85</td>\n",
-       "        <td>[Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>86</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>87</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>88</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>89</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>90</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>91</td>\n",
-       "        <td>[Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>92</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>93</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>94</td>\n",
-       "        <td>[Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>95</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>96</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>97</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>98</td>\n",
-       "        <td>[Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>99</td>\n",
-       "        <td>[Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>100</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')]</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>101</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>102</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>103</td>\n",
-       "        <td>[Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>104</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>105</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>106</td>\n",
-       "        <td>[Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>107</td>\n",
-       "        <td>[Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>108</td>\n",
-       "        <td>[Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>109</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>110</td>\n",
-       "        <td>[Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>111</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>112</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>113</td>\n",
-       "        <td>[Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>114</td>\n",
-       "        <td>[Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>115</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>116</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>117</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>118</td>\n",
-       "        <td>[Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>119</td>\n",
-       "        <td>[Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>120</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>121</td>\n",
-       "        <td>[Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>122</td>\n",
-       "        <td>[Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>123</td>\n",
-       "        <td>[Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>124</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>125</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>126</td>\n",
-       "        <td>[Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>127</td>\n",
-       "        <td>[Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>128</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>129</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>130</td>\n",
-       "        <td>[Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>131</td>\n",
-       "        <td>[Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>132</td>\n",
-       "        <td>[Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>133</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>134</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>135</td>\n",
-       "        <td>[Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>136</td>\n",
-       "        <td>[Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>137</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>138</td>\n",
-       "        <td>[Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>139</td>\n",
-       "        <td>[Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>140</td>\n",
-       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>141</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>142</td>\n",
-       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>143</td>\n",
-       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>144</td>\n",
-       "        <td>[Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>145</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>146</td>\n",
-       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>147</td>\n",
-       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>148</td>\n",
-       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>149</td>\n",
-       "        <td>[Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>150</td>\n",
-       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')]</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (2, [Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (3, [Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (4, [Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (5, [Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (6, [Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (7, [Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (8, [Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (9, [Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (10, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (11, [Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (12, [Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (13, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (14, [Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (15, [Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (16, [Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (17, [Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (18, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (19, [Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (20, [Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (21, [Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (22, [Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (23, [Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (24, [Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')], u'Iris-setosa'),\n",
-       " (25, [Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (26, [Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (27, [Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (28, [Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (29, [Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (30, [Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (31, [Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (32, [Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (33, [Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (34, [Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (35, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (36, [Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (37, [Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (38, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
-       " (39, [Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (40, [Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (41, [Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (42, [Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (43, [Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (44, [Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')], u'Iris-setosa'),\n",
-       " (45, [Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')], u'Iris-setosa'),\n",
-       " (46, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
-       " (47, [Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (48, [Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (49, [Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (50, [Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
-       " (51, [Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (52, [Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (53, [Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (54, [Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (55, [Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (56, [Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (57, [Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')], u'Iris-versicolor'),\n",
-       " (58, [Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (59, [Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (60, [Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (61, [Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (62, [Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (63, [Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (64, [Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (65, [Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (66, [Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (67, [Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (68, [Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (69, [Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (70, [Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')], u'Iris-versicolor'),\n",
-       " (71, [Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')], u'Iris-versicolor'),\n",
-       " (72, [Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (73, [Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (74, [Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (75, [Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (76, [Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (77, [Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (78, [Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')], u'Iris-versicolor'),\n",
-       " (79, [Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (80, [Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (81, [Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')], u'Iris-versicolor'),\n",
-       " (82, [Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (83, [Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (84, [Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')], u'Iris-versicolor'),\n",
-       " (85, [Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (86, [Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')], u'Iris-versicolor'),\n",
-       " (87, [Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')], u'Iris-versicolor'),\n",
-       " (88, [Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (89, [Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (90, [Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (91, [Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (92, [Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')], u'Iris-versicolor'),\n",
-       " (93, [Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (94, [Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
-       " (95, [Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (96, [Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')], u'Iris-versicolor'),\n",
-       " (97, [Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (98, [Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (99, [Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')], u'Iris-versicolor'),\n",
-       " (100, [Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
-       " (101, [Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')], u'Iris-virginica'),\n",
-       " (102, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (103, [Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (104, [Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (105, [Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')], u'Iris-virginica'),\n",
-       " (106, [Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (107, [Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')], u'Iris-virginica'),\n",
-       " (108, [Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (109, [Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (110, [Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')], u'Iris-virginica'),\n",
-       " (111, [Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (112, [Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (113, [Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (114, [Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (115, [Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')], u'Iris-virginica'),\n",
-       " (116, [Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (117, [Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (118, [Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')], u'Iris-virginica'),\n",
-       " (119, [Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (120, [Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')], u'Iris-virginica'),\n",
-       " (121, [Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (122, [Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (123, [Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (124, [Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (125, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (126, [Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (127, [Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (128, [Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (129, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (130, [Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')], u'Iris-virginica'),\n",
-       " (131, [Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (132, [Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (133, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')], u'Iris-virginica'),\n",
-       " (134, [Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')], u'Iris-virginica'),\n",
-       " (135, [Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')], u'Iris-virginica'),\n",
-       " (136, [Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (137, [Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
-       " (138, [Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (139, [Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
-       " (140, [Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')], u'Iris-virginica'),\n",
-       " (141, [Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
-       " (142, [Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (143, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (144, [Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (145, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')], u'Iris-virginica'),\n",
-       " (146, [Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (147, [Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')], u'Iris-virginica'),\n",
-       " (148, [Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')], u'Iris-virginica'),\n",
-       " (149, [Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')], u'Iris-virginica'),\n",
-       " (150, [Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')], u'Iris-virginica')]"
-      ]
-     },
-     "execution_count": 4,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql \n",
-    "DROP TABLE IF EXISTS iris_data;\n",
-    "\n",
-    "CREATE TABLE iris_data(\n",
-    "    id serial,\n",
-    "    attributes numeric[],\n",
-    "    class_text varchar\n",
-    ");\n",
-    "\n",
-    "INSERT INTO iris_data(id, attributes, class_text) VALUES\n",
-    "(1,ARRAY[5.1,3.5,1.4,0.2],'Iris-setosa'),\n",
-    "(2,ARRAY[4.9,3.0,1.4,0.2],'Iris-setosa'),\n",
-    "(3,ARRAY[4.7,3.2,1.3,0.2],'Iris-setosa'),\n",
-    "(4,ARRAY[4.6,3.1,1.5,0.2],'Iris-setosa'),\n",
-    "(5,ARRAY[5.0,3.6,1.4,0.2],'Iris-setosa'),\n",
-    "(6,ARRAY[5.4,3.9,1.7,0.4],'Iris-setosa'),\n",
-    "(7,ARRAY[4.6,3.4,1.4,0.3],'Iris-setosa'),\n",
-    "(8,ARRAY[5.0,3.4,1.5,0.2],'Iris-setosa'),\n",
-    "(9,ARRAY[4.4,2.9,1.4,0.2],'Iris-setosa'),\n",
-    "(10,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
-    "(11,ARRAY[5.4,3.7,1.5,0.2],'Iris-setosa'),\n",
-    "(12,ARRAY[4.8,3.4,1.6,0.2],'Iris-setosa'),\n",
-    "(13,ARRAY[4.8,3.0,1.4,0.1],'Iris-setosa'),\n",
-    "(14,ARRAY[4.3,3.0,1.1,0.1],'Iris-setosa'),\n",
-    "(15,ARRAY[5.8,4.0,1.2,0.2],'Iris-setosa'),\n",
-    "(16,ARRAY[5.7,4.4,1.5,0.4],'Iris-setosa'),\n",
-    "(17,ARRAY[5.4,3.9,1.3,0.4],'Iris-setosa'),\n",
-    "(18,ARRAY[5.1,3.5,1.4,0.3],'Iris-setosa'),\n",
-    "(19,ARRAY[5.7,3.8,1.7,0.3],'Iris-setosa'),\n",
-    "(20,ARRAY[5.1,3.8,1.5,0.3],'Iris-setosa'),\n",
-    "(21,ARRAY[5.4,3.4,1.7,0.2],'Iris-setosa'),\n",
-    "(22,ARRAY[5.1,3.7,1.5,0.4],'Iris-setosa'),\n",
-    "(23,ARRAY[4.6,3.6,1.0,0.2],'Iris-setosa'),\n",
-    "(24,ARRAY[5.1,3.3,1.7,0.5],'Iris-setosa'),\n",
-    "(25,ARRAY[4.8,3.4,1.9,0.2],'Iris-setosa'),\n",
-    "(26,ARRAY[5.0,3.0,1.6,0.2],'Iris-setosa'),\n",
-    "(27,ARRAY[5.0,3.4,1.6,0.4],'Iris-setosa'),\n",
-    "(28,ARRAY[5.2,3.5,1.5,0.2],'Iris-setosa'),\n",
-    "(29,ARRAY[5.2,3.4,1.4,0.2],'Iris-setosa'),\n",
-    "(30,ARRAY[4.7,3.2,1.6,0.2],'Iris-setosa'),\n",
-    "(31,ARRAY[4.8,3.1,1.6,0.2],'Iris-setosa'),\n",
-    "(32,ARRAY[5.4,3.4,1.5,0.4],'Iris-setosa'),\n",
-    "(33,ARRAY[5.2,4.1,1.5,0.1],'Iris-setosa'),\n",
-    "(34,ARRAY[5.5,4.2,1.4,0.2],'Iris-setosa'),\n",
-    "(35,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
-    "(36,ARRAY[5.0,3.2,1.2,0.2],'Iris-setosa'),\n",
-    "(37,ARRAY[5.5,3.5,1.3,0.2],'Iris-setosa'),\n",
-    "(38,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
-    "(39,ARRAY[4.4,3.0,1.3,0.2],'Iris-setosa'),\n",
-    "(40,ARRAY[5.1,3.4,1.5,0.2],'Iris-setosa'),\n",
-    "(41,ARRAY[5.0,3.5,1.3,0.3],'Iris-setosa'),\n",
-    "(42,ARRAY[4.5,2.3,1.3,0.3],'Iris-setosa'),\n",
-    "(43,ARRAY[4.4,3.2,1.3,0.2],'Iris-setosa'),\n",
-    "(44,ARRAY[5.0,3.5,1.6,0.6],'Iris-setosa'),\n",
-    "(45,ARRAY[5.1,3.8,1.9,0.4],'Iris-setosa'),\n",
-    "(46,ARRAY[4.8,3.0,1.4,0.3],'Iris-setosa'),\n",
-    "(47,ARRAY[5.1,3.8,1.6,0.2],'Iris-setosa'),\n",
-    "(48,ARRAY[4.6,3.2,1.4,0.2],'Iris-setosa'),\n",
-    "(49,ARRAY[5.3,3.7,1.5,0.2],'Iris-setosa'),\n",
-    "(50,ARRAY[5.0,3.3,1.4,0.2],'Iris-setosa'),\n",
-    "(51,ARRAY[7.0,3.2,4.7,1.4],'Iris-versicolor'),\n",
-    "(52,ARRAY[6.4,3.2,4.5,1.5],'Iris-versicolor'),\n",
-    "(53,ARRAY[6.9,3.1,4.9,1.5],'Iris-versicolor'),\n",
-    "(54,ARRAY[5.5,2.3,4.0,1.3],'Iris-versicolor'),\n",
-    "(55,ARRAY[6.5,2.8,4.6,1.5],'Iris-versicolor'),\n",
-    "(56,ARRAY[5.7,2.8,4.5,1.3],'Iris-versicolor'),\n",
-    "(57,ARRAY[6.3,3.3,4.7,1.6],'Iris-versicolor'),\n",
-    "(58,ARRAY[4.9,2.4,3.3,1.0],'Iris-versicolor'),\n",
-    "(59,ARRAY[6.6,2.9,4.6,1.3],'Iris-versicolor'),\n",
-    "(60,ARRAY[5.2,2.7,3.9,1.4],'Iris-versicolor'),\n",
-    "(61,ARRAY[5.0,2.0,3.5,1.0],'Iris-versicolor'),\n",
-    "(62,ARRAY[5.9,3.0,4.2,1.5],'Iris-versicolor'),\n",
-    "(63,ARRAY[6.0,2.2,4.0,1.0],'Iris-versicolor'),\n",
-    "(64,ARRAY[6.1,2.9,4.7,1.4],'Iris-versicolor'),\n",
-    "(65,ARRAY[5.6,2.9,3.6,1.3],'Iris-versicolor'),\n",
-    "(66,ARRAY[6.7,3.1,4.4,1.4],'Iris-versicolor'),\n",
-    "(67,ARRAY[5.6,3.0,4.5,1.5],'Iris-versicolor'),\n",
-    "(68,ARRAY[5.8,2.7,4.1,1.0],'Iris-versicolor'),\n",
-    "(69,ARRAY[6.2,2.2,4.5,1.5],'Iris-versicolor'),\n",
-    "(70,ARRAY[5.6,2.5,3.9,1.1],'Iris-versicolor'),\n",
-    "(71,ARRAY[5.9,3.2,4.8,1.8],'Iris-versicolor'),\n",
-    "(72,ARRAY[6.1,2.8,4.0,1.3],'Iris-versicolor'),\n",
-    "(73,ARRAY[6.3,2.5,4.9,1.5],'Iris-versicolor'),\n",
-    "(74,ARRAY[6.1,2.8,4.7,1.2],'Iris-versicolor'),\n",
-    "(75,ARRAY[6.4,2.9,4.3,1.3],'Iris-versicolor'),\n",
-    "(76,ARRAY[6.6,3.0,4.4,1.4],'Iris-versicolor'),\n",
-    "(77,ARRAY[6.8,2.8,4.8,1.4],'Iris-versicolor'),\n",
-    "(78,ARRAY[6.7,3.0,5.0,1.7],'Iris-versicolor'),\n",
-    "(79,ARRAY[6.0,2.9,4.5,1.5],'Iris-versicolor'),\n",
-    "(80,ARRAY[5.7,2.6,3.5,1.0],'Iris-versicolor'),\n",
-    "(81,ARRAY[5.5,2.4,3.8,1.1],'Iris-versicolor'),\n",
-    "(82,ARRAY[5.5,2.4,3.7,1.0],'Iris-versicolor'),\n",
-    "(83,ARRAY[5.8,2.7,3.9,1.2],'Iris-versicolor'),\n",
-    "(84,ARRAY[6.0,2.7,5.1,1.6],'Iris-versicolor'),\n",
-    "(85,ARRAY[5.4,3.0,4.5,1.5],'Iris-versicolor'),\n",
-    "(86,ARRAY[6.0,3.4,4.5,1.6],'Iris-versicolor'),\n",
-    "(87,ARRAY[6.7,3.1,4.7,1.5],'Iris-versicolor'),\n",
-    "(88,ARRAY[6.3,2.3,4.4,1.3],'Iris-versicolor'),\n",
-    "(89,ARRAY[5.6,3.0,4.1,1.3],'Iris-versicolor'),\n",
-    "(90,ARRAY[5.5,2.5,4.0,1.3],'Iris-versicolor'),\n",
-    "(91,ARRAY[5.5,2.6,4.4,1.2],'Iris-versicolor'),\n",
-    "(92,ARRAY[6.1,3.0,4.6,1.4],'Iris-versicolor'),\n",
-    "(93,ARRAY[5.8,2.6,4.0,1.2],'Iris-versicolor'),\n",
-    "(94,ARRAY[5.0,2.3,3.3,1.0],'Iris-versicolor'),\n",
-    "(95,ARRAY[5.6,2.7,4.2,1.3],'Iris-versicolor'),\n",
-    "(96,ARRAY[5.7,3.0,4.2,1.2],'Iris-versicolor'),\n",
-    "(97,ARRAY[5.7,2.9,4.2,1.3],'Iris-versicolor'),\n",
-    "(98,ARRAY[6.2,2.9,4.3,1.3],'Iris-versicolor'),\n",
-    "(99,ARRAY[5.1,2.5,3.0,1.1],'Iris-versicolor'),\n",
-    "(100,ARRAY[5.7,2.8,4.1,1.3],'Iris-versicolor'),\n",
-    "(101,ARRAY[6.3,3.3,6.0,2.5],'Iris-virginica'),\n",
-    "(102,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
-    "(103,ARRAY[7.1,3.0,5.9,2.1],'Iris-virginica'),\n",
-    "(104,ARRAY[6.3,2.9,5.6,1.8],'Iris-virginica'),\n",
-    "(105,ARRAY[6.5,3.0,5.8,2.2],'Iris-virginica'),\n",
-    "(106,ARRAY[7.6,3.0,6.6,2.1],'Iris-virginica'),\n",
-    "(107,ARRAY[4.9,2.5,4.5,1.7],'Iris-virginica'),\n",
-    "(108,ARRAY[7.3,2.9,6.3,1.8],'Iris-virginica'),\n",
-    "(109,ARRAY[6.7,2.5,5.8,1.8],'Iris-virginica'),\n",
-    "(110,ARRAY[7.2,3.6,6.1,2.5],'Iris-virginica'),\n",
-    "(111,ARRAY[6.5,3.2,5.1,2.0],'Iris-virginica'),\n",
-    "(112,ARRAY[6.4,2.7,5.3,1.9],'Iris-virginica'),\n",
-    "(113,ARRAY[6.8,3.0,5.5,2.1],'Iris-virginica'),\n",
-    "(114,ARRAY[5.7,2.5,5.0,2.0],'Iris-virginica'),\n",
-    "(115,ARRAY[5.8,2.8,5.1,2.4],'Iris-virginica'),\n",
-    "(116,ARRAY[6.4,3.2,5.3,2.3],'Iris-virginica'),\n",
-    "(117,ARRAY[6.5,3.0,5.5,1.8],'Iris-virginica'),\n",
-    "(118,ARRAY[7.7,3.8,6.7,2.2],'Iris-virginica'),\n",
-    "(119,ARRAY[7.7,2.6,6.9,2.3],'Iris-virginica'),\n",
-    "(120,ARRAY[6.0,2.2,5.0,1.5],'Iris-virginica'),\n",
-    "(121,ARRAY[6.9,3.2,5.7,2.3],'Iris-virginica'),\n",
-    "(122,ARRAY[5.6,2.8,4.9,2.0],'Iris-virginica'),\n",
-    "(123,ARRAY[7.7,2.8,6.7,2.0],'Iris-virginica'),\n",
-    "(124,ARRAY[6.3,2.7,4.9,1.8],'Iris-virginica'),\n",
-    "(125,ARRAY[6.7,3.3,5.7,2.1],'Iris-virginica'),\n",
-    "(126,ARRAY[7.2,3.2,6.0,1.8],'Iris-virginica'),\n",
-    "(127,ARRAY[6.2,2.8,4.8,1.8],'Iris-virginica'),\n",
-    "(128,ARRAY[6.1,3.0,4.9,1.8],'Iris-virginica'),\n",
-    "(129,ARRAY[6.4,2.8,5.6,2.1],'Iris-virginica'),\n",
-    "(130,ARRAY[7.2,3.0,5.8,1.6],'Iris-virginica'),\n",
-    "(131,ARRAY[7.4,2.8,6.1,1.9],'Iris-virginica'),\n",
-    "(132,ARRAY[7.9,3.8,6.4,2.0],'Iris-virginica'),\n",
-    "(133,ARRAY[6.4,2.8,5.6,2.2],'Iris-virginica'),\n",
-    "(134,ARRAY[6.3,2.8,5.1,1.5],'Iris-virginica'),\n",
-    "(135,ARRAY[6.1,2.6,5.6,1.4],'Iris-virginica'),\n",
-    "(136,ARRAY[7.7,3.0,6.1,2.3],'Iris-virginica'),\n",
-    "(137,ARRAY[6.3,3.4,5.6,2.4],'Iris-virginica'),\n",
-    "(138,ARRAY[6.4,3.1,5.5,1.8],'Iris-virginica'),\n",
-    "(139,ARRAY[6.0,3.0,4.8,1.8],'Iris-virginica'),\n",
-    "(140,ARRAY[6.9,3.1,5.4,2.1],'Iris-virginica'),\n",
-    "(141,ARRAY[6.7,3.1,5.6,2.4],'Iris-virginica'),\n",
-    "(142,ARRAY[6.9,3.1,5.1,2.3],'Iris-virginica'),\n",
-    "(143,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
-    "(144,ARRAY[6.8,3.2,5.9,2.3],'Iris-virginica'),\n",
-    "(145,ARRAY[6.7,3.3,5.7,2.5],'Iris-virginica'),\n",
-    "(146,ARRAY[6.7,3.0,5.2,2.3],'Iris-virginica'),\n",
-    "(147,ARRAY[6.3,2.5,5.0,1.9],'Iris-virginica'),\n",
-    "(148,ARRAY[6.5,3.0,5.2,2.0],'Iris-virginica'),\n",
-    "(149,ARRAY[6.2,3.4,5.4,2.3],'Iris-virginica'),\n",
-    "(150,ARRAY[5.9,3.0,5.1,1.8],'Iris-virginica');\n",
-    "\n",
-    "SELECT * FROM iris_data ORDER BY id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Create a test/validation dataset from the training data"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 5,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>120</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(120L,)]"
-      ]
-     },
-     "execution_count": 5,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_train, iris_test;\n",
-    "\n",
-    "-- Set seed so results are reproducible\n",
-    "SELECT setseed(0);\n",
-    "\n",
-    "SELECT madlib.train_test_split('iris_data',     -- Source table\n",
-    "                               'iris',          -- Output table root name\n",
-    "                                0.8,            -- Train proportion\n",
-    "                                NULL,           -- Test proportion (0.2)\n",
-    "                                NULL,           -- Strata definition\n",
-    "                                NULL,           -- Output all columns\n",
-    "                                NULL,           -- Sample without replacement\n",
-    "                                TRUE            -- Separate output tables\n",
-    "                              );\n",
-    "\n",
-    "SELECT COUNT(*) FROM iris_train;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"pp\"></a>\n",
-    "# 2. Call preprocessor for deep learning\n",
-    "Training dataset (uses training preprocessor):"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 6,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "2 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
-       "        <th>buffer_id</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[60, 4]</td>\n",
-       "        <td>[60, 3]</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[60, 4]</td>\n",
-       "        <td>[60, 3]</td>\n",
-       "        <td>1</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[([60, 4], [60, 3], 0), ([60, 4], [60, 3], 1)]"
-      ]
-     },
-     "execution_count": 6,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_train_packed, iris_train_packed_summary;\n",
-    "\n",
-    "SELECT madlib.training_preprocessor_dl('iris_train',         -- Source table\n",
-    "                                       'iris_train_packed',  -- Output table\n",
-    "                                       'class_text',        -- Dependent variable\n",
-    "                                       'attributes'         -- Independent variable\n",
-    "                                        ); \n",
-    "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM iris_train_packed ORDER BY buffer_id;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 7,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_train</td>\n",
-       "        <td>iris_train_packed</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>60</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>3</td>\n",
-       "        <td>all_segments</td>\n",
-       "        <td>all_segments</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_train', u'iris_train_packed', u'class_text', u'attributes', u'character varying', [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 60, 1.0, 3, 'all_segments', 'all_segments')]"
-      ]
-     },
-     "execution_count": 7,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_train_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Validation dataset (uses validation preprocessor):"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 8,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "2 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
-       "        <th>buffer_id</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[15, 4]</td>\n",
-       "        <td>[15, 3]</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[15, 4]</td>\n",
-       "        <td>[15, 3]</td>\n",
-       "        <td>1</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[([15, 4], [15, 3], 0), ([15, 4], [15, 3], 1)]"
-      ]
-     },
-     "execution_count": 8,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_test_packed, iris_test_packed_summary;\n",
-    "\n",
-    "SELECT madlib.validation_preprocessor_dl('iris_test',          -- Source table\n",
-    "                                         'iris_test_packed',   -- Output table\n",
-    "                                         'class_text',         -- Dependent variable\n",
-    "                                         'attributes',         -- Independent variable\n",
-    "                                         'iris_train_packed'   -- From training preprocessor step\n",
-    "                                          ); \n",
-    "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM iris_test_packed ORDER BY buffer_id;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 9,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_test</td>\n",
-       "        <td>iris_test_packed</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>15</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>3</td>\n",
-       "        <td>all_segments</td>\n",
-       "        <td>all_segments</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_test', u'iris_test_packed', u'class_text', u'attributes', u'character varying', [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 15, 1.0, 3, 'all_segments', 'all_segments')]"
-      ]
-     },
-     "execution_count": 9,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_test_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load\"></a>\n",
-    "# 3. Define and load model architecture\n",
-    "Import Keras libraries"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 10,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "Using TensorFlow backend.\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Couldn't import dot_parser, loading of dot files will not be possible.\n"
-     ]
-    }
-   ],
-   "source": [
-    "import keras\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Define model architecture with 1 hidden layer:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 11,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "dense_1 (Dense)              (None, 10)                50        \n",
-      "_________________________________________________________________\n",
-      "dense_2 (Dense)              (None, 10)                110       \n",
-      "_________________________________________________________________\n",
-      "dense_3 (Dense)              (None, 3)                 33        \n",
-      "=================================================================\n",
-      "Total params: 193\n",
-      "Trainable params: 193\n",
-      "Non-trainable params: 0\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "model1 = Sequential()\n",
-    "model1.add(Dense(10, activation='relu', input_shape=(4,)))\n",
-    "model1.add(Dense(10, activation='relu'))\n",
-    "model1.add(Dense(3, activation='softmax'))\n",
-    "    \n",
-    "model1.summary()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 12,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
-      ]
-     },
-     "execution_count": 12,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "model1.to_json()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Define model architecture with 2 hidden layers:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 13,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "dense_4 (Dense)              (None, 10)                50        \n",
-      "_________________________________________________________________\n",
-      "dense_5 (Dense)              (None, 10)                110       \n",
-      "_________________________________________________________________\n",
-      "dense_6 (Dense)              (None, 10)                110       \n",
-      "_________________________________________________________________\n",
-      "dense_7 (Dense)              (None, 3)                 33        \n",
-      "=================================================================\n",
-      "Total params: 303\n",
-      "Trainable params: 303\n",
-      "Non-trainable params: 0\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "model2 = Sequential()\n",
-    "model2.add(Dense(10, activation='relu', input_shape=(4,)))\n",
-    "model2.add(Dense(10, activation='relu'))\n",
-    "model2.add(Dense(10, activation='relu'))\n",
-    "model2.add(Dense(3, activation='softmax'))\n",
-    "    \n",
-    "model2.summary()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 14,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
-      ]
-     },
-     "execution_count": 14,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "model2.to_json()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Load into model architecture table"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 15,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "2 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_id</th>\n",
-       "        <th>model_arch</th>\n",
-       "        <th>model_weights</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>__internal_madlib_id__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
-       "        <td>None</td>\n",
-       "        <td>Sophie</td>\n",
-       "        <td>MLP with 1 hidden layer</td>\n",
-       "        <td>__madlib_temp_96702431_1576708421_6956281__</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
-       "        <td>None</td>\n",
-       "        <td>Maria</td>\n",
-       "        <td>MLP with 2 hidden layers</td>\n",
-       "        <td>__madlib_temp_85244704_1576708422_1853942__</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'MLP with 1 hidden layer', u'__madlib_temp_96702431_1576708421_6956281__'),\n",
-       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'MLP with 2 hidden layers', u'__madlib_temp_85244704_1576708422_1853942__')]"
-      ]
-     },
-     "execution_count": 15,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS model_arch_library;\n",
-    "\n",
-    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
-    "                               \n",
-    "$$\n",
-    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
-    "$$\n",
-    "::json,         -- JSON blob\n",
-    "                               NULL,                  -- Weights\n",
-    "                               'Sophie',              -- Name\n",
-    "                               'MLP with 1 hidden layer'       -- Descr\n",
-    ");\n",
-    "\n",
-    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
-    "                               \n",
-    "$$\n",
-    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
-    "$$\n",
-    "::json,         -- JSON blob\n",
-    "                               NULL,                  -- Weights\n",
-    "                               'Maria',               -- Name\n",
-    "                               'MLP with 2 hidden layers'       -- Descr\n",
-    ");\n",
-    "\n",
-    "SELECT * FROM model_arch_library ORDER BY model_id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"def_mst\"></a>\n",
-    "# 4.  Define and load model selection tuples"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Select the model(s) from the model architecture table that you want to run, along with the compile and fit parameters. Permutations will be created for the set of model selection parameters will be loaded:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 16,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "12 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>mst_key</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>11</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>12</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (3, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (4, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (5, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (6, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (7, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (8, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (9, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (10, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
-       " (11, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
-       " (12, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1')]"
-      ]
-     },
-     "execution_count": 16,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
-    "\n",
-    "SELECT madlib.load_model_selection_table('model_arch_library', -- model architecture table\n",
-    "                                         'mst_table',          -- model selection table output\n",
-    "                                          ARRAY[1,2],              -- model ids from model architecture table\n",
-    "                                          ARRAY[                   -- compile params\n",
-    "                                              $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$,\n",
-    "                                              $$loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']$$,\n",
-    "                                              $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$\n",
-    "                                          ],\n",
-    "                                          ARRAY[                    -- fit params\n",
-    "                                              $$batch_size=4,epochs=1$$,\n",
-    "                                              $$batch_size=8,epochs=1$$\n",
-    "                                          ]\n",
-    "                                         );\n",
-    "                                  \n",
-    "SELECT * FROM mst_table ORDER BY mst_key;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "This is the name of the model architecture table that corresponds to the model selection table:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 17,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_arch_table</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>model_arch_library</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'model_arch_library',)]"
-      ]
-     },
-     "execution_count": 17,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM mst_table_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"train\"></a>\n",
-    "# 5.  Train\n",
-    "Train multiple models:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 18,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>madlib_keras_fit_multiple_model</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td></td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[('',)]"
-      ]
-     },
-     "execution_count": 18,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_multi_model, iris_multi_model_summary, iris_multi_model_info;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
-    "                                              'iris_multi_model',     -- model_output_table\n",
-    "                                              'mst_table',            -- model_selection_table\n",
-    "                                              10,                     -- num_iterations\n",
-    "                                              FALSE                   -- use gpus\n",
-    "                                             );"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View the model summary:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 19,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>validation_table</th>\n",
-       "        <th>model</th>\n",
-       "        <th>model_info</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>metrics_compute_frequency</th>\n",
-       "        <th>warm_start</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>madlib_version</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>metrics_iters</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_train_packed</td>\n",
-       "        <td>None</td>\n",
-       "        <td>iris_multi_model</td>\n",
-       "        <td>iris_multi_model_info</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>model_arch_library</td>\n",
-       "        <td>10</td>\n",
-       "        <td>10</td>\n",
-       "        <td>False</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>2019-12-18 22:33:49.706384</td>\n",
-       "        <td>2019-12-18 22:35:34.547961</td>\n",
-       "        <td>1.17-dev</td>\n",
-       "        <td>3</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>[10]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_train_packed', None, u'iris_multi_model', u'iris_multi_model_info', u'class_text', u'attributes', u'model_arch_library', 10, 10, False, None, None, datetime.datetime(2019, 12, 18, 22, 33, 49, 706384), datetime.datetime(2019, 12, 18, 22, 35, 34, 547961), u'1.17-dev', 3, [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], u'character varying', 1.0, [10])]"
-      ]
-     },
-     "execution_count": 19,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_multi_model_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View results for each model:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 20,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "12 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>mst_key</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.148514986038208]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.975000023842</td>\n",
-       "        <td>0.12241948396</td>\n",
-       "        <td>[0.975000023841858]</td>\n",
-       "        <td>[0.122419483959675]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.172315120697021]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.975000023842</td>\n",
-       "        <td>0.123081341386</td>\n",
-       "        <td>[0.975000023841858]</td>\n",
-       "        <td>[0.123081341385841]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.274233102798462]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.925000011921</td>\n",
-       "        <td>0.171397775412</td>\n",
-       "        <td>[0.925000011920929]</td>\n",
-       "        <td>[0.171397775411606]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.155992984771729]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.925000011921</td>\n",
-       "        <td>0.51177251339</td>\n",
-       "        <td>[0.925000011920929]</td>\n",
-       "        <td>[0.511772513389587]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.220170021057129]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.908333361149</td>\n",
-       "        <td>0.214677110314</td>\n",
-       "        <td>[0.908333361148834]</td>\n",
-       "        <td>[0.214677110314369]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>12</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.191344022750854]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.833333313465</td>\n",
-       "        <td>0.524632036686</td>\n",
-       "        <td>[0.833333313465118]</td>\n",
-       "        <td>[0.524632036685944]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.181636810302734]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.758333325386</td>\n",
-       "        <td>0.393412530422</td>\n",
-       "        <td>[0.758333325386047]</td>\n",
-       "        <td>[0.393412530422211]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.181061029434204]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.658333361149</td>\n",
-       "        <td>0.474381148815</td>\n",
-       "        <td>[0.658333361148834]</td>\n",
-       "        <td>[0.474381148815155]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.20294713973999]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.658333361149</td>\n",
-       "        <td>0.475430130959</td>\n",
-       "        <td>[0.658333361148834]</td>\n",
-       "        <td>[0.475430130958557]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.207202911376953]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.574999988079</td>\n",
-       "        <td>0.885546028614</td>\n",
-       "        <td>[0.574999988079071]</td>\n",
-       "        <td>[0.885546028614044]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>11</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.374184846878052]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.433333337307</td>\n",
-       "        <td>0.82793289423</td>\n",
-       "        <td>[0.433333337306976]</td>\n",
-       "        <td>[0.827932894229889]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.216787099838257]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.316666662693</td>\n",
-       "        <td>1.10255157948</td>\n",
-       "        <td>[0.316666662693024]</td>\n",
-       "        <td>[1.1025515794754]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(4, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 0.7900390625, [0.148514986038208], [u'accuracy'], 0.975000023842, 0.12241948396, [0.975000023841858], [0.122419483959675], None, None, None, None),\n",
-       " (10, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 1.2197265625, [0.172315120697021], [u'accuracy'], 0.975000023842, 0.123081341386, [0.975000023841858], [0.123081341385841], None, None, None, None),\n",
-       " (9, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 1.2197265625, [0.274233102798462], [u'accuracy'], 0.925000011921, 0.171397775412, [0.925000011920929], [0.171397775411606], None, None, None, None),\n",
-       " (5, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 0.7900390625, [0.155992984771729], [u'accuracy'], 0.925000011921, 0.51177251339, [0.925000011920929], [0.511772513389587], None, None, None, None),\n",
-       " (3, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 0.7900390625, [0.220170021057129], [u'accuracy'], 0.908333361149, 0.214677110314, [0.908333361148834], [0.214677110314369], None, None, None, None),\n",
-       " (12, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 1.2197265625, [0.191344022750854], [u'accuracy'], 0.833333313465, 0.524632036686, [0.833333313465118], [0.524632036685944], None, None, None, None),\n",
-       " (8, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 1.2197265625, [0.181636810302734], [u'accuracy'], 0.758333325386, 0.393412530422, [0.758333325386047], [0.393412530422211], None, None, None, None),\n",
-       " (7, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 1.2197265625, [0.181061029434204], [u'accuracy'], 0.658333361149, 0.474381148815, [0.658333361148834], [0.474381148815155], None, None, None, None),\n",
-       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 0.7900390625, [0.20294713973999], [u'accuracy'], 0.658333361149, 0.475430130959, [0.658333361148834], [0.475430130958557], None, None, None, None),\n",
-       " (6, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 0.7900390625, [0.207202911376953], [u'accuracy'], 0.574999988079, 0.885546028614, [0.574999988079071], [0.885546028614044], None, None, None, None),\n",
-       " (11, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 1.2197265625, [0.374184846878052], [u'accuracy'], 0.433333337307, 0.82793289423, [0.433333337306976], [0.827932894229889], None, None, None, None),\n",
-       " (1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 0.7900390625, [0.216787099838257], [u'accuracy'], 0.316666662693, 1.10255157948, [0.316666662693024], [1.1025515794754], None, None, None, None)]"
-      ]
-     },
-     "execution_count": 20,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_multi_model_info ORDER BY training_metrics_final DESC, training_loss_final;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"eval\"></a>\n",
-    "# 6. Evaluate\n",
-    "\n",
-    "Now run evaluate using model we built above:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 21,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>loss</th>\n",
-       "        <th>metric</th>\n",
-       "        <th>metrics_type</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>0.15500420332</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(0.15500420331955, 0.966666638851166, [u'accuracy'])]"
-      ]
-     },
-     "execution_count": 21,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_validate;\n",
-    "SELECT madlib.madlib_keras_evaluate('iris_multi_model',  -- model\n",
-    "                                    'iris_test_packed',  -- test table\n",
-    "                                    'iris_validate',     -- output table\n",
-    "                                     NULL,               -- use gpus\n",
-    "                                     3                   -- mst_key to use\n",
-    "                                   );\n",
-    "\n",
-    "SELECT * FROM iris_validate;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"pred\"></a>\n",
-    "# 7. Predict\n",
-    "\n",
-    "Now predict using model we built.  We will use the validation data set for prediction as well, which is not usual but serves to show the syntax. The prediction is in the estimated_class_text column:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 22,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "30 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>estimated_class_text</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>19</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>25</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>26</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>28</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>38</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>44</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>45</td>\n",
-       "        <td>Iris-setosa</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>51</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>53</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>57</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>59</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>62</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>69</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>75</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>77</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>97</td>\n",
-       "        <td>Iris-versicolor</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>102</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>107</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>114</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>118</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>120</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>122</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>132</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>146</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>147</td>\n",
-       "        <td>Iris-virginica</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(3, u'Iris-setosa'),\n",
-       " (5, u'Iris-setosa'),\n",
-       " (7, u'Iris-setosa'),\n",
-       " (8, u'Iris-setosa'),\n",
-       " (10, u'Iris-setosa'),\n",
-       " (19, u'Iris-setosa'),\n",
-       " (25, u'Iris-setosa'),\n",
-       " (26, u'Iris-setosa'),\n",
-       " (28, u'Iris-setosa'),\n",
-       " (38, u'Iris-setosa'),\n",
-       " (44, u'Iris-setosa'),\n",
-       " (45, u'Iris-setosa'),\n",
-       " (51, u'Iris-versicolor'),\n",
-       " (53, u'Iris-versicolor'),\n",
-       " (57, u'Iris-versicolor'),\n",
-       " (59, u'Iris-versicolor'),\n",
-       " (62, u'Iris-versicolor'),\n",
-       " (69, u'Iris-virginica'),\n",
-       " (75, u'Iris-versicolor'),\n",
-       " (77, u'Iris-versicolor'),\n",
-       " (97, u'Iris-versicolor'),\n",
-       " (102, u'Iris-virginica'),\n",
-       " (107, u'Iris-virginica'),\n",
-       " (114, u'Iris-virginica'),\n",
-       " (118, u'Iris-virginica'),\n",
-       " (120, u'Iris-virginica'),\n",
-       " (122, u'Iris-virginica'),\n",
-       " (132, u'Iris-virginica'),\n",
-       " (146, u'Iris-virginica'),\n",
-       " (147, u'Iris-virginica')]"
-      ]
-     },
-     "execution_count": 22,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_predict;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_predict('iris_multi_model', -- model\n",
-    "                                   'iris_test',        -- test_table\n",
-    "                                   'id',               -- id column\n",
-    "                                   'attributes',       -- independent var\n",
-    "                                   'iris_predict',     -- output table\n",
-    "                                    'response',        -- prediction type\n",
-    "                                    FALSE,             -- use gpus\n",
-    "                                    3                  -- mst_key to use\n",
-    "                                   );\n",
-    "\n",
-    "SELECT * FROM iris_predict ORDER BY id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Count missclassifications"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 23,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1L,)]"
-      ]
-     },
-     "execution_count": 23,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT COUNT(*) FROM iris_predict JOIN iris_test USING (id) \n",
-    "WHERE iris_predict.estimated_class_text != iris_test.class_text;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Percent missclassifications"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 24,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>test_accuracy_percent</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>96.67</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(Decimal('96.67'),)]"
-      ]
-     },
-     "execution_count": 24,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
-    "    (select iris_test.class_text as actual, iris_predict.estimated_class_text as estimated\n",
-    "     from iris_predict inner join iris_test\n",
-    "     on iris_test.id=iris_predict.id) q\n",
-    "WHERE q.actual=q.estimated;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"class2\"></a>\n",
-    "# Classification with Other Parameters\n",
-    "\n",
-    "<a id=\"val_dataset\"></a>\n",
-    "# 1.  Validation dataset\n",
-    "\n",
-    "Now use a validation dataset and compute metrics every 2nd iteration using the 'metrics_compute_frequency' parameter.  This can help reduce run time if you do not need metrics computed at every iteration."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 25,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>madlib_keras_fit_multiple_model</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td></td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[('',)]"
-      ]
-     },
-     "execution_count": 25,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_multi_model, iris_multi_model_summary, iris_multi_model_info;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
-    "                                              'iris_multi_model',     -- model_output_table\n",
-    "                                              'mst_table',            -- model_selection_table\n",
-    "                                               10,                     -- num_iterations\n",
-    "                                               FALSE,                 -- use gpus\n",
-    "                                              'iris_test_packed',     -- validation dataset\n",
-    "                                               3,                     -- metrics compute frequency\n",
-    "                                               FALSE,                 -- warm start\n",
-    "                                              'Sophie L.',            -- name\n",
-    "                                              'Model selection for iris dataset'  -- description\n",
-    "                                             );"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View the model summary:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 26,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>validation_table</th>\n",
-       "        <th>model</th>\n",
-       "        <th>model_info</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>metrics_compute_frequency</th>\n",
-       "        <th>warm_start</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>madlib_version</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>metrics_iters</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_train_packed</td>\n",
-       "        <td>iris_test_packed</td>\n",
-       "        <td>iris_multi_model</td>\n",
-       "        <td>iris_multi_model_info</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>model_arch_library</td>\n",
-       "        <td>10</td>\n",
-       "        <td>3</td>\n",
-       "        <td>False</td>\n",
-       "        <td>Sophie L.</td>\n",
-       "        <td>Model selection for iris dataset</td>\n",
-       "        <td>2019-12-18 22:35:49.962345</td>\n",
-       "        <td>2019-12-18 22:37:51.230499</td>\n",
-       "        <td>1.17-dev</td>\n",
-       "        <td>3</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>[3, 6, 9, 10]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_train_packed', u'iris_test_packed', u'iris_multi_model', u'iris_multi_model_info', u'class_text', u'attributes', u'model_arch_library', 10, 3, False, u'Sophie L.', u'Model selection for iris dataset', datetime.datetime(2019, 12, 18, 22, 35, 49, 962345), datetime.datetime(2019, 12, 18, 22, 37, 51, 230499), u'1.17-dev', 3, [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], u'character varying', 1.0, [3, 6, 9, 10])]"
-      ]
-     },
-     "execution_count": 26,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_multi_model_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View performance of each model:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 27,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "12 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>mst_key</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.400555849075317, 0.175060987472534, 0.161082029342651, 0.159379005432129]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.958333313465</td>\n",
-       "        <td>0.370426625013</td>\n",
-       "        <td>[0.841666638851166, 0.875, 0.958333313465118, 0.958333313465118]</td>\n",
-       "        <td>[0.597030103206635, 0.467845916748047, 0.394165992736816, 0.370426625013351]</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>0.32715767622</td>\n",
-       "        <td>[0.866666674613953, 0.933333337306976, 1.0, 1.0]</td>\n",
-       "        <td>[0.587784588336945, 0.432697623968124, 0.352933287620544, 0.32715767621994]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.157984018325806, 0.146160840988159, 0.446839094161987, 0.217149972915649]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.916666686535</td>\n",
-       "        <td>0.176682218909</td>\n",
-       "        <td>[0.958333313465118, 0.891666650772095, 0.841666638851166, 0.916666686534882]</td>\n",
-       "        <td>[0.340974450111389, 0.224177747964859, 0.315857976675034, 0.176682218909264]</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.146555349231</td>\n",
-       "        <td>[0.966666638851166, 0.933333337306976, 0.866666674613953, 0.966666638851166]</td>\n",
-       "        <td>[0.306026995182037, 0.204480707645416, 0.291850447654724, 0.146555349230766]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.158334016799927, 0.492121934890747, 0.168816804885864, 0.160614013671875]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.949999988079</td>\n",
-       "        <td>0.137093007565</td>\n",
-       "        <td>[0.75, 0.808333337306976, 0.941666662693024, 0.949999988079071]</td>\n",
-       "        <td>[0.861838400363922, 0.306531131267548, 0.267581582069397, 0.137093007564545]</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.0812632590532</td>\n",
-       "        <td>[0.533333361148834, 0.733333349227905, 1.0, 0.966666638851166]</td>\n",
-       "        <td>[1.17265951633453, 0.347328811883926, 0.0795030668377876, 0.0812632590532303]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.206979990005493, 0.175852060317993, 0.18351411819458, 0.173283100128174]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.841666638851</td>\n",
-       "        <td>0.319059103727</td>\n",
-       "        <td>[0.833333313465118, 0.916666686534882, 0.958333313465118, 0.841666638851166]</td>\n",
-       "        <td>[0.375581055879593, 0.235803470015526, 0.119093284010887, 0.319059103727341]</td>\n",
-       "        <td>0.866666674614</td>\n",
-       "        <td>0.294114112854</td>\n",
-       "        <td>[0.866666674613953, 0.966666638851166, 0.933333337306976, 0.866666674613953]</td>\n",
-       "        <td>[0.332203418016434, 0.206457450985909, 0.09817935526371, 0.294114112854004]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.154335021972656, 0.14276385307312, 0.160094022750854, 0.147177934646606]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.833333313465</td>\n",
-       "        <td>0.315035998821</td>\n",
-       "        <td>[0.850000023841858, 0.966666638851166, 0.966666638851166, 0.833333313465118]</td>\n",
-       "        <td>[0.39260533452034, 0.207864001393318, 0.14202418923378, 0.315035998821259]</td>\n",
-       "        <td>0.833333313465</td>\n",
-       "        <td>0.287047833204</td>\n",
-       "        <td>[0.833333313465118, 0.966666638851166, 0.933333337306976, 0.833333313465118]</td>\n",
-       "        <td>[0.350265830755234, 0.179627984762192, 0.119969591498375, 0.287047833204269]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.183771848678589, 0.442173957824707, 0.196517944335938, 0.183962106704712]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.683333337307</td>\n",
-       "        <td>0.773626208305</td>\n",
-       "        <td>[0.983333349227905, 0.783333361148834, 0.841666638851166, 0.683333337306976]</td>\n",
-       "        <td>[0.323956668376923, 0.355609774589539, 0.289077579975128, 0.773626208305359]</td>\n",
-       "        <td>0.733333349228</td>\n",
-       "        <td>0.598832905293</td>\n",
-       "        <td>[0.966666638851166, 0.733333349227905, 0.866666674613953, 0.733333349227905]</td>\n",
-       "        <td>[0.292185336351395, 0.310099214315414, 0.278687566518784, 0.598832905292511]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>11</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.215842962265015, 0.183883190155029, 0.181258201599121, 0.233398914337158]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.658333361149</td>\n",
-       "        <td>0.501300632954</td>\n",
-       "        <td>[0.341666668653488, 0.658333361148834, 0.658333361148834, 0.658333361148834]</td>\n",
-       "        <td>[0.947986364364624, 0.807084918022156, 0.549242556095123, 0.501300632953644]</td>\n",
-       "        <td>0.699999988079</td>\n",
-       "        <td>0.459856539965</td>\n",
-       "        <td>[0.300000011920929, 0.699999988079071, 0.699999988079071, 0.699999988079071]</td>\n",
-       "        <td>[0.971994161605835, 0.821518063545227, 0.513974606990814, 0.459856539964676]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.181059837341309, 0.156504154205322, 0.154800891876221, 0.165037870407104]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.675000011921</td>\n",
-       "        <td>0.500130057335</td>\n",
-       "        <td>[0.658333361148834, 0.908333361148834, 0.908333361148834, 0.675000011920929]</td>\n",
-       "        <td>[0.822371363639832, 0.354260504245758, 0.206746637821198, 0.5001300573349]</td>\n",
-       "        <td>0.699999988079</td>\n",
-       "        <td>0.511800050735</td>\n",
-       "        <td>[0.699999988079071, 0.933333337306976, 0.966666638851166, 0.699999988079071]</td>\n",
-       "        <td>[0.784473180770874, 0.314396589994431, 0.171932756900787, 0.511800050735474]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.16503119468689, 0.165420055389404, 0.163087844848633, 0.157285213470459]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.600000023842</td>\n",
-       "        <td>0.536593079567</td>\n",
-       "        <td>[0.625, 0.491666674613953, 0.508333325386047, 0.600000023841858]</td>\n",
-       "        <td>[0.877406716346741, 0.665770947933197, 0.563206613063812, 0.536593079566956]</td>\n",
-       "        <td>0.600000023842</td>\n",
-       "        <td>0.50565046072</td>\n",
-       "        <td>[0.566666662693024, 0.533333361148834, 0.600000023841858, 0.600000023841858]</td>\n",
-       "        <td>[0.898801684379578, 0.642534494400024, 0.529698371887207, 0.505650460720062]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>12</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.180193901062012, 0.230684041976929, 0.202606916427612, 0.182677030563354]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.5</td>\n",
-       "        <td>1.01774513721</td>\n",
-       "        <td>[0.341666668653488, 0.491666674613953, 0.524999976158142, 0.5]</td>\n",
-       "        <td>[1.10608339309692, 1.06158423423767, 1.02908384799957, 1.01774513721466]</td>\n",
-       "        <td>0.5</td>\n",
-       "        <td>1.01636135578</td>\n",
-       "        <td>[0.300000011920929, 0.466666668653488, 0.466666668653488, 0.5]</td>\n",
-       "        <td>[1.10331404209137, 1.05365967750549, 1.02413082122803, 1.01636135578156]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.181950092315674, 0.197594881057739, 0.187069177627563, 0.183701992034912]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.316666662693</td>\n",
-       "        <td>1.10080897808</td>\n",
-       "        <td>[0.316666662693024, 0.341666668653488, 0.341666668653488, 0.316666662693024]</td>\n",
-       "        <td>[1.1043815612793, 1.11140048503876, 1.09834468364716, 1.10080897808075]</td>\n",
-       "        <td>0.40000000596</td>\n",
-       "        <td>1.09380173683</td>\n",
-       "        <td>[0.400000005960464, 0.300000011920929, 0.300000011920929, 0.400000005960464]</td>\n",
-       "        <td>[1.09075009822845, 1.09998726844788, 1.10155093669891, 1.09380173683167]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.182392835617065, 0.206873893737793, 0.192094087600708, 0.185320854187012]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.341666668653</td>\n",
-       "        <td>1.10410153866</td>\n",
-       "        <td>[0.341666668653488, 0.316666662693024, 0.341666668653488, 0.341666668653488]</td>\n",
-       "        <td>[1.10291886329651, 1.10132431983948, 1.10635650157928, 1.10410153865814]</td>\n",
-       "        <td>0.300000011921</td>\n",
-       "        <td>1.10918176174</td>\n",
-       "        <td>[0.300000011920929, 0.400000005960464, 0.300000011920929, 0.300000011920929]</td>\n",
-       "        <td>[1.10382485389709, 1.09316170215607, 1.1332186460495, 1.10918176174164]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(6, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 0.7900390625, [0.400555849075317, 0.175060987472534, 0.161082029342651, 0.159379005432129], [u'accuracy'], 0.958333313465, 0.370426625013, [0.841666638851166, 0.875, 0.958333313465118, 0.958333313465118], [0.597030103206635, 0.467845916748047, 0.394165992736816, 0.370426625013351], 1.0, 0.32715767622, [0.866666674613953, 0.933333337306976, 1.0, 1.0], [0.587784588336945, 0.432697623968124, 0.352933287620544, 0.32715767621994]),\n",
-       " (3, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 0.7900390625, [0.157984018325806, 0.146160840988159, 0.446839094161987, 0.217149972915649], [u'accuracy'], 0.916666686535, 0.176682218909, [0.958333313465118, 0.891666650772095, 0.841666638851166, 0.916666686534882], [0.340974450111389, 0.224177747964859, 0.315857976675034, 0.176682218909264], 0.966666638851, 0.146555349231, [0.966666638851166, 0.933333337306976, 0.866666674613953, 0.966666638851166], [0.306026995182037, 0.204480707645416, 0.291850447654724, 0.146555349230766]),\n",
-       " (1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 0.7900390625, [0.158334016799927, 0.492121934890747, 0.168816804885864, 0.160614013671875], [u'accuracy'], 0.949999988079, 0.137093007565, [0.75, 0.808333337306976, 0.941666662693024, 0.949999988079071], [0.861838400363922, 0.306531131267548, 0.267581582069397, 0.137093007564545], 0.966666638851, 0.0812632590532, [0.533333361148834, 0.733333349227905, 1.0, 0.966666638851166], [1.17265951633453, 0.347328811883926, 0.0795030668377876, 0.0812632590532303]),\n",
-       " (10, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 1.2197265625, [0.206979990005493, 0.175852060317993, 0.18351411819458, 0.173283100128174], [u'accuracy'], 0.841666638851, 0.319059103727, [0.833333313465118, 0.916666686534882, 0.958333313465118, 0.841666638851166], [0.375581055879593, 0.235803470015526, 0.119093284010887, 0.319059103727341], 0.866666674614, 0.294114112854, [0.866666674613953, 0.966666638851166, 0.933333337306976, 0.866666674613953], [0.332203418016434, 0.206457450985909, 0.09817935526371, 0.294114112854004]),\n",
-       " (4, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 0.7900390625, [0.154335021972656, 0.14276385307312, 0.160094022750854, 0.147177934646606], [u'accuracy'], 0.833333313465, 0.315035998821, [0.850000023841858, 0.966666638851166, 0.966666638851166, 0.833333313465118], [0.39260533452034, 0.207864001393318, 0.14202418923378, 0.315035998821259], 0.833333313465, 0.287047833204, [0.833333313465118, 0.966666638851166, 0.933333337306976, 0.833333313465118], [0.350265830755234, 0.179627984762192, 0.119969591498375, 0.287047833204269]),\n",
-       " (9, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 1.2197265625, [0.183771848678589, 0.442173957824707, 0.196517944335938, 0.183962106704712], [u'accuracy'], 0.683333337307, 0.773626208305, [0.983333349227905, 0.783333361148834, 0.841666638851166, 0.683333337306976], [0.323956668376923, 0.355609774589539, 0.289077579975128, 0.773626208305359], 0.733333349228, 0.598832905293, [0.966666638851166, 0.733333349227905, 0.866666674613953, 0.733333349227905], [0.292185336351395, 0.310099214315414, 0.278687566518784, 0.598832905292511]),\n",
-       " (11, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 1.2197265625, [0.215842962265015, 0.183883190155029, 0.181258201599121, 0.233398914337158], [u'accuracy'], 0.658333361149, 0.501300632954, [0.341666668653488, 0.658333361148834, 0.658333361148834, 0.658333361148834], [0.947986364364624, 0.807084918022156, 0.549242556095123, 0.501300632953644], 0.699999988079, 0.459856539965, [0.300000011920929, 0.699999988079071, 0.699999988079071, 0.699999988079071], [0.971994161605835, 0.821518063545227, 0.513974606990814, 0.459856539964676]),\n",
-       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 0.7900390625, [0.181059837341309, 0.156504154205322, 0.154800891876221, 0.165037870407104], [u'accuracy'], 0.675000011921, 0.500130057335, [0.658333361148834, 0.908333361148834, 0.908333361148834, 0.675000011920929], [0.822371363639832, 0.354260504245758, 0.206746637821198, 0.5001300573349], 0.699999988079, 0.511800050735, [0.699999988079071, 0.933333337306976, 0.966666638851166, 0.699999988079071], [0.784473180770874, 0.314396589994431, 0.171932756900787, 0.511800050735474]),\n",
-       " (5, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 0.7900390625, [0.16503119468689, 0.165420055389404, 0.163087844848633, 0.157285213470459], [u'accuracy'], 0.600000023842, 0.536593079567, [0.625, 0.491666674613953, 0.508333325386047, 0.600000023841858], [0.877406716346741, 0.665770947933197, 0.563206613063812, 0.536593079566956], 0.600000023842, 0.50565046072, [0.566666662693024, 0.533333361148834, 0.600000023841858, 0.600000023841858], [0.898801684379578, 0.642534494400024, 0.529698371887207, 0.505650460720062]),\n",
-       " (12, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 1.2197265625, [0.180193901062012, 0.230684041976929, 0.202606916427612, 0.182677030563354], [u'accuracy'], 0.5, 1.01774513721, [0.341666668653488, 0.491666674613953, 0.524999976158142, 0.5], [1.10608339309692, 1.06158423423767, 1.02908384799957, 1.01774513721466], 0.5, 1.01636135578, [0.300000011920929, 0.466666668653488, 0.466666668653488, 0.5], [1.10331404209137, 1.05365967750549, 1.02413082122803, 1.01636135578156]),\n",
-       " (7, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 1.2197265625, [0.181950092315674, 0.197594881057739, 0.187069177627563, 0.183701992034912], [u'accuracy'], 0.316666662693, 1.10080897808, [0.316666662693024, 0.341666668653488, 0.341666668653488, 0.316666662693024], [1.1043815612793, 1.11140048503876, 1.09834468364716, 1.10080897808075], 0.40000000596, 1.09380173683, [0.400000005960464, 0.300000011920929, 0.300000011920929, 0.400000005960464], [1.09075009822845, 1.09998726844788, 1.10155093669891, 1.09380173683167]),\n",
-       " (8, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 1.2197265625, [0.182392835617065, 0.206873893737793, 0.192094087600708, 0.185320854187012], [u'accuracy'], 0.341666668653, 1.10410153866, [0.341666668653488, 0.316666662693024, 0.341666668653488, 0.341666668653488], [1.10291886329651, 1.10132431983948, 1.10635650157928, 1.10410153865814], 0.300000011921, 1.10918176174, [0.300000011920929, 0.400000005960464, 0.300000011920929, 0.300000011920929], [1.10382485389709, 1.09316170215607, 1.1332186460495, 1.10918176174164])]"
-      ]
-     },
-     "execution_count": 27,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_multi_model_info ORDER BY validation_metrics_final DESC;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Plot validation results"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 28,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "%matplotlib notebook\n",
-    "import matplotlib.pyplot as plt\n",
-    "from matplotlib.ticker import MaxNLocator\n",
-    "from collections import defaultdict\n",
-    "import pandas as pd\n",
-    "import seaborn as sns\n",
-    "sns.set_palette(sns.color_palette(\"hls\", 20))\n",
-    "plt.rcParams.update({'font.size': 12})\n",
-    "pd.set_option('display.max_colwidth', -1)"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 29,
-   "metadata": {
-    "scrolled": false
-   },
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "7 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "application/javascript": [
-       "/* Put everything inside the global mpl namespace */\n",
-       "window.mpl = {};\n",
-       "\n",
-       "\n",
-       "mpl.get_websocket_type = function() {\n",
-       "    if (typeof(WebSocket) !== 'undefined') {\n",
-       "        return WebSocket;\n",
-       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
-       "        return MozWebSocket;\n",
-       "    } else {\n",
-       "        alert('Your browser does not have WebSocket support.' +\n",
-       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
-       "              'Firefox 4 and 5 are also supported but you ' +\n",
-       "              'have to enable WebSockets in about:config.');\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
-       "    this.id = figure_id;\n",
-       "\n",
-       "    this.ws = websocket;\n",
-       "\n",
-       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
-       "\n",
-       "    if (!this.supports_binary) {\n",
-       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
-       "        if (warnings) {\n",
-       "            warnings.style.display = 'block';\n",
-       "            warnings.textContent = (\n",
-       "                \"This browser does not support binary websocket messages. \" +\n",
-       "                    \"Performance may be slow.\");\n",
-       "        }\n",
-       "    }\n",
-       "\n",
-       "    this.imageObj = new Image();\n",
-       "\n",
-       "    this.context = undefined;\n",
-       "    this.message = undefined;\n",
-       "    this.canvas = undefined;\n",
-       "    this.rubberband_canvas = undefined;\n",
-       "    this.rubberband_context = undefined;\n",
-       "    this.format_dropdown = undefined;\n",
-       "\n",
-       "    this.image_mode = 'full';\n",
-       "\n",
-       "    this.root = $('<div/>');\n",
-       "    this._root_extra_style(this.root)\n",
-       "    this.root.attr('style', 'display: inline-block');\n",
-       "\n",
-       "    $(parent_element).append(this.root);\n",
-       "\n",
-       "    this._init_header(this);\n",
-       "    this._init_canvas(this);\n",
-       "    this._init_toolbar(this);\n",
-       "\n",
-       "    var fig = this;\n",
-       "\n",
-       "    this.waiting = false;\n",
-       "\n",
-       "    this.ws.onopen =  function () {\n",
-       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
-       "            fig.send_message(\"send_image_mode\", {});\n",
-       "            if (mpl.ratio != 1) {\n",
-       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
-       "            }\n",
-       "            fig.send_message(\"refresh\", {});\n",
-       "        }\n",
-       "\n",
-       "    this.imageObj.onload = function() {\n",
-       "            if (fig.image_mode == 'full') {\n",
-       "                // Full images could contain transparency (where diff images\n",
-       "                // almost always do), so we need to clear the canvas so that\n",
-       "                // there is no ghosting.\n",
-       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
-       "            }\n",
-       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
-       "        };\n",
-       "\n",
-       "    this.imageObj.onunload = function() {\n",
-       "        fig.ws.close();\n",
-       "    }\n",
-       "\n",
-       "    this.ws.onmessage = this._make_on_message_function(this);\n",
-       "\n",
-       "    this.ondownload = ondownload;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_header = function() {\n",
-       "    var titlebar = $(\n",
-       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
-       "        'ui-helper-clearfix\"/>');\n",
-       "    var titletext = $(\n",
-       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
-       "        'text-align: center; padding: 3px;\"/>');\n",
-       "    titlebar.append(titletext)\n",
-       "    this.root.append(titlebar);\n",
-       "    this.header = titletext[0];\n",
-       "}\n",
-       "\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
-       "\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
-       "\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_canvas = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var canvas_div = $('<div/>');\n",
-       "\n",
-       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
-       "\n",
-       "    function canvas_keyboard_event(event) {\n",
-       "        return fig.key_event(event, event['data']);\n",
-       "    }\n",
-       "\n",
-       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
-       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
-       "    this.canvas_div = canvas_div\n",
-       "    this._canvas_extra_style(canvas_div)\n",
-       "    this.root.append(canvas_div);\n",
-       "\n",
-       "    var canvas = $('<canvas/>');\n",
-       "    canvas.addClass('mpl-canvas');\n",
-       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
-       "\n",
-       "    this.canvas = canvas[0];\n",
-       "    this.context = canvas[0].getContext(\"2d\");\n",
-       "\n",
-       "    var backingStore = this.context.backingStorePixelRatio ||\n",
-       "\tthis.context.webkitBackingStorePixelRatio ||\n",
-       "\tthis.context.mozBackingStorePixelRatio ||\n",
-       "\tthis.context.msBackingStorePixelRatio ||\n",
-       "\tthis.context.oBackingStorePixelRatio ||\n",
-       "\tthis.context.backingStorePixelRatio || 1;\n",
-       "\n",
-       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
-       "\n",
-       "    var rubberband = $('<canvas/>');\n",
-       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
-       "\n",
-       "    var pass_mouse_events = true;\n",
-       "\n",
-       "    canvas_div.resizable({\n",
-       "        start: function(event, ui) {\n",
-       "            pass_mouse_events = false;\n",
-       "        },\n",
-       "        resize: function(event, ui) {\n",
-       "            fig.request_resize(ui.size.width, ui.size.height);\n",
-       "        },\n",
-       "        stop: function(event, ui) {\n",
-       "            pass_mouse_events = true;\n",
-       "            fig.request_resize(ui.size.width, ui.size.height);\n",
-       "        },\n",
-       "    });\n",
-       "\n",
-       "    function mouse_event_fn(event) {\n",
-       "        if (pass_mouse_events)\n",
-       "            return fig.mouse_event(event, event['data']);\n",
-       "    }\n",
-       "\n",
-       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
-       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
-       "    // Throttle sequential mouse events to 1 every 20ms.\n",
-       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
-       "\n",
-       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
-       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
-       "\n",
-       "    canvas_div.on(\"wheel\", function (event) {\n",
-       "        event = event.originalEvent;\n",
-       "        event['data'] = 'scroll'\n",
-       "        if (event.deltaY < 0) {\n",
-       "            event.step = 1;\n",
-       "        } else {\n",
-       "            event.step = -1;\n",
-       "        }\n",
-       "        mouse_event_fn(event);\n",
-       "    });\n",
-       "\n",
-       "    canvas_div.append(canvas);\n",
-       "    canvas_div.append(rubberband);\n",
-       "\n",
-       "    this.rubberband = rubberband;\n",
-       "    this.rubberband_canvas = rubberband[0];\n",
-       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
-       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
-       "\n",
-       "    this._resize_canvas = function(width, height) {\n",
-       "        // Keep the size of the canvas, canvas container, and rubber band\n",
-       "        // canvas in synch.\n",
-       "        canvas_div.css('width', width)\n",
-       "        canvas_div.css('height', height)\n",
-       "\n",
-       "        canvas.attr('width', width * mpl.ratio);\n",
-       "        canvas.attr('height', height * mpl.ratio);\n",
-       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
-       "\n",
-       "        rubberband.attr('width', width);\n",
-       "        rubberband.attr('height', height);\n",
-       "    }\n",
-       "\n",
-       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
-       "    // upon first draw.\n",
-       "    this._resize_canvas(600, 600);\n",
-       "\n",
-       "    // Disable right mouse context menu.\n",
-       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
-       "        return false;\n",
-       "    });\n",
-       "\n",
-       "    function set_focus () {\n",
-       "        canvas.focus();\n",
-       "        canvas_div.focus();\n",
-       "    }\n",
-       "\n",
-       "    window.setTimeout(set_focus, 100);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_toolbar = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var nav_element = $('<div/>')\n",
-       "    nav_element.attr('style', 'width: 100%');\n",
-       "    this.root.append(nav_element);\n",
-       "\n",
-       "    // Define a callback function for later on.\n",
-       "    function toolbar_event(event) {\n",
-       "        return fig.toolbar_button_onclick(event['data']);\n",
-       "    }\n",
-       "    function toolbar_mouse_event(event) {\n",
-       "        return fig.toolbar_button_onmouseover(event['data']);\n",
-       "    }\n",
-       "\n",
-       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
-       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
-       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
-       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
-       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
-       "\n",
-       "        if (!name) {\n",
-       "            // put a spacer in here.\n",
-       "            continue;\n",
-       "        }\n",
-       "        var button = $('<button/>');\n",
-       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
-       "                        'ui-button-icon-only');\n",
-       "        button.attr('role', 'button');\n",
-       "        button.attr('aria-disabled', 'false');\n",
-       "        button.click(method_name, toolbar_event);\n",
-       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
-       "\n",
-       "        var icon_img = $('<span/>');\n",
-       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
-       "        icon_img.addClass(image);\n",
-       "        icon_img.addClass('ui-corner-all');\n",
-       "\n",
-       "        var tooltip_span = $('<span/>');\n",
-       "        tooltip_span.addClass('ui-button-text');\n",
-       "        tooltip_span.html(tooltip);\n",
-       "\n",
-       "        button.append(icon_img);\n",
-       "        button.append(tooltip_span);\n",
-       "\n",
-       "        nav_element.append(button);\n",
-       "    }\n",
-       "\n",
-       "    var fmt_picker_span = $('<span/>');\n",
-       "\n",
-       "    var fmt_picker = $('<select/>');\n",
-       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
-       "    fmt_picker_span.append(fmt_picker);\n",
-       "    nav_element.append(fmt_picker_span);\n",
-       "    this.format_dropdown = fmt_picker[0];\n",
-       "\n",
-       "    for (var ind in mpl.extensions) {\n",
-       "        var fmt = mpl.extensions[ind];\n",
-       "        var option = $(\n",
-       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
-       "        fmt_picker.append(option)\n",
-       "    }\n",
-       "\n",
-       "    // Add hover states to the ui-buttons\n",
-       "    $( \".ui-button\" ).hover(\n",
-       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
-       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
-       "    );\n",
-       "\n",
-       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
-       "    nav_element.append(status_bar);\n",
-       "    this.message = status_bar[0];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
-       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
-       "    // which will in turn request a refresh of the image.\n",
-       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.send_message = function(type, properties) {\n",
-       "    properties['type'] = type;\n",
-       "    properties['figure_id'] = this.id;\n",
-       "    this.ws.send(JSON.stringify(properties));\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.send_draw_message = function() {\n",
-       "    if (!this.waiting) {\n",
-       "        this.waiting = true;\n",
-       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
-       "    var format_dropdown = fig.format_dropdown;\n",
-       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
-       "    fig.ondownload(fig, format);\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
-       "    var size = msg['size'];\n",
-       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
-       "        fig._resize_canvas(size[0], size[1]);\n",
-       "        fig.send_message(\"refresh\", {});\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
-       "    var x0 = msg['x0'] / mpl.ratio;\n",
-       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
-       "    var x1 = msg['x1'] / mpl.ratio;\n",
-       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
-       "    x0 = Math.floor(x0) + 0.5;\n",
-       "    y0 = Math.floor(y0) + 0.5;\n",
-       "    x1 = Math.floor(x1) + 0.5;\n",
-       "    y1 = Math.floor(y1) + 0.5;\n",
-       "    var min_x = Math.min(x0, x1);\n",
-       "    var min_y = Math.min(y0, y1);\n",
-       "    var width = Math.abs(x1 - x0);\n",
-       "    var height = Math.abs(y1 - y0);\n",
-       "\n",
-       "    fig.rubberband_context.clearRect(\n",
-       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
-       "\n",
-       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
-       "    // Updates the figure title.\n",
-       "    fig.header.textContent = msg['label'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
-       "    var cursor = msg['cursor'];\n",
-       "    switch(cursor)\n",
-       "    {\n",
-       "    case 0:\n",
-       "        cursor = 'pointer';\n",
-       "        break;\n",
-       "    case 1:\n",
-       "        cursor = 'default';\n",
-       "        break;\n",
-       "    case 2:\n",
-       "        cursor = 'crosshair';\n",
-       "        break;\n",
-       "    case 3:\n",
-       "        cursor = 'move';\n",
-       "        break;\n",
-       "    }\n",
-       "    fig.rubberband_canvas.style.cursor = cursor;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
-       "    fig.message.textContent = msg['message'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
-       "    // Request the server to send over a new figure.\n",
-       "    fig.send_draw_message();\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
-       "    fig.image_mode = msg['mode'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.updated_canvas_event = function() {\n",
-       "    // Called whenever the canvas gets updated.\n",
-       "    this.send_message(\"ack\", {});\n",
-       "}\n",
-       "\n",
-       "// A function to construct a web socket function for onmessage handling.\n",
-       "// Called in the figure constructor.\n",
-       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
-       "    return function socket_on_message(evt) {\n",
-       "        if (evt.data instanceof Blob) {\n",
-       "            /* FIXME: We get \"Resource interpreted as Image but\n",
-       "             * transferred with MIME type text/plain:\" errors on\n",
-       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
-       "             * to be part of the websocket stream */\n",
-       "            evt.data.type = \"image/png\";\n",
-       "\n",
-       "            /* Free the memory for the previous frames */\n",
-       "            if (fig.imageObj.src) {\n",
-       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
-       "                    fig.imageObj.src);\n",
-       "            }\n",
-       "\n",
-       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
-       "                evt.data);\n",
-       "            fig.updated_canvas_event();\n",
-       "            fig.waiting = false;\n",
-       "            return;\n",
-       "        }\n",
-       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
-       "            fig.imageObj.src = evt.data;\n",
-       "            fig.updated_canvas_event();\n",
-       "            fig.waiting = false;\n",
-       "            return;\n",
-       "        }\n",
-       "\n",
-       "        var msg = JSON.parse(evt.data);\n",
-       "        var msg_type = msg['type'];\n",
-       "\n",
-       "        // Call the  \"handle_{type}\" callback, which takes\n",
-       "        // the figure and JSON message as its only arguments.\n",
-       "        try {\n",
-       "            var callback = fig[\"handle_\" + msg_type];\n",
-       "        } catch (e) {\n",
-       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
-       "            return;\n",
-       "        }\n",
-       "\n",
-       "        if (callback) {\n",
-       "            try {\n",
-       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
-       "                callback(fig, msg);\n",
-       "            } catch (e) {\n",
-       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
-       "            }\n",
-       "        }\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
-       "mpl.findpos = function(e) {\n",
-       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
-       "    var targ;\n",
-       "    if (!e)\n",
-       "        e = window.event;\n",
-       "    if (e.target)\n",
-       "        targ = e.target;\n",
-       "    else if (e.srcElement)\n",
-       "        targ = e.srcElement;\n",
-       "    if (targ.nodeType == 3) // defeat Safari bug\n",
-       "        targ = targ.parentNode;\n",
-       "\n",
-       "    // jQuery normalizes the pageX and pageY\n",
-       "    // pageX,Y are the mouse positions relative to the document\n",
-       "    // offset() returns the position of the element relative to the document\n",
-       "    var x = e.pageX - $(targ).offset().left;\n",
-       "    var y = e.pageY - $(targ).offset().top;\n",
-       "\n",
-       "    return {\"x\": x, \"y\": y};\n",
-       "};\n",
-       "\n",
-       "/*\n",
-       " * return a copy of an object with only non-object keys\n",
-       " * we need this to avoid circular references\n",
-       " * http://stackoverflow.com/a/24161582/3208463\n",
-       " */\n",
-       "function simpleKeys (original) {\n",
-       "  return Object.keys(original).reduce(function (obj, key) {\n",
-       "    if (typeof original[key] !== 'object')\n",
-       "        obj[key] = original[key]\n",
-       "    return obj;\n",
-       "  }, {});\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
-       "    var canvas_pos = mpl.findpos(event)\n",
-       "\n",
-       "    if (name === 'button_press')\n",
-       "    {\n",
-       "        this.canvas.focus();\n",
-       "        this.canvas_div.focus();\n",
-       "    }\n",
-       "\n",
-       "    var x = canvas_pos.x * mpl.ratio;\n",
-       "    var y = canvas_pos.y * mpl.ratio;\n",
-       "\n",
-       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
-       "                             step: event.step,\n",
-       "                             guiEvent: simpleKeys(event)});\n",
-       "\n",
-       "    /* This prevents the web browser from automatically changing to\n",
-       "     * the text insertion cursor when the button is pressed.  We want\n",
-       "     * to control all of the cursor setting manually through the\n",
-       "     * 'cursor' event from matplotlib */\n",
-       "    event.preventDefault();\n",
-       "    return false;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
-       "    // Handle any extra behaviour associated with a key event\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.key_event = function(event, name) {\n",
-       "\n",
-       "    // Prevent repeat events\n",
-       "    if (name == 'key_press')\n",
-       "    {\n",
-       "        if (event.which === this._key)\n",
-       "            return;\n",
-       "        else\n",
-       "            this._key = event.which;\n",
-       "    }\n",
-       "    if (name == 'key_release')\n",
-       "        this._key = null;\n",
-       "\n",
-       "    var value = '';\n",
-       "    if (event.ctrlKey && event.which != 17)\n",
-       "        value += \"ctrl+\";\n",
-       "    if (event.altKey && event.which != 18)\n",
-       "        value += \"alt+\";\n",
-       "    if (event.shiftKey && event.which != 16)\n",
-       "        value += \"shift+\";\n",
-       "\n",
-       "    value += 'k';\n",
-       "    value += event.which.toString();\n",
-       "\n",
-       "    this._key_event_extra(event, name);\n",
-       "\n",
-       "    this.send_message(name, {key: value,\n",
-       "                             guiEvent: simpleKeys(event)});\n",
-       "    return false;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
-       "    if (name == 'download') {\n",
-       "        this.handle_save(this, null);\n",
-       "    } else {\n",
-       "        this.send_message(\"toolbar_button\", {name: name});\n",
-       "    }\n",
-       "};\n",
-       "\n",
-       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
-       "    this.message.textContent = tooltip;\n",
-       "};\n",
-       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
-       "\n",
-       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
-       "\n",
-       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
-       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
-       "    // object with the appropriate methods. Currently this is a non binary\n",
-       "    // socket, so there is still some room for performance tuning.\n",
-       "    var ws = {};\n",
-       "\n",
-       "    ws.close = function() {\n",
-       "        comm.close()\n",
-       "    };\n",
-       "    ws.send = function(m) {\n",
-       "        //console.log('sending', m);\n",
-       "        comm.send(m);\n",
-       "    };\n",
-       "    // Register the callback with on_msg.\n",
-       "    comm.on_msg(function(msg) {\n",
-       "        //console.log('receiving', msg['content']['data'], msg);\n",
-       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
-       "        ws.onmessage(msg['content']['data'])\n",
-       "    });\n",
-       "    return ws;\n",
-       "}\n",
-       "\n",
-       "mpl.mpl_figure_comm = function(comm, msg) {\n",
-       "    // This is the function which gets called when the mpl process\n",
-       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
-       "\n",
-       "    var id = msg.content.data.id;\n",
-       "    // Get hold of the div created by the display call when the Comm\n",
-       "    // socket was opened in Python.\n",
-       "    var element = $(\"#\" + id);\n",
-       "    var ws_proxy = comm_websocket_adapter(comm)\n",
-       "\n",
-       "    function ondownload(figure, format) {\n",
-       "        window.open(figure.imageObj.src);\n",
-       "    }\n",
-       "\n",
-       "    var fig = new mpl.figure(id, ws_proxy,\n",
-       "                           ondownload,\n",
-       "                           element.get(0));\n",
-       "\n",
-       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
-       "    // web socket which is closed, not our websocket->open comm proxy.\n",
-       "    ws_proxy.onopen();\n",
-       "\n",
-       "    fig.parent_element = element.get(0);\n",
-       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
-       "    if (!fig.cell_info) {\n",
-       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
-       "        return;\n",
-       "    }\n",
-       "\n",
-       "    var output_index = fig.cell_info[2]\n",
-       "    var cell = fig.cell_info[0];\n",
-       "\n",
-       "};\n",
-       "\n",
-       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
-       "    var width = fig.canvas.width/mpl.ratio\n",
-       "    fig.root.unbind('remove')\n",
-       "\n",
-       "    // Update the output cell to use the data from the current canvas.\n",
-       "    fig.push_to_output();\n",
-       "    var dataURL = fig.canvas.toDataURL();\n",
-       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
-       "    // the notebook keyboard shortcuts fail.\n",
-       "    IPython.keyboard_manager.enable()\n",
-       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
-       "    fig.close_ws(fig, msg);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
-       "    fig.send_message('closing', msg);\n",
-       "    // fig.ws.close()\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
-       "    // Turn the data on the canvas into data in the output cell.\n",
-       "    var width = this.canvas.width/mpl.ratio\n",
-       "    var dataURL = this.canvas.toDataURL();\n",
-       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.updated_canvas_event = function() {\n",
-       "    // Tell IPython that the notebook contents must change.\n",
-       "    IPython.notebook.set_dirty(true);\n",
-       "    this.send_message(\"ack\", {});\n",
-       "    var fig = this;\n",
-       "    // Wait a second, then push the new image to the DOM so\n",
-       "    // that it is saved nicely (might be nice to debounce this).\n",
-       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_toolbar = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var nav_element = $('<div/>')\n",
-       "    nav_element.attr('style', 'width: 100%');\n",
-       "    this.root.append(nav_element);\n",
-       "\n",
-       "    // Define a callback function for later on.\n",
-       "    function toolbar_event(event) {\n",
-       "        return fig.toolbar_button_onclick(event['data']);\n",
-       "    }\n",
-       "    function toolbar_mouse_event(event) {\n",
-       "        return fig.toolbar_button_onmouseover(event['data']);\n",
-       "    }\n",
-       "\n",
-       "    for(var toolbar_ind in mpl.toolbar_items){\n",
-       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
-       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
-       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
-       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
-       "\n",
-       "        if (!name) { continue; };\n",
-       "\n",
-       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
-       "        button.click(method_name, toolbar_event);\n",
-       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
-       "        nav_element.append(button);\n",
-       "    }\n",
-       "\n",
-       "    // Add the status bar.\n",
-       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
-       "    nav_element.append(status_bar);\n",
-       "    this.message = status_bar[0];\n",
-       "\n",
-       "    // Add the close button to the window.\n",
-       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
-       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
-       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
-       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
-       "    buttongrp.append(button);\n",
-       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
-       "    titlebar.prepend(buttongrp);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._root_extra_style = function(el){\n",
-       "    var fig = this\n",
-       "    el.on(\"remove\", function(){\n",
-       "\tfig.close_ws(fig, {});\n",
-       "    });\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
-       "    // this is important to make the div 'focusable\n",
-       "    el.attr('tabindex', 0)\n",
-       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
-       "    // off when our div gets focus\n",
-       "\n",
-       "    // location in version 3\n",
-       "    if (IPython.notebook.keyboard_manager) {\n",
-       "        IPython.notebook.keyboard_manager.register_events(el);\n",
-       "    }\n",
-       "    else {\n",
-       "        // location in version 2\n",
-       "        IPython.keyboard_manager.register_events(el);\n",
-       "    }\n",
-       "\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
-       "    var manager = IPython.notebook.keyboard_manager;\n",
-       "    if (!manager)\n",
-       "        manager = IPython.keyboard_manager;\n",
-       "\n",
-       "    // Check for shift+enter\n",
-       "    if (event.shiftKey && event.which == 13) {\n",
-       "        this.canvas_div.blur();\n",
-       "        event.shiftKey = false;\n",
-       "        // Send a \"J\" for go to next cell\n",
-       "        event.which = 74;\n",
-       "        event.keyCode = 74;\n",
-       "        manager.command_mode();\n",
-       "        manager.handle_keydown(event);\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
-       "    fig.ondownload(fig, null);\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.find_output_cell = function(html_output) {\n",
-       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
-       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
-       "    // IPython event is triggered only after the cells have been serialised, which for\n",
-       "    // our purposes (turning an active figure into a static one), is too late.\n",
-       "    var cells = IPython.notebook.get_cells();\n",
-       "    var ncells = cells.length;\n",
-       "    for (var i=0; i<ncells; i++) {\n",
-       "        var cell = cells[i];\n",
-       "        if (cell.cell_type === 'code'){\n",
-       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
-       "                var data = cell.output_area.outputs[j];\n",
-       "                if (data.data) {\n",
-       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
-       "                    data = data.data;\n",
-       "                }\n",
-       "                if (data['text/html'] == html_output) {\n",
-       "                    return [cell, data, j];\n",
-       "                }\n",
-       "            }\n",
-       "        }\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "// Register the function which deals with the matplotlib target/channel.\n",
-       "// The kernel may be null if the page has been refreshed.\n",
-       "if (IPython.notebook.kernel != null) {\n",
-       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
-       "}\n"
-      ],
-      "text/plain": [
-       "<IPython.core.display.Javascript object>"
-      ]
-     },
-     "metadata": {},
-     "output_type": "display_data"
-    },
-    {
-     "data": {
-      "text/html": [
-       "<img src=\"\" width=\"1000\">"
-      ],
-      "text/plain": [
-       "<IPython.core.display.HTML object>"
-      ]
-     },
-     "metadata": {},
-     "output_type": "display_data"
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x12e9ae7d0>"
-      ]
-     },
-     "execution_count": 29,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "df_results = %sql SELECT * FROM iris_multi_model_info ORDER BY validation_loss ASC LIMIT 7;\n",
-    "df_results = df_results.DataFrame()\n",
-    "\n",
-    "df_summary = %sql SELECT * FROM iris_multi_model_summary;\n",
-    "df_summary = df_summary.DataFrame()\n",
-    "\n",
-    "#set up plots\n",
-    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
-    "fig.legend(ncol=4)\n",
-    "fig.tight_layout()\n",
-    "\n",
-    "ax_metric = axs[0]\n",
-    "ax_loss = axs[1]\n",
-    "\n",
-    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
-    "ax_metric.set_xlabel('Iteration')\n",
-    "ax_metric.set_ylabel('Metric')\n",
-    "ax_metric.set_title('Validation metric curve')\n",
-    "\n",
-    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
-    "ax_loss.set_xlabel('Iteration')\n",
-    "ax_loss.set_ylabel('Loss')\n",
-    "ax_loss.set_title('Validation loss curve')\n",
-    "\n",
-    "iters = df_summary['metrics_iters'][0]\n",
-    "\n",
-    "for mst_key in df_results['mst_key']:\n",
-    "    df_output_info = %sql SELECT validation_metrics,validation_loss FROM iris_multi_model_info WHERE mst_key = $mst_key\n",
-    "    df_output_info = df_output_info.DataFrame()\n",
-    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
-    "    validation_loss = df_output_info['validation_loss'][0]\n",
-    "    \n",
-    "    ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
-    "    ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
-    "\n",
-    "plt.legend()\n",
-    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"pred_prob\"></a>\n",
-    "# 2.  Predict probabilities\n",
-    "\n",
-    "Predict with probabilities for each class:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 30,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "30 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>prob_Iris-setosa</th>\n",
-       "        <th>prob_Iris-versicolor</th>\n",
-       "        <th>prob_Iris-virginica</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>0.9999416</td>\n",
-       "        <td>5.8360623e-05</td>\n",
-       "        <td>3.9093355e-12</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>0.99998116</td>\n",
-       "        <td>1.8880675e-05</td>\n",
-       "        <td>2.5342377e-13</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>0.99994814</td>\n",
-       "        <td>5.1881765e-05</td>\n",
-       "        <td>2.5964983e-12</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>0.99996114</td>\n",
-       "        <td>3.8810744e-05</td>\n",
-       "        <td>1.176443e-12</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>0.99992573</td>\n",
-       "        <td>7.4317446e-05</td>\n",
-       "        <td>5.4237942e-12</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>19</td>\n",
-       "        <td>0.9999845</td>\n",
-       "        <td>1.5514812e-05</td>\n",
-       "        <td>1.034207e-13</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>25</td>\n",
-       "        <td>0.99992156</td>\n",
-       "        <td>7.845682e-05</td>\n",
-       "        <td>3.7364413e-12</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>26</td>\n",
-       "        <td>0.9998591</td>\n",
-       "        <td>0.00014085071</td>\n",
-       "        <td>2.0146884e-11</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>28</td>\n",
-       "        <td>0.9999734</td>\n",
-       "        <td>2.6542659e-05</td>\n",
-       "        <td>4.8342347e-13</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>38</td>\n",
-       "        <td>0.99992573</td>\n",
-       "        <td>7.4317446e-05</td>\n",
-       "        <td>5.4237942e-12</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>44</td>\n",
-       "        <td>0.99990726</td>\n",
-       "        <td>9.278052e-05</td>\n",
-       "        <td>6.9040372e-12</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>45</td>\n",
-       "        <td>0.999964</td>\n",
-       "        <td>3.6013742e-05</td>\n",
-       "        <td>5.7615945e-13</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>51</td>\n",
-       "        <td>0.00025041687</td>\n",
-       "        <td>0.99780566</td>\n",
-       "        <td>0.0019439155</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>53</td>\n",
-       "        <td>1.843269e-05</td>\n",
-       "        <td>0.9889865</td>\n",
-       "        <td>0.010995116</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>57</td>\n",
-       "        <td>2.4158675e-05</td>\n",
-       "        <td>0.99005336</td>\n",
-       "        <td>0.00992243</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>59</td>\n",
-       "        <td>0.00011159414</td>\n",
-       "        <td>0.9942708</td>\n",
-       "        <td>0.0056176083</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>62</td>\n",
-       "        <td>0.00014697485</td>\n",
-       "        <td>0.99189115</td>\n",
-       "        <td>0.007961868</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>69</td>\n",
-       "        <td>8.6406266e-07</td>\n",
-       "        <td>0.6961896</td>\n",
-       "        <td>0.30380967</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>75</td>\n",
-       "        <td>0.0005239165</td>\n",
-       "        <td>0.9965855</td>\n",
-       "        <td>0.0028905326</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>77</td>\n",
-       "        <td>1.5155997e-05</td>\n",
-       "        <td>0.97978914</td>\n",
-       "        <td>0.020195633</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>97</td>\n",
-       "        <td>0.00023696794</td>\n",
-       "        <td>0.9938279</td>\n",
-       "        <td>0.005935215</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>102</td>\n",
-       "        <td>1.3247301e-09</td>\n",
-       "        <td>0.18419608</td>\n",
-       "        <td>0.8158039</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>107</td>\n",
-       "        <td>2.5100556e-08</td>\n",
-       "        <td>0.30281228</td>\n",
-       "        <td>0.69718766</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>114</td>\n",
-       "        <td>3.2222575e-10</td>\n",
-       "        <td>0.08682407</td>\n",
-       "        <td>0.913176</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>118</td>\n",
-       "        <td>5.33606e-11</td>\n",
-       "        <td>0.34179842</td>\n",
-       "        <td>0.6582016</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>120</td>\n",
-       "        <td>9.134116e-09</td>\n",
-       "        <td>0.27099058</td>\n",
-       "        <td>0.72900945</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>122</td>\n",
-       "        <td>2.9710499e-09</td>\n",
-       "        <td>0.21993305</td>\n",
-       "        <td>0.7800669</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>132</td>\n",
-       "        <td>5.2177818e-09</td>\n",
-       "        <td>0.8370931</td>\n",
-       "        <td>0.16290687</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>146</td>\n",
-       "        <td>1.4404147e-09</td>\n",
-       "        <td>0.2293714</td>\n",
-       "        <td>0.7706286</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>147</td>\n",
-       "        <td>3.8019614e-09</td>\n",
-       "        <td>0.2240861</td>\n",
-       "        <td>0.77591395</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(3, 0.9999416, 5.8360623e-05, 3.9093355e-12),\n",
-       " (5, 0.99998116, 1.8880675e-05, 2.5342377e-13),\n",
-       " (7, 0.99994814, 5.1881765e-05, 2.5964983e-12),\n",
-       " (8, 0.99996114, 3.8810744e-05, 1.176443e-12),\n",
-       " (10, 0.99992573, 7.4317446e-05, 5.4237942e-12),\n",
-       " (19, 0.9999845, 1.5514812e-05, 1.034207e-13),\n",
-       " (25, 0.99992156, 7.845682e-05, 3.7364413e-12),\n",
-       " (26, 0.9998591, 0.00014085071, 2.0146884e-11),\n",
-       " (28, 0.9999734, 2.6542659e-05, 4.8342347e-13),\n",
-       " (38, 0.99992573, 7.4317446e-05, 5.4237942e-12),\n",
-       " (44, 0.99990726, 9.278052e-05, 6.9040372e-12),\n",
-       " (45, 0.999964, 3.6013742e-05, 5.7615945e-13),\n",
-       " (51, 0.00025041687, 0.99780566, 0.0019439155),\n",
-       " (53, 1.843269e-05, 0.9889865, 0.010995116),\n",
-       " (57, 2.4158675e-05, 0.99005336, 0.00992243),\n",
-       " (59, 0.00011159414, 0.9942708, 0.0056176083),\n",
-       " (62, 0.00014697485, 0.99189115, 0.007961868),\n",
-       " (69, 8.6406266e-07, 0.6961896, 0.30380967),\n",
-       " (75, 0.0005239165, 0.9965855, 0.0028905326),\n",
-       " (77, 1.5155997e-05, 0.97978914, 0.020195633),\n",
-       " (97, 0.00023696794, 0.9938279, 0.005935215),\n",
-       " (102, 1.3247301e-09, 0.18419608, 0.8158039),\n",
-       " (107, 2.5100556e-08, 0.30281228, 0.69718766),\n",
-       " (114, 3.2222575e-10, 0.08682407, 0.913176),\n",
-       " (118, 5.33606e-11, 0.34179842, 0.6582016),\n",
-       " (120, 9.134116e-09, 0.27099058, 0.72900945),\n",
-       " (122, 2.9710499e-09, 0.21993305, 0.7800669),\n",
-       " (132, 5.2177818e-09, 0.8370931, 0.16290687),\n",
-       " (146, 1.4404147e-09, 0.2293714, 0.7706286),\n",
-       " (147, 3.8019614e-09, 0.2240861, 0.77591395)]"
-      ]
-     },
-     "execution_count": 30,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS iris_predict;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_predict('iris_multi_model', -- model\n",
-    "                                   'iris_test',        -- test_table\n",
-    "                                   'id',               -- id column\n",
-    "                                   'attributes',       -- independent var\n",
-    "                                   'iris_predict',     -- output table\n",
-    "                                    'prob',            -- prediction type\n",
-    "                                    FALSE,             -- use gpus\n",
-    "                                    3                  -- mst_key to use\n",
-    "                                   );\n",
-    "\n",
-    "SELECT * FROM iris_predict ORDER BY id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"warm_start\"></a>\n",
-    "# 3.  Warm start\n",
-    "\n",
-    "Next, use the warm_start parameter to continue learning, using the coefficients from the run above. Note that we don't drop the model table or model summary table:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 31,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>madlib_keras_fit_multiple_model</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td></td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[('',)]"
-      ]
-     },
-     "execution_count": 31,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
-    "                                              'iris_multi_model',     -- model_output_table\n",
-    "                                              'mst_table',            -- model_selection_table\n",
-    "                                               3,                     -- num_iterations\n",
-    "                                               FALSE,                 -- use gpus\n",
-    "                                              'iris_test_packed',     -- validation dataset\n",
-    "                                               1,                     -- metrics compute frequency\n",
-    "                                               TRUE,                  -- warm start\n",
-    "                                              'Sophie L.',            -- name\n",
-    "                                              'Simple MLP for iris dataset'  -- description\n",
-    "                                             );"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View summary:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 32,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>validation_table</th>\n",
-       "        <th>model</th>\n",
-       "        <th>model_info</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>metrics_compute_frequency</th>\n",
-       "        <th>warm_start</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>madlib_version</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>metrics_iters</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>iris_train_packed</td>\n",
-       "        <td>iris_test_packed</td>\n",
-       "        <td>iris_multi_model</td>\n",
-       "        <td>iris_multi_model_info</td>\n",
-       "        <td>class_text</td>\n",
-       "        <td>attributes</td>\n",
-       "        <td>model_arch_library</td>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "        <td>True</td>\n",
-       "        <td>Sophie L.</td>\n",
-       "        <td>Simple MLP for iris dataset</td>\n",
-       "        <td>2019-12-18 22:37:57.948805</td>\n",
-       "        <td>2019-12-18 22:38:43.967187</td>\n",
-       "        <td>1.17-dev</td>\n",
-       "        <td>3</td>\n",
-       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
-       "        <td>character varying</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>[1, 2, 3]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'iris_train_packed', u'iris_test_packed', u'iris_multi_model', u'iris_multi_model_info', u'class_text', u'attributes', u'model_arch_library', 3, 1, True, u'Sophie L.', u'Simple MLP for iris dataset', datetime.datetime(2019, 12, 18, 22, 37, 57, 948805), datetime.datetime(2019, 12, 18, 22, 38, 43, 967187), u'1.17-dev', 3, [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], u'character varying', 1.0, [1, 2, 3])]"
-      ]
-     },
-     "execution_count": 32,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_multi_model_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View performance of each model:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 33,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "12 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>mst_key</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.17091703414917, 0.163390159606934, 0.155634164810181]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.958333313465</td>\n",
-       "        <td>0.31917694211</td>\n",
-       "        <td>[0.958333313465118, 0.958333313465118, 0.958333313465118]</td>\n",
-       "        <td>[0.348434448242188, 0.334388434886932, 0.319176942110062]</td>\n",
-       "        <td>1.0</td>\n",
-       "        <td>0.272621482611</td>\n",
-       "        <td>[1.0, 1.0, 1.0]</td>\n",
-       "        <td>[0.306039541959763, 0.28966349363327, 0.272621482610703]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.172316074371338, 0.188217163085938, 0.503840208053589]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.899999976158</td>\n",
-       "        <td>0.193531006575</td>\n",
-       "        <td>[0.958333313465118, 0.925000011920929, 0.899999976158142]</td>\n",
-       "        <td>[0.147025644779205, 0.144938006997108, 0.193531006574631]</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.153077676892</td>\n",
-       "        <td>[0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
-       "        <td>[0.132363379001617, 0.116448685526848, 0.153077676892281]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.147105932235718, 0.158121824264526, 0.174723863601685]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.100400544703</td>\n",
-       "        <td>[0.966666638851166, 0.908333361148834, 0.966666638851166]</td>\n",
-       "        <td>[0.112152323126793, 0.197978660464287, 0.100400544703007]</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.0844493880868</td>\n",
-       "        <td>[0.933333337306976, 0.966666638851166, 0.966666638851166]</td>\n",
-       "        <td>[0.0945712551474571, 0.170254677534103, 0.0844493880867958]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.224463939666748, 0.412797927856445, 0.193319797515869]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.958333313465</td>\n",
-       "        <td>0.139601364732</td>\n",
-       "        <td>[0.966666638851166, 0.966666638851166, 0.958333313465118]</td>\n",
-       "        <td>[0.122705578804016, 0.0809410735964775, 0.139601364731789]</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.131209135056</td>\n",
-       "        <td>[0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
-       "        <td>[0.115778811275959, 0.0698963403701782, 0.131209135055542]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.160850048065186, 0.224483013153076, 0.163106918334961]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.0839553326368</td>\n",
-       "        <td>[0.966666638851166, 0.908333361148834, 0.966666638851166]</td>\n",
-       "        <td>[0.124577566981316, 0.196399554610252, 0.0839553326368332]</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.074150800705</td>\n",
-       "        <td>[0.966666638851166, 0.866666674613953, 0.966666638851166]</td>\n",
-       "        <td>[0.137340381741524, 0.232466518878937, 0.0741508007049561]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.14374303817749, 0.154287099838257, 0.17367696762085]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.966666638851</td>\n",
-       "        <td>0.0860244855285</td>\n",
-       "        <td>[0.966666638851166, 0.841666638851166, 0.966666638851166]</td>\n",
-       "        <td>[0.0824147835373878, 0.337884455919266, 0.0860244855284691]</td>\n",
-       "        <td>0.933333337307</td>\n",
-       "        <td>0.0704526007175</td>\n",
-       "        <td>[0.966666638851166, 0.866666674613953, 0.933333337306976]</td>\n",
-       "        <td>[0.0690516456961632, 0.295713990926743, 0.0704526007175446]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.155812978744507, 0.158360004425049, 0.159363031387329]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.833333313465</td>\n",
-       "        <td>0.344228476286</td>\n",
-       "        <td>[0.666666686534882, 0.675000011920929, 0.833333313465118]</td>\n",
-       "        <td>[1.01126325130463, 1.33927237987518, 0.344228476285934]</td>\n",
-       "        <td>0.800000011921</td>\n",
-       "        <td>0.305708706379</td>\n",
-       "        <td>[0.699999988079071, 0.699999988079071, 0.800000011920929]</td>\n",
-       "        <td>[1.02303433418274, 1.36952638626099, 0.305708706378937]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>11</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.187958955764771, 0.186024904251099, 0.501762866973877]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.725000023842</td>\n",
-       "        <td>0.423261642456</td>\n",
-       "        <td>[0.658333361148834, 0.658333361148834, 0.725000023841858]</td>\n",
-       "        <td>[0.46866175532341, 0.445532470941544, 0.423261642456055]</td>\n",
-       "        <td>0.699999988079</td>\n",
-       "        <td>0.378630697727</td>\n",
-       "        <td>[0.699999988079071, 0.699999988079071, 0.699999988079071]</td>\n",
-       "        <td>[0.422465175390244, 0.398104608058929, 0.378630697727203]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>1</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>0.7900390625</td>\n",
-       "        <td>[0.176413059234619, 0.169157981872559, 0.15624213218689]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.675000011921</td>\n",
-       "        <td>0.470171242952</td>\n",
-       "        <td>[0.641666650772095, 0.658333361148834, 0.675000011920929]</td>\n",
-       "        <td>[0.504463493824005, 0.486825525760651, 0.470171242952347]</td>\n",
-       "        <td>0.699999988079</td>\n",
-       "        <td>0.436036229134</td>\n",
-       "        <td>[0.699999988079071, 0.699999988079071, 0.699999988079071]</td>\n",
-       "        <td>[0.470719456672668, 0.452698260545731, 0.436036229133606]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>12</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.164397954940796, 0.486438035964966, 0.192479133605957]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.550000011921</td>\n",
-       "        <td>0.975017726421</td>\n",
-       "        <td>[0.508333325386047, 0.533333361148834, 0.550000011920929]</td>\n",
-       "        <td>[1.00239539146423, 0.986684203147888, 0.975017726421356]</td>\n",
-       "        <td>0.466666668653</td>\n",
-       "        <td>0.981434583664</td>\n",
-       "        <td>[0.5, 0.466666668653488, 0.466666668653488]</td>\n",
-       "        <td>[1.00223970413208, 0.989481270313263, 0.98143458366394]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>8</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=8,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.467766046524048, 0.198179006576538, 0.186810970306396]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.341666668653</td>\n",
-       "        <td>1.10613942146</td>\n",
-       "        <td>[0.316666662693024, 0.316666662693024, 0.341666668653488]</td>\n",
-       "        <td>[1.1275190114975, 1.10920584201813, 1.10613942146301]</td>\n",
-       "        <td>0.300000011921</td>\n",
-       "        <td>1.10817503929</td>\n",
-       "        <td>[0.400000005960464, 0.400000005960464, 0.300000011920929]</td>\n",
-       "        <td>[1.10070872306824, 1.09047472476959, 1.10817503929138]</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>7</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=4,epochs=1</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>1.2197265625</td>\n",
-       "        <td>[0.467660903930664, 0.195011138916016, 0.185934066772461]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.341666668653</td>\n",
-       "        <td>1.10524618626</td>\n",
-       "        <td>[0.316666662693024, 0.341666668653488, 0.341666668653488]</td>\n",
-       "        <td>[1.10246300697327, 1.09976887702942, 1.10524618625641]</td>\n",
-       "        <td>0.300000011921</td>\n",
-       "        <td>1.10809886456</td>\n",
-       "        <td>[0.400000005960464, 0.300000011920929, 0.300000011920929]</td>\n",
-       "        <td>[1.09229254722595, 1.09808218479156, 1.10809886455536]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(6, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 0.7900390625, [0.17091703414917, 0.163390159606934, 0.155634164810181], [u'accuracy'], 0.958333313465, 0.31917694211, [0.958333313465118, 0.958333313465118, 0.958333313465118], [0.348434448242188, 0.334388434886932, 0.319176942110062], 1.0, 0.272621482611, [1.0, 1.0, 1.0], [0.306039541959763, 0.28966349363327, 0.272621482610703]),\n",
-       " (10, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 1.2197265625, [0.172316074371338, 0.188217163085938, 0.503840208053589], [u'accuracy'], 0.899999976158, 0.193531006575, [0.958333313465118, 0.925000011920929, 0.899999976158142], [0.147025644779205, 0.144938006997108, 0.193531006574631], 0.966666638851, 0.153077676892, [0.966666638851166, 0.966666638851166, 0.966666638851166], [0.132363379001617, 0.116448685526848, 0.153077676892281]),\n",
-       " (4, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 0.7900390625, [0.147105932235718, 0.158121824264526, 0.174723863601685], [u'accuracy'], 0.966666638851, 0.100400544703, [0.966666638851166, 0.908333361148834, 0.966666638851166], [0.112152323126793, 0.197978660464287, 0.100400544703007], 0.966666638851, 0.0844493880868, [0.933333337306976, 0.966666638851166, 0.966666638851166], [0.0945712551474571, 0.170254677534103, 0.0844493880867958]),\n",
-       " (9, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 1.2197265625, [0.224463939666748, 0.412797927856445, 0.193319797515869], [u'accuracy'], 0.958333313465, 0.139601364732, [0.966666638851166, 0.966666638851166, 0.958333313465118], [0.122705578804016, 0.0809410735964775, 0.139601364731789], 0.966666638851, 0.131209135056, [0.966666638851166, 0.966666638851166, 0.966666638851166], [0.115778811275959, 0.0698963403701782, 0.131209135055542]),\n",
-       " (1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 0.7900390625, [0.160850048065186, 0.224483013153076, 0.163106918334961], [u'accuracy'], 0.966666638851, 0.0839553326368, [0.966666638851166, 0.908333361148834, 0.966666638851166], [0.124577566981316, 0.196399554610252, 0.0839553326368332], 0.966666638851, 0.074150800705, [0.966666638851166, 0.866666674613953, 0.966666638851166], [0.137340381741524, 0.232466518878937, 0.0741508007049561]),\n",
-       " (3, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 0.7900390625, [0.14374303817749, 0.154287099838257, 0.17367696762085], [u'accuracy'], 0.966666638851, 0.0860244855285, [0.966666638851166, 0.841666638851166, 0.966666638851166], [0.0824147835373878, 0.337884455919266, 0.0860244855284691], 0.933333337307, 0.0704526007175, [0.966666638851166, 0.866666674613953, 0.933333337306976], [0.0690516456961632, 0.295713990926743, 0.0704526007175446]),\n",
-       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 0.7900390625, [0.155812978744507, 0.158360004425049, 0.159363031387329], [u'accuracy'], 0.833333313465, 0.344228476286, [0.666666686534882, 0.675000011920929, 0.833333313465118], [1.01126325130463, 1.33927237987518, 0.344228476285934], 0.800000011921, 0.305708706379, [0.699999988079071, 0.699999988079071, 0.800000011920929], [1.02303433418274, 1.36952638626099, 0.305708706378937]),\n",
-       " (11, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 1.2197265625, [0.187958955764771, 0.186024904251099, 0.501762866973877], [u'accuracy'], 0.725000023842, 0.423261642456, [0.658333361148834, 0.658333361148834, 0.725000023841858], [0.46866175532341, 0.445532470941544, 0.423261642456055], 0.699999988079, 0.378630697727, [0.699999988079071, 0.699999988079071, 0.699999988079071], [0.422465175390244, 0.398104608058929, 0.378630697727203]),\n",
-       " (5, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 0.7900390625, [0.176413059234619, 0.169157981872559, 0.15624213218689], [u'accuracy'], 0.675000011921, 0.470171242952, [0.641666650772095, 0.658333361148834, 0.675000011920929], [0.504463493824005, 0.486825525760651, 0.470171242952347], 0.699999988079, 0.436036229134, [0.699999988079071, 0.699999988079071, 0.699999988079071], [0.470719456672668, 0.452698260545731, 0.436036229133606]),\n",
-       " (12, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 1.2197265625, [0.164397954940796, 0.486438035964966, 0.192479133605957], [u'accuracy'], 0.550000011921, 0.975017726421, [0.508333325386047, 0.533333361148834, 0.550000011920929], [1.00239539146423, 0.986684203147888, 0.975017726421356], 0.466666668653, 0.981434583664, [0.5, 0.466666668653488, 0.466666668653488], [1.00223970413208, 0.989481270313263, 0.98143458366394]),\n",
-       " (8, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1', u'madlib_keras', 1.2197265625, [0.467766046524048, 0.198179006576538, 0.186810970306396], [u'accuracy'], 0.341666668653, 1.10613942146, [0.316666662693024, 0.316666662693024, 0.341666668653488], [1.1275190114975, 1.10920584201813, 1.10613942146301], 0.300000011921, 1.10817503929, [0.400000005960464, 0.400000005960464, 0.300000011920929], [1.10070872306824, 1.09047472476959, 1.10817503929138]),\n",
-       " (7, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1', u'madlib_keras', 1.2197265625, [0.467660903930664, 0.195011138916016, 0.185934066772461], [u'accuracy'], 0.341666668653, 1.10524618626, [0.316666662693024, 0.341666668653488, 0.341666668653488], [1.10246300697327, 1.09976887702942, 1.10524618625641], 0.300000011921, 1.10809886456, [0.400000005960464, 0.300000011920929, 0.300000011920929], [1.09229254722595, 1.09808218479156, 1.10809886455536])]"
-      ]
-     },
-     "execution_count": 33,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM iris_multi_model_info ORDER BY validation_metrics_final DESC;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Plot validation results:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 34,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "7 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "application/javascript": [
-       "/* Put everything inside the global mpl namespace */\n",
-       "window.mpl = {};\n",
-       "\n",
-       "\n",
-       "mpl.get_websocket_type = function() {\n",
-       "    if (typeof(WebSocket) !== 'undefined') {\n",
-       "        return WebSocket;\n",
-       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
-       "        return MozWebSocket;\n",
-       "    } else {\n",
-       "        alert('Your browser does not have WebSocket support.' +\n",
-       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
-       "              'Firefox 4 and 5 are also supported but you ' +\n",
-       "              'have to enable WebSockets in about:config.');\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
-       "    this.id = figure_id;\n",
-       "\n",
-       "    this.ws = websocket;\n",
-       "\n",
-       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
-       "\n",
-       "    if (!this.supports_binary) {\n",
-       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
-       "        if (warnings) {\n",
-       "            warnings.style.display = 'block';\n",
-       "            warnings.textContent = (\n",
-       "                \"This browser does not support binary websocket messages. \" +\n",
-       "                    \"Performance may be slow.\");\n",
-       "        }\n",
-       "    }\n",
-       "\n",
-       "    this.imageObj = new Image();\n",
-       "\n",
-       "    this.context = undefined;\n",
-       "    this.message = undefined;\n",
-       "    this.canvas = undefined;\n",
-       "    this.rubberband_canvas = undefined;\n",
-       "    this.rubberband_context = undefined;\n",
-       "    this.format_dropdown = undefined;\n",
-       "\n",
-       "    this.image_mode = 'full';\n",
-       "\n",
-       "    this.root = $('<div/>');\n",
-       "    this._root_extra_style(this.root)\n",
-       "    this.root.attr('style', 'display: inline-block');\n",
-       "\n",
-       "    $(parent_element).append(this.root);\n",
-       "\n",
-       "    this._init_header(this);\n",
-       "    this._init_canvas(this);\n",
-       "    this._init_toolbar(this);\n",
-       "\n",
-       "    var fig = this;\n",
-       "\n",
-       "    this.waiting = false;\n",
-       "\n",
-       "    this.ws.onopen =  function () {\n",
-       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
-       "            fig.send_message(\"send_image_mode\", {});\n",
-       "            if (mpl.ratio != 1) {\n",
-       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
-       "            }\n",
-       "            fig.send_message(\"refresh\", {});\n",
-       "        }\n",
-       "\n",
-       "    this.imageObj.onload = function() {\n",
-       "            if (fig.image_mode == 'full') {\n",
-       "                // Full images could contain transparency (where diff images\n",
-       "                // almost always do), so we need to clear the canvas so that\n",
-       "                // there is no ghosting.\n",
-       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
-       "            }\n",
-       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
-       "        };\n",
-       "\n",
-       "    this.imageObj.onunload = function() {\n",
-       "        fig.ws.close();\n",
-       "    }\n",
-       "\n",
-       "    this.ws.onmessage = this._make_on_message_function(this);\n",
-       "\n",
-       "    this.ondownload = ondownload;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_header = function() {\n",
-       "    var titlebar = $(\n",
-       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
-       "        'ui-helper-clearfix\"/>');\n",
-       "    var titletext = $(\n",
-       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
-       "        'text-align: center; padding: 3px;\"/>');\n",
-       "    titlebar.append(titletext)\n",
-       "    this.root.append(titlebar);\n",
-       "    this.header = titletext[0];\n",
-       "}\n",
-       "\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
-       "\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
-       "\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_canvas = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var canvas_div = $('<div/>');\n",
-       "\n",
-       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
-       "\n",
-       "    function canvas_keyboard_event(event) {\n",
-       "        return fig.key_event(event, event['data']);\n",
-       "    }\n",
-       "\n",
-       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
-       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
-       "    this.canvas_div = canvas_div\n",
-       "    this._canvas_extra_style(canvas_div)\n",
-       "    this.root.append(canvas_div);\n",
-       "\n",
-       "    var canvas = $('<canvas/>');\n",
-       "    canvas.addClass('mpl-canvas');\n",
-       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
-       "\n",
-       "    this.canvas = canvas[0];\n",
-       "    this.context = canvas[0].getContext(\"2d\");\n",
-       "\n",
-       "    var backingStore = this.context.backingStorePixelRatio ||\n",
-       "\tthis.context.webkitBackingStorePixelRatio ||\n",
-       "\tthis.context.mozBackingStorePixelRatio ||\n",
-       "\tthis.context.msBackingStorePixelRatio ||\n",
-       "\tthis.context.oBackingStorePixelRatio ||\n",
-       "\tthis.context.backingStorePixelRatio || 1;\n",
-       "\n",
-       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
-       "\n",
-       "    var rubberband = $('<canvas/>');\n",
-       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
-       "\n",
-       "    var pass_mouse_events = true;\n",
-       "\n",
-       "    canvas_div.resizable({\n",
-       "        start: function(event, ui) {\n",
-       "            pass_mouse_events = false;\n",
-       "        },\n",
-       "        resize: function(event, ui) {\n",
-       "            fig.request_resize(ui.size.width, ui.size.height);\n",
-       "        },\n",
-       "        stop: function(event, ui) {\n",
-       "            pass_mouse_events = true;\n",
-       "            fig.request_resize(ui.size.width, ui.size.height);\n",
-       "        },\n",
-       "    });\n",
-       "\n",
-       "    function mouse_event_fn(event) {\n",
-       "        if (pass_mouse_events)\n",
-       "            return fig.mouse_event(event, event['data']);\n",
-       "    }\n",
-       "\n",
-       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
-       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
-       "    // Throttle sequential mouse events to 1 every 20ms.\n",
-       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
-       "\n",
-       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
-       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
-       "\n",
-       "    canvas_div.on(\"wheel\", function (event) {\n",
-       "        event = event.originalEvent;\n",
-       "        event['data'] = 'scroll'\n",
-       "        if (event.deltaY < 0) {\n",
-       "            event.step = 1;\n",
-       "        } else {\n",
-       "            event.step = -1;\n",
-       "        }\n",
-       "        mouse_event_fn(event);\n",
-       "    });\n",
-       "\n",
-       "    canvas_div.append(canvas);\n",
-       "    canvas_div.append(rubberband);\n",
-       "\n",
-       "    this.rubberband = rubberband;\n",
-       "    this.rubberband_canvas = rubberband[0];\n",
-       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
-       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
-       "\n",
-       "    this._resize_canvas = function(width, height) {\n",
-       "        // Keep the size of the canvas, canvas container, and rubber band\n",
-       "        // canvas in synch.\n",
-       "        canvas_div.css('width', width)\n",
-       "        canvas_div.css('height', height)\n",
-       "\n",
-       "        canvas.attr('width', width * mpl.ratio);\n",
-       "        canvas.attr('height', height * mpl.ratio);\n",
-       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
-       "\n",
-       "        rubberband.attr('width', width);\n",
-       "        rubberband.attr('height', height);\n",
-       "    }\n",
-       "\n",
-       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
-       "    // upon first draw.\n",
-       "    this._resize_canvas(600, 600);\n",
-       "\n",
-       "    // Disable right mouse context menu.\n",
-       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
-       "        return false;\n",
-       "    });\n",
-       "\n",
-       "    function set_focus () {\n",
-       "        canvas.focus();\n",
-       "        canvas_div.focus();\n",
-       "    }\n",
-       "\n",
-       "    window.setTimeout(set_focus, 100);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_toolbar = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var nav_element = $('<div/>')\n",
-       "    nav_element.attr('style', 'width: 100%');\n",
-       "    this.root.append(nav_element);\n",
-       "\n",
-       "    // Define a callback function for later on.\n",
-       "    function toolbar_event(event) {\n",
-       "        return fig.toolbar_button_onclick(event['data']);\n",
-       "    }\n",
-       "    function toolbar_mouse_event(event) {\n",
-       "        return fig.toolbar_button_onmouseover(event['data']);\n",
-       "    }\n",
-       "\n",
-       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
-       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
-       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
-       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
-       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
-       "\n",
-       "        if (!name) {\n",
-       "            // put a spacer in here.\n",
-       "            continue;\n",
-       "        }\n",
-       "        var button = $('<button/>');\n",
-       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
-       "                        'ui-button-icon-only');\n",
-       "        button.attr('role', 'button');\n",
-       "        button.attr('aria-disabled', 'false');\n",
-       "        button.click(method_name, toolbar_event);\n",
-       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
-       "\n",
-       "        var icon_img = $('<span/>');\n",
-       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
-       "        icon_img.addClass(image);\n",
-       "        icon_img.addClass('ui-corner-all');\n",
-       "\n",
-       "        var tooltip_span = $('<span/>');\n",
-       "        tooltip_span.addClass('ui-button-text');\n",
-       "        tooltip_span.html(tooltip);\n",
-       "\n",
-       "        button.append(icon_img);\n",
-       "        button.append(tooltip_span);\n",
-       "\n",
-       "        nav_element.append(button);\n",
-       "    }\n",
-       "\n",
-       "    var fmt_picker_span = $('<span/>');\n",
-       "\n",
-       "    var fmt_picker = $('<select/>');\n",
-       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
-       "    fmt_picker_span.append(fmt_picker);\n",
-       "    nav_element.append(fmt_picker_span);\n",
-       "    this.format_dropdown = fmt_picker[0];\n",
-       "\n",
-       "    for (var ind in mpl.extensions) {\n",
-       "        var fmt = mpl.extensions[ind];\n",
-       "        var option = $(\n",
-       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
-       "        fmt_picker.append(option)\n",
-       "    }\n",
-       "\n",
-       "    // Add hover states to the ui-buttons\n",
-       "    $( \".ui-button\" ).hover(\n",
-       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
-       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
-       "    );\n",
-       "\n",
-       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
-       "    nav_element.append(status_bar);\n",
-       "    this.message = status_bar[0];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
-       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
-       "    // which will in turn request a refresh of the image.\n",
-       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.send_message = function(type, properties) {\n",
-       "    properties['type'] = type;\n",
-       "    properties['figure_id'] = this.id;\n",
-       "    this.ws.send(JSON.stringify(properties));\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.send_draw_message = function() {\n",
-       "    if (!this.waiting) {\n",
-       "        this.waiting = true;\n",
-       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
-       "    var format_dropdown = fig.format_dropdown;\n",
-       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
-       "    fig.ondownload(fig, format);\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
-       "    var size = msg['size'];\n",
-       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
-       "        fig._resize_canvas(size[0], size[1]);\n",
-       "        fig.send_message(\"refresh\", {});\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
-       "    var x0 = msg['x0'] / mpl.ratio;\n",
-       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
-       "    var x1 = msg['x1'] / mpl.ratio;\n",
-       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
-       "    x0 = Math.floor(x0) + 0.5;\n",
-       "    y0 = Math.floor(y0) + 0.5;\n",
-       "    x1 = Math.floor(x1) + 0.5;\n",
-       "    y1 = Math.floor(y1) + 0.5;\n",
-       "    var min_x = Math.min(x0, x1);\n",
-       "    var min_y = Math.min(y0, y1);\n",
-       "    var width = Math.abs(x1 - x0);\n",
-       "    var height = Math.abs(y1 - y0);\n",
-       "\n",
-       "    fig.rubberband_context.clearRect(\n",
-       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
-       "\n",
-       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
-       "    // Updates the figure title.\n",
-       "    fig.header.textContent = msg['label'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
-       "    var cursor = msg['cursor'];\n",
-       "    switch(cursor)\n",
-       "    {\n",
-       "    case 0:\n",
-       "        cursor = 'pointer';\n",
-       "        break;\n",
-       "    case 1:\n",
-       "        cursor = 'default';\n",
-       "        break;\n",
-       "    case 2:\n",
-       "        cursor = 'crosshair';\n",
-       "        break;\n",
-       "    case 3:\n",
-       "        cursor = 'move';\n",
-       "        break;\n",
-       "    }\n",
-       "    fig.rubberband_canvas.style.cursor = cursor;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
-       "    fig.message.textContent = msg['message'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
-       "    // Request the server to send over a new figure.\n",
-       "    fig.send_draw_message();\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
-       "    fig.image_mode = msg['mode'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.updated_canvas_event = function() {\n",
-       "    // Called whenever the canvas gets updated.\n",
-       "    this.send_message(\"ack\", {});\n",
-       "}\n",
-       "\n",
-       "// A function to construct a web socket function for onmessage handling.\n",
-       "// Called in the figure constructor.\n",
-       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
-       "    return function socket_on_message(evt) {\n",
-       "        if (evt.data instanceof Blob) {\n",
-       "            /* FIXME: We get \"Resource interpreted as Image but\n",
-       "             * transferred with MIME type text/plain:\" errors on\n",
-       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
-       "             * to be part of the websocket stream */\n",
-       "            evt.data.type = \"image/png\";\n",
-       "\n",
-       "            /* Free the memory for the previous frames */\n",
-       "            if (fig.imageObj.src) {\n",
-       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
-       "                    fig.imageObj.src);\n",
-       "            }\n",
-       "\n",
-       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
-       "                evt.data);\n",
-       "            fig.updated_canvas_event();\n",
-       "            fig.waiting = false;\n",
-       "            return;\n",
-       "        }\n",
-       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
-       "            fig.imageObj.src = evt.data;\n",
-       "            fig.updated_canvas_event();\n",
-       "            fig.waiting = false;\n",
-       "            return;\n",
-       "        }\n",
-       "\n",
-       "        var msg = JSON.parse(evt.data);\n",
-       "        var msg_type = msg['type'];\n",
-       "\n",
-       "        // Call the  \"handle_{type}\" callback, which takes\n",
-       "        // the figure and JSON message as its only arguments.\n",
-       "        try {\n",
-       "            var callback = fig[\"handle_\" + msg_type];\n",
-       "        } catch (e) {\n",
-       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
-       "            return;\n",
-       "        }\n",
-       "\n",
-       "        if (callback) {\n",
-       "            try {\n",
-       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
-       "                callback(fig, msg);\n",
-       "            } catch (e) {\n",
-       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
-       "            }\n",
-       "        }\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
-       "mpl.findpos = function(e) {\n",
-       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
-       "    var targ;\n",
-       "    if (!e)\n",
-       "        e = window.event;\n",
-       "    if (e.target)\n",
-       "        targ = e.target;\n",
-       "    else if (e.srcElement)\n",
-       "        targ = e.srcElement;\n",
-       "    if (targ.nodeType == 3) // defeat Safari bug\n",
-       "        targ = targ.parentNode;\n",
-       "\n",
-       "    // jQuery normalizes the pageX and pageY\n",
-       "    // pageX,Y are the mouse positions relative to the document\n",
-       "    // offset() returns the position of the element relative to the document\n",
-       "    var x = e.pageX - $(targ).offset().left;\n",
-       "    var y = e.pageY - $(targ).offset().top;\n",
-       "\n",
-       "    return {\"x\": x, \"y\": y};\n",
-       "};\n",
-       "\n",
-       "/*\n",
-       " * return a copy of an object with only non-object keys\n",
-       " * we need this to avoid circular references\n",
-       " * http://stackoverflow.com/a/24161582/3208463\n",
-       " */\n",
-       "function simpleKeys (original) {\n",
-       "  return Object.keys(original).reduce(function (obj, key) {\n",
-       "    if (typeof original[key] !== 'object')\n",
-       "        obj[key] = original[key]\n",
-       "    return obj;\n",
-       "  }, {});\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
-       "    var canvas_pos = mpl.findpos(event)\n",
-       "\n",
-       "    if (name === 'button_press')\n",
-       "    {\n",
-       "        this.canvas.focus();\n",
-       "        this.canvas_div.focus();\n",
-       "    }\n",
-       "\n",
-       "    var x = canvas_pos.x * mpl.ratio;\n",
-       "    var y = canvas_pos.y * mpl.ratio;\n",
-       "\n",
-       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
-       "                             step: event.step,\n",
-       "                             guiEvent: simpleKeys(event)});\n",
-       "\n",
-       "    /* This prevents the web browser from automatically changing to\n",
-       "     * the text insertion cursor when the button is pressed.  We want\n",
-       "     * to control all of the cursor setting manually through the\n",
-       "     * 'cursor' event from matplotlib */\n",
-       "    event.preventDefault();\n",
-       "    return false;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
-       "    // Handle any extra behaviour associated with a key event\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.key_event = function(event, name) {\n",
-       "\n",
-       "    // Prevent repeat events\n",
-       "    if (name == 'key_press')\n",
-       "    {\n",
-       "        if (event.which === this._key)\n",
-       "            return;\n",
-       "        else\n",
-       "            this._key = event.which;\n",
-       "    }\n",
-       "    if (name == 'key_release')\n",
-       "        this._key = null;\n",
-       "\n",
-       "    var value = '';\n",
-       "    if (event.ctrlKey && event.which != 17)\n",
-       "        value += \"ctrl+\";\n",
-       "    if (event.altKey && event.which != 18)\n",
-       "        value += \"alt+\";\n",
-       "    if (event.shiftKey && event.which != 16)\n",
-       "        value += \"shift+\";\n",
-       "\n",
-       "    value += 'k';\n",
-       "    value += event.which.toString();\n",
-       "\n",
-       "    this._key_event_extra(event, name);\n",
-       "\n",
-       "    this.send_message(name, {key: value,\n",
-       "                             guiEvent: simpleKeys(event)});\n",
-       "    return false;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
-       "    if (name == 'download') {\n",
-       "        this.handle_save(this, null);\n",
-       "    } else {\n",
-       "        this.send_message(\"toolbar_button\", {name: name});\n",
-       "    }\n",
-       "};\n",
-       "\n",
-       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
-       "    this.message.textContent = tooltip;\n",
-       "};\n",
-       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
-       "\n",
-       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
-       "\n",
-       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
-       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
-       "    // object with the appropriate methods. Currently this is a non binary\n",
-       "    // socket, so there is still some room for performance tuning.\n",
-       "    var ws = {};\n",
-       "\n",
-       "    ws.close = function() {\n",
-       "        comm.close()\n",
-       "    };\n",
-       "    ws.send = function(m) {\n",
-       "        //console.log('sending', m);\n",
-       "        comm.send(m);\n",
-       "    };\n",
-       "    // Register the callback with on_msg.\n",
-       "    comm.on_msg(function(msg) {\n",
-       "        //console.log('receiving', msg['content']['data'], msg);\n",
-       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
-       "        ws.onmessage(msg['content']['data'])\n",
-       "    });\n",
-       "    return ws;\n",
-       "}\n",
-       "\n",
-       "mpl.mpl_figure_comm = function(comm, msg) {\n",
-       "    // This is the function which gets called when the mpl process\n",
-       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
-       "\n",
-       "    var id = msg.content.data.id;\n",
-       "    // Get hold of the div created by the display call when the Comm\n",
-       "    // socket was opened in Python.\n",
-       "    var element = $(\"#\" + id);\n",
-       "    var ws_proxy = comm_websocket_adapter(comm)\n",
-       "\n",
-       "    function ondownload(figure, format) {\n",
-       "        window.open(figure.imageObj.src);\n",
-       "    }\n",
-       "\n",
-       "    var fig = new mpl.figure(id, ws_proxy,\n",
-       "                           ondownload,\n",
-       "                           element.get(0));\n",
-       "\n",
-       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
-       "    // web socket which is closed, not our websocket->open comm proxy.\n",
-       "    ws_proxy.onopen();\n",
-       "\n",
-       "    fig.parent_element = element.get(0);\n",
-       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
-       "    if (!fig.cell_info) {\n",
-       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
-       "        return;\n",
-       "    }\n",
-       "\n",
-       "    var output_index = fig.cell_info[2]\n",
-       "    var cell = fig.cell_info[0];\n",
-       "\n",
-       "};\n",
-       "\n",
-       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
-       "    var width = fig.canvas.width/mpl.ratio\n",
-       "    fig.root.unbind('remove')\n",
-       "\n",
-       "    // Update the output cell to use the data from the current canvas.\n",
-       "    fig.push_to_output();\n",
-       "    var dataURL = fig.canvas.toDataURL();\n",
-       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
-       "    // the notebook keyboard shortcuts fail.\n",
-       "    IPython.keyboard_manager.enable()\n",
-       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
-       "    fig.close_ws(fig, msg);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
-       "    fig.send_message('closing', msg);\n",
-       "    // fig.ws.close()\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
-       "    // Turn the data on the canvas into data in the output cell.\n",
-       "    var width = this.canvas.width/mpl.ratio\n",
-       "    var dataURL = this.canvas.toDataURL();\n",
-       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.updated_canvas_event = function() {\n",
-       "    // Tell IPython that the notebook contents must change.\n",
-       "    IPython.notebook.set_dirty(true);\n",
-       "    this.send_message(\"ack\", {});\n",
-       "    var fig = this;\n",
-       "    // Wait a second, then push the new image to the DOM so\n",
-       "    // that it is saved nicely (might be nice to debounce this).\n",
-       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_toolbar = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var nav_element = $('<div/>')\n",
-       "    nav_element.attr('style', 'width: 100%');\n",
-       "    this.root.append(nav_element);\n",
-       "\n",
-       "    // Define a callback function for later on.\n",
-       "    function toolbar_event(event) {\n",
-       "        return fig.toolbar_button_onclick(event['data']);\n",
-       "    }\n",
-       "    function toolbar_mouse_event(event) {\n",
-       "        return fig.toolbar_button_onmouseover(event['data']);\n",
-       "    }\n",
-       "\n",
-       "    for(var toolbar_ind in mpl.toolbar_items){\n",
-       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
-       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
-       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
-       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
-       "\n",
-       "        if (!name) { continue; };\n",
-       "\n",
-       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
-       "        button.click(method_name, toolbar_event);\n",
-       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
-       "        nav_element.append(button);\n",
-       "    }\n",
-       "\n",
-       "    // Add the status bar.\n",
-       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
-       "    nav_element.append(status_bar);\n",
-       "    this.message = status_bar[0];\n",
-       "\n",
-       "    // Add the close button to the window.\n",
-       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
-       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
-       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
-       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
-       "    buttongrp.append(button);\n",
-       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
-       "    titlebar.prepend(buttongrp);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._root_extra_style = function(el){\n",
-       "    var fig = this\n",
-       "    el.on(\"remove\", function(){\n",
-       "\tfig.close_ws(fig, {});\n",
-       "    });\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
-       "    // this is important to make the div 'focusable\n",
-       "    el.attr('tabindex', 0)\n",
-       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
-       "    // off when our div gets focus\n",
-       "\n",
-       "    // location in version 3\n",
-       "    if (IPython.notebook.keyboard_manager) {\n",
-       "        IPython.notebook.keyboard_manager.register_events(el);\n",
-       "    }\n",
-       "    else {\n",
-       "        // location in version 2\n",
-       "        IPython.keyboard_manager.register_events(el);\n",
-       "    }\n",
-       "\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
-       "    var manager = IPython.notebook.keyboard_manager;\n",
-       "    if (!manager)\n",
-       "        manager = IPython.keyboard_manager;\n",
-       "\n",
-       "    // Check for shift+enter\n",
-       "    if (event.shiftKey && event.which == 13) {\n",
-       "        this.canvas_div.blur();\n",
-       "        event.shiftKey = false;\n",
-       "        // Send a \"J\" for go to next cell\n",
-       "        event.which = 74;\n",
-       "        event.keyCode = 74;\n",
-       "        manager.command_mode();\n",
-       "        manager.handle_keydown(event);\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
-       "    fig.ondownload(fig, null);\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.find_output_cell = function(html_output) {\n",
-       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
-       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
-       "    // IPython event is triggered only after the cells have been serialised, which for\n",
-       "    // our purposes (turning an active figure into a static one), is too late.\n",
-       "    var cells = IPython.notebook.get_cells();\n",
-       "    var ncells = cells.length;\n",
-       "    for (var i=0; i<ncells; i++) {\n",
-       "        var cell = cells[i];\n",
-       "        if (cell.cell_type === 'code'){\n",
-       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
-       "                var data = cell.output_area.outputs[j];\n",
-       "                if (data.data) {\n",
-       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
-       "                    data = data.data;\n",
-       "                }\n",
-       "                if (data['text/html'] == html_output) {\n",
-       "                    return [cell, data, j];\n",
-       "                }\n",
-       "            }\n",
-       "        }\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "// Register the function which deals with the matplotlib target/channel.\n",
-       "// The kernel may be null if the page has been refreshed.\n",
-       "if (IPython.notebook.kernel != null) {\n",
-       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
-       "}\n"
-      ],
-      "text/plain": [
-       "<IPython.core.display.Javascript object>"
-      ]
-     },
-     "metadata": {},
-     "output_type": "display_data"
-    },
-    {
-     "data": {
-      "text/html": [
-       "<img src=\"\" width=\"1000\">"
-      ],
-      "text/plain": [
-       "<IPython.core.display.HTML object>"
-      ]
-     },
-     "metadata": {},
-     "output_type": "display_data"
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "<matplotlib.legend.Legend at 0x130da4150>"
-      ]
-     },
-     "execution_count": 34,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "df_results = %sql SELECT * FROM iris_multi_model_info ORDER BY validation_loss ASC LIMIT 7;\n",
-    "df_results = df_results.DataFrame()\n",
-    "\n",
-    "df_summary = %sql SELECT * FROM iris_multi_model_summary;\n",
-    "df_summary = df_summary.DataFrame()\n",
-    "\n",
-    "#set up plots\n",
-    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
-    "fig.legend(ncol=4)\n",
-    "fig.tight_layout()\n",
-    "\n",
-    "ax_metric = axs[0]\n",
-    "ax_loss = axs[1]\n",
-    "\n",
-    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
-    "ax_metric.set_xlabel('Iteration')\n",
-    "ax_metric.set_ylabel('Metric')\n",
-    "ax_metric.set_title('Validation metric curve')\n",
-    "\n",
-    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
-    "ax_loss.set_xlabel('Iteration')\n",
-    "ax_loss.set_ylabel('Loss')\n",
-    "ax_loss.set_title('Validation loss curve')\n",
-    "\n",
-    "iters = df_summary['metrics_iters'][0]\n",
-    "\n",
-    "for mst_key in df_results['mst_key']:\n",
-    "    df_output_info = %sql SELECT validation_metrics,validation_loss FROM iris_multi_model_info WHERE mst_key = $mst_key\n",
-    "    df_output_info = df_output_info.DataFrame()\n",
-    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
-    "    validation_loss = df_output_info['validation_loss'][0]\n",
-    "    \n",
-    "    ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
-    "    ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
-    "\n",
-    "plt.legend()\n",
-    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
-   ]
-  }
- ],
- "metadata": {
-  "kernelspec": {
-   "display_name": "Python 2",
-   "language": "python",
-   "name": "python2"
-  },
-  "language_info": {
-   "codemirror_mode": {
-    "name": "ipython",
-    "version": 2
-   },
-   "file_extension": ".py",
-   "mimetype": "text/x-python",
-   "name": "python",
-   "nbconvert_exporter": "python",
-   "pygments_lexer": "ipython2",
-   "version": "2.7.16"
-  }
- },
- "nbformat": 4,
- "nbformat_minor": 1
-}
diff --git a/community-artifacts/Deep-learning/MADlib-Keras-transfer-learning-v3.ipynb b/community-artifacts/Deep-learning/MADlib-Keras-transfer-learning-v3.ipynb
deleted file mode 100644
index d025b81..0000000
--- a/community-artifacts/Deep-learning/MADlib-Keras-transfer-learning-v3.ipynb
+++ /dev/null
@@ -1,1635 +0,0 @@
-{
- "cells": [
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "# Transfer Learning Using Keras and MADlib\n",
-    "\n",
-    "This is a transfer learning example based on https://keras.io/examples/mnist_transfer_cnn/ \n",
-    "\n",
-    "To load images into tables we use the script called <em>madlib_image_loader.py</em> located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning which uses the Python Imaging Library so supports multiple formats http://www.pythonware.com/products/pil/\n",
-    "\n",
-    "## Table of contents\n",
-    "<a href=\"#import_libraries\">1. Import libraries</a>\n",
-    "\n",
-    "<a href=\"#load_and_prepare_data\">2. Load and prepare data</a>\n",
-    "\n",
-    "<a href=\"#image_preproc\">3. Call image preprocessor</a>\n",
-    "\n",
-    "<a href=\"#define_and_load_model\">4. Define and load model architecture</a>\n",
-    "\n",
-    "<a href=\"#train\">5. Train</a>\n",
-    "\n",
-    "<a href=\"#transfer_learning\">6. Transfer learning</a>"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 12,
-   "metadata": {
-    "scrolled": true
-   },
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "The sql extension is already loaded. To reload it, use:\n",
-      "  %reload_ext sql\n"
-     ]
-    }
-   ],
-   "source": [
-    "%load_ext sql"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 13,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 13,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
-    "# Greenplum Database 5.x on GCP - via tunnel\n",
-    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
-    "        \n",
-    "# PostgreSQL local\n",
-    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 14,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>version</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
-      ]
-     },
-     "execution_count": 14,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql select madlib.version();\n",
-    "#%sql select version();"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"import_libraries\"></a>\n",
-    "# 1.  Import libraries\n",
-    "From https://keras.io/examples/mnist_transfer_cnn/ import libraries and define some params"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 15,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "from __future__ import print_function\n",
-    "\n",
-    "import datetime\n",
-    "import keras\n",
-    "from keras.datasets import mnist\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense, Dropout, Activation, Flatten\n",
-    "from keras.layers import Conv2D, MaxPooling2D\n",
-    "from keras import backend as K\n",
-    "\n",
-    "now = datetime.datetime.now\n",
-    "\n",
-    "batch_size = 128\n",
-    "num_classes = 5\n",
-    "epochs = 5\n",
-    "\n",
-    "# input image dimensions\n",
-    "img_rows, img_cols = 28, 28\n",
-    "# number of convolutional filters to use\n",
-    "filters = 32\n",
-    "# size of pooling area for max pooling\n",
-    "pool_size = 2\n",
-    "# convolution kernel size\n",
-    "kernel_size = 3\n",
-    "\n",
-    "if K.image_data_format() == 'channels_first':\n",
-    "    input_shape = (1, img_rows, img_cols)\n",
-    "else:\n",
-    "    input_shape = (img_rows, img_cols, 1)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Others needed in this workbook"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 16,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "import pandas as pd\n",
-    "import numpy as np"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load_and_prepare_data\"></a>\n",
-    "# 2.  Load and prepare data\n",
-    "\n",
-    "First load MNIST data from Keras, consisting of 60,000 28x28 grayscale images of the 10 digits, along with a test set of 10,000 images."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 17,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "(4861, 28, 28)\n",
-      "(4861, 28, 28, 1)\n"
-     ]
-    }
-   ],
-   "source": [
-    "# the data, split between train and test sets\n",
-    "(x_train, y_train), (x_test, y_test) = mnist.load_data()\n",
-    "\n",
-    "# create two datasets one with digits below 5 and one with 5 and above\n",
-    "x_train_lt5 = x_train[y_train < 5]\n",
-    "y_train_lt5 = y_train[y_train < 5]\n",
-    "x_test_lt5 = x_test[y_test < 5]\n",
-    "y_test_lt5 = y_test[y_test < 5]\n",
-    "\n",
-    "x_train_gte5 = x_train[y_train >= 5]\n",
-    "y_train_gte5 = y_train[y_train >= 5] - 5\n",
-    "x_test_gte5 = x_test[y_test >= 5]\n",
-    "y_test_gte5 = y_test[y_test >= 5] - 5\n",
-    "\n",
-    "# reshape to match model architecture\n",
-    "print(x_test_gte5.shape)\n",
-    "x_train_lt5=x_train_lt5.reshape(len(x_train_lt5), *input_shape)\n",
-    "x_test_lt5 = x_test_lt5.reshape(len(x_test_lt5), *input_shape)\n",
-    "x_train_gte5=x_train_gte5.reshape(len(x_train_gte5), *input_shape)\n",
-    "x_test_gte5 = x_test_gte5.reshape(len(x_test_gte5), *input_shape)\n",
-    "print(x_test_gte5.shape)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Load datasets into tables using image loader scripts called <em>madlib_image_loader.py</em> located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 18,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "# MADlib tools directory\n",
-    "import sys\n",
-    "import os\n",
-    "madlib_site_dir = '/Users/fmcquillan/Documents/Product/MADlib/Demos/data'\n",
-    "sys.path.append(madlib_site_dir)\n",
-    "\n",
-    "# Import image loader module\n",
-    "from madlib_image_loader import ImageLoader, DbCredentials"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 20,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "# Specify database credentials, for connecting to db\n",
-    "#db_creds = DbCredentials(user='gpadmin',\n",
-    "#                         host='35.239.240.26',\n",
-    "#                         port='5432',\n",
-    "#                         password='')\n",
-    "\n",
-    "db_creds = DbCredentials(user='gpadmin',\n",
-    "                         host='localhost',\n",
-    "                         port='8000',\n",
-    "                         password='')"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 21,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "# Initialize ImageLoader (increase num_workers to run faster)\n",
-    "iloader = ImageLoader(num_workers=5, db_creds=db_creds)"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 22,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "MainProcess: Connected to madlib db.\n",
-      "Executing: CREATE TABLE train_lt5 (id SERIAL, x REAL[], y TEXT)\n",
-      "CREATE TABLE\n",
-      "Created table train_lt5 in madlib db\n",
-      "Spawning 5 workers...\n",
-      "Initializing PoolWorker-1 [pid 32140]\n",
-      "PoolWorker-1: Created temporary directory /tmp/madlib_bKeQrDW6UN\n",
-      "Initializing PoolWorker-2 [pid 32141]\n",
-      "PoolWorker-2: Created temporary directory /tmp/madlib_QnitPXnQvV\n",
-      "Initializing PoolWorker-3 [pid 32142]\n",
-      "PoolWorker-3: Created temporary directory /tmp/madlib_DvxZnkrs2R\n",
-      "Initializing PoolWorker-4 [pid 32143]\n",
-      "PoolWorker-4: Created temporary directory /tmp/madlib_qpVWvqPyAv\n",
-      "Initializing PoolWorker-5 [pid 32144]\n",
-      "PoolWorker-5: Created temporary directory /tmp/madlib_L4Q5odzoED\n",
-      "PoolWorker-4: Connected to madlib db.\n",
-      "PoolWorker-3: Connected to madlib db.\n",
-      "PoolWorker-2: Connected to madlib db.\n",
-      "PoolWorker-1: Connected to madlib db.\n",
-      "PoolWorker-5: Connected to madlib db.\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_QnitPXnQvV/train_lt50000.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_DvxZnkrs2R/train_lt50000.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_qpVWvqPyAv/train_lt50000.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_L4Q5odzoED/train_lt50000.tmp\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_bKeQrDW6UN/train_lt50000.tmp\n",
-      "PoolWorker-2: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-3: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-4: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-5: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-1: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_QnitPXnQvV/train_lt50001.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_DvxZnkrs2R/train_lt50001.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_qpVWvqPyAv/train_lt50001.tmp\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_bKeQrDW6UN/train_lt50001.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_L4Q5odzoED/train_lt50001.tmp\n",
-      "PoolWorker-2: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_QnitPXnQvV/train_lt50002.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-3: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-4: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-5: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_bKeQrDW6UN/train_lt50002.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_qpVWvqPyAv/train_lt50002.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_DvxZnkrs2R/train_lt50002.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_L4Q5odzoED/train_lt50002.tmp\n",
-      "PoolWorker-2: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_QnitPXnQvV/train_lt50003.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-3: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-5: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-4: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_bKeQrDW6UN/train_lt50003.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_qpVWvqPyAv/train_lt50003.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_DvxZnkrs2R/train_lt50003.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_L4Q5odzoED/train_lt50003.tmp\n",
-      "PoolWorker-2: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_QnitPXnQvV/train_lt50004.tmp\n",
-      "PoolWorker-5: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-1: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-3: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-4: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-2: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_L4Q5odzoED/train_lt50004.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_qpVWvqPyAv/train_lt50004.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_DvxZnkrs2R/train_lt50004.tmp\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_bKeQrDW6UN/train_lt50004.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_QnitPXnQvV/train_lt50005.tmp\n",
-      "PoolWorker-5: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_L4Q5odzoED/train_lt50005.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-4: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-3: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-2: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-2: Wrote 596 images to /tmp/madlib_QnitPXnQvV/train_lt50006.tmp\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_bKeQrDW6UN/train_lt50005.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_qpVWvqPyAv/train_lt50005.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_DvxZnkrs2R/train_lt50005.tmp\n",
-      "PoolWorker-5: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-2: Loaded 596 images into train_lt5\n",
-      "PoolWorker-1: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-3: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-4: Loaded 1000 images into train_lt5\n",
-      "PoolWorker-2: Removed temporary directory /tmp/madlib_QnitPXnQvV\n",
-      "PoolWorker-5: Removed temporary directory /tmp/madlib_L4Q5odzoED\n",
-      "PoolWorker-4: Removed temporary directory /tmp/madlib_qpVWvqPyAv\n",
-      "PoolWorker-1: Removed temporary directory /tmp/madlib_bKeQrDW6UN\n",
-      "PoolWorker-3: Removed temporary directory /tmp/madlib_DvxZnkrs2R\n",
-      "Done!  Loaded 30596 images in 82.6620368958s\n",
-      "5 workers terminated.\n",
-      "MainProcess: Connected to madlib db.\n",
-      "Executing: CREATE TABLE test_lt5 (id SERIAL, x REAL[], y TEXT)\n",
-      "CREATE TABLE\n",
-      "Created table test_lt5 in madlib db\n",
-      "Spawning 5 workers...\n",
-      "Initializing PoolWorker-6 [pid 32147]\n",
-      "PoolWorker-6: Created temporary directory /tmp/madlib_qiEVuC6H2f\n",
-      "Initializing PoolWorker-7 [pid 32148]\n",
-      "PoolWorker-7: Created temporary directory /tmp/madlib_01Yrlb4kiZ\n",
-      "Initializing PoolWorker-8 [pid 32149]\n",
-      "PoolWorker-8: Created temporary directory /tmp/madlib_SjOxjuWfuf\n",
-      "Initializing PoolWorker-9 [pid 32150]\n",
-      "PoolWorker-9: Created temporary directory /tmp/madlib_Jc01ED7LdY\n",
-      "Initializing PoolWorker-10 [pid 32151]\n",
-      "PoolWorker-10: Created temporary directory /tmp/madlib_q26dtBioOd\n",
-      "PoolWorker-7: Connected to madlib db.\n",
-      "PoolWorker-8: Connected to madlib db.\n",
-      "PoolWorker-6: Connected to madlib db.\n",
-      "PoolWorker-10: Connected to madlib db.\n",
-      "PoolWorker-9: Connected to madlib db.\n",
-      "PoolWorker-10: Wrote 1000 images to /tmp/madlib_q26dtBioOd/test_lt50000.tmp\n",
-      "PoolWorker-8: Wrote 1000 images to /tmp/madlib_SjOxjuWfuf/test_lt50000.tmp\n",
-      "PoolWorker-9: Wrote 1000 images to /tmp/madlib_Jc01ED7LdY/test_lt50000.tmp\n",
-      "PoolWorker-7: Wrote 1000 images to /tmp/madlib_01Yrlb4kiZ/test_lt50000.tmp\n",
-      "PoolWorker-6: Wrote 1000 images to /tmp/madlib_qiEVuC6H2f/test_lt50000.tmp\n",
-      "PoolWorker-10: Loaded 1000 images into test_lt5\n",
-      "PoolWorker-10: Wrote 139 images to /tmp/madlib_q26dtBioOd/test_lt50001.tmp\n",
-      "PoolWorker-8: Loaded 1000 images into test_lt5\n",
-      "PoolWorker-10: Loaded 139 images into test_lt5\n",
-      "PoolWorker-9: Loaded 1000 images into test_lt5\n",
-      "PoolWorker-6: Loaded 1000 images into test_lt5\n",
-      "PoolWorker-7: Loaded 1000 images into test_lt5\n",
-      "PoolWorker-8: Removed temporary directory /tmp/madlib_SjOxjuWfuf\n",
-      "PoolWorker-10: Removed temporary directory /tmp/madlib_q26dtBioOd\n",
-      "PoolWorker-7: Removed temporary directory /tmp/madlib_01Yrlb4kiZ\n",
-      "PoolWorker-9: Removed temporary directory /tmp/madlib_Jc01ED7LdY\n",
-      "PoolWorker-6: Removed temporary directory /tmp/madlib_qiEVuC6H2f\n",
-      "Done!  Loaded 5139 images in 16.0947310925s\n",
-      "5 workers terminated.\n",
-      "MainProcess: Connected to madlib db.\n",
-      "Executing: CREATE TABLE train_gte5 (id SERIAL, x REAL[], y TEXT)\n",
-      "CREATE TABLE\n",
-      "Created table train_gte5 in madlib db\n",
-      "Spawning 5 workers...\n",
-      "Initializing PoolWorker-11 [pid 32152]\n",
-      "PoolWorker-11: Created temporary directory /tmp/madlib_W4MvrjG6AB\n",
-      "Initializing PoolWorker-12 [pid 32153]\n",
-      "PoolWorker-12: Created temporary directory /tmp/madlib_xjoS5riF15\n",
-      "Initializing PoolWorker-13 [pid 32154]\n",
-      "PoolWorker-13: Created temporary directory /tmp/madlib_LRF1X1Nbjw\n",
-      "Initializing PoolWorker-14 [pid 32155]\n",
-      "PoolWorker-14: Created temporary directory /tmp/madlib_81JT2Bqk6q\n",
-      "Initializing PoolWorker-15 [pid 32156]\n",
-      "PoolWorker-15: Created temporary directory /tmp/madlib_cw8IBZoiUb\n",
-      "PoolWorker-11: Connected to madlib db.\n",
-      "PoolWorker-14: Connected to madlib db.\n",
-      "PoolWorker-13: Connected to madlib db.\n",
-      "PoolWorker-12: Connected to madlib db.\n",
-      "PoolWorker-15: Connected to madlib db.\n",
-      "PoolWorker-11: Wrote 1000 images to /tmp/madlib_W4MvrjG6AB/train_gte50000.tmp\n",
-      "PoolWorker-13: Wrote 1000 images to /tmp/madlib_LRF1X1Nbjw/train_gte50000.tmp\n",
-      "PoolWorker-12: Wrote 1000 images to /tmp/madlib_xjoS5riF15/train_gte50000.tmp\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "PoolWorker-15: Wrote 1000 images to /tmp/madlib_cw8IBZoiUb/train_gte50000.tmp\n",
-      "PoolWorker-14: Wrote 1000 images to /tmp/madlib_81JT2Bqk6q/train_gte50000.tmp\n",
-      "PoolWorker-11: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-11: Wrote 1000 images to /tmp/madlib_W4MvrjG6AB/train_gte50001.tmp\n",
-      "PoolWorker-13: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-14: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-12: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-15: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-13: Wrote 1000 images to /tmp/madlib_LRF1X1Nbjw/train_gte50001.tmp\n",
-      "PoolWorker-12: Wrote 1000 images to /tmp/madlib_xjoS5riF15/train_gte50001.tmp\n",
-      "PoolWorker-14: Wrote 1000 images to /tmp/madlib_81JT2Bqk6q/train_gte50001.tmp\n",
-      "PoolWorker-15: Wrote 1000 images to /tmp/madlib_cw8IBZoiUb/train_gte50001.tmp\n",
-      "PoolWorker-11: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-11: Wrote 1000 images to /tmp/madlib_W4MvrjG6AB/train_gte50002.tmp\n",
-      "PoolWorker-13: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-13: Wrote 1000 images to /tmp/madlib_LRF1X1Nbjw/train_gte50002.tmp\n",
-      "PoolWorker-14: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-15: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-12: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-11: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-15: Wrote 1000 images to /tmp/madlib_cw8IBZoiUb/train_gte50002.tmp\n",
-      "PoolWorker-14: Wrote 1000 images to /tmp/madlib_81JT2Bqk6q/train_gte50002.tmp\n",
-      "PoolWorker-12: Wrote 1000 images to /tmp/madlib_xjoS5riF15/train_gte50002.tmp\n",
-      "PoolWorker-11: Wrote 1000 images to /tmp/madlib_W4MvrjG6AB/train_gte50003.tmp\n",
-      "PoolWorker-13: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-13: Wrote 1000 images to /tmp/madlib_LRF1X1Nbjw/train_gte50003.tmp\n",
-      "PoolWorker-12: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-12: Wrote 1000 images to /tmp/madlib_xjoS5riF15/train_gte50003.tmp\n",
-      "PoolWorker-14: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-15: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-11: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-14: Wrote 1000 images to /tmp/madlib_81JT2Bqk6q/train_gte50003.tmp\n",
-      "PoolWorker-15: Wrote 1000 images to /tmp/madlib_cw8IBZoiUb/train_gte50003.tmp\n",
-      "PoolWorker-11: Wrote 1000 images to /tmp/madlib_W4MvrjG6AB/train_gte50004.tmp\n",
-      "PoolWorker-13: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-13: Wrote 1000 images to /tmp/madlib_LRF1X1Nbjw/train_gte50004.tmp\n",
-      "PoolWorker-12: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-12: Wrote 1000 images to /tmp/madlib_xjoS5riF15/train_gte50004.tmp\n",
-      "PoolWorker-14: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-15: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-11: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-11: Wrote 1000 images to /tmp/madlib_W4MvrjG6AB/train_gte50005.tmp\n",
-      "PoolWorker-15: Wrote 1000 images to /tmp/madlib_cw8IBZoiUb/train_gte50004.tmp\n",
-      "PoolWorker-14: Wrote 1000 images to /tmp/madlib_81JT2Bqk6q/train_gte50004.tmp\n",
-      "PoolWorker-13: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-13: Wrote 1000 images to /tmp/madlib_LRF1X1Nbjw/train_gte50005.tmp\n",
-      "PoolWorker-12: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-12: Wrote 1000 images to /tmp/madlib_xjoS5riF15/train_gte50005.tmp\n",
-      "PoolWorker-15: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-11: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-14: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-14: Wrote 404 images to /tmp/madlib_81JT2Bqk6q/train_gte50005.tmp\n",
-      "PoolWorker-15: Wrote 1000 images to /tmp/madlib_cw8IBZoiUb/train_gte50005.tmp\n",
-      "PoolWorker-13: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-12: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-14: Loaded 404 images into train_gte5\n",
-      "PoolWorker-15: Loaded 1000 images into train_gte5\n",
-      "PoolWorker-11: Removed temporary directory /tmp/madlib_W4MvrjG6AB\n",
-      "PoolWorker-14: Removed temporary directory /tmp/madlib_81JT2Bqk6q\n",
-      "PoolWorker-13: Removed temporary directory /tmp/madlib_LRF1X1Nbjw\n",
-      "PoolWorker-12: Removed temporary directory /tmp/madlib_xjoS5riF15\n",
-      "PoolWorker-15: Removed temporary directory /tmp/madlib_cw8IBZoiUb\n",
-      "Done!  Loaded 29404 images in 68.5753850937s\n",
-      "5 workers terminated.\n",
-      "MainProcess: Connected to madlib db.\n",
-      "Executing: CREATE TABLE test_gte5 (id SERIAL, x REAL[], y TEXT)\n",
-      "CREATE TABLE\n",
-      "Created table test_gte5 in madlib db\n",
-      "Spawning 5 workers...\n",
-      "Initializing PoolWorker-16 [pid 32159]\n",
-      "PoolWorker-16: Created temporary directory /tmp/madlib_u7tI3bg7jU\n",
-      "Initializing PoolWorker-17 [pid 32160]\n",
-      "PoolWorker-17: Created temporary directory /tmp/madlib_2bBkL2sPMZ\n",
-      "Initializing PoolWorker-18 [pid 32161]\n",
-      "PoolWorker-18: Created temporary directory /tmp/madlib_kOLOpYin2F\n",
-      "Initializing PoolWorker-19 [pid 32162]\n",
-      "PoolWorker-19: Created temporary directory /tmp/madlib_Nd5O8pUMIS\n",
-      "Initializing PoolWorker-20 [pid 32163]\n",
-      "PoolWorker-20: Created temporary directory /tmp/madlib_TmWQybahi2\n",
-      "PoolWorker-16: Connected to madlib db.\n",
-      "PoolWorker-20: Connected to madlib db.\n",
-      "PoolWorker-18: Connected to madlib db.\n",
-      "PoolWorker-19: Connected to madlib db.\n",
-      "PoolWorker-17: Connected to madlib db.\n",
-      "PoolWorker-17: Wrote 861 images to /tmp/madlib_2bBkL2sPMZ/test_gte50000.tmp\n",
-      "PoolWorker-16: Wrote 1000 images to /tmp/madlib_u7tI3bg7jU/test_gte50000.tmp\n",
-      "PoolWorker-18: Wrote 1000 images to /tmp/madlib_kOLOpYin2F/test_gte50000.tmp\n",
-      "PoolWorker-20: Wrote 1000 images to /tmp/madlib_TmWQybahi2/test_gte50000.tmp\n",
-      "PoolWorker-19: Wrote 1000 images to /tmp/madlib_Nd5O8pUMIS/test_gte50000.tmp\n",
-      "PoolWorker-17: Loaded 861 images into test_gte5\n",
-      "PoolWorker-16: Loaded 1000 images into test_gte5\n",
-      "PoolWorker-20: Loaded 1000 images into test_gte5\n",
-      "PoolWorker-19: Loaded 1000 images into test_gte5\n",
-      "PoolWorker-18: Loaded 1000 images into test_gte5\n",
-      "PoolWorker-20: Removed temporary directory /tmp/madlib_TmWQybahi2\n",
-      "PoolWorker-16: Removed temporary directory /tmp/madlib_u7tI3bg7jU\n",
-      "PoolWorker-17: Removed temporary directory /tmp/madlib_2bBkL2sPMZ\n",
-      "PoolWorker-19: Removed temporary directory /tmp/madlib_Nd5O8pUMIS\n",
-      "PoolWorker-18: Removed temporary directory /tmp/madlib_kOLOpYin2F\n",
-      "Done!  Loaded 4861 images in 11.068821907s\n",
-      "5 workers terminated.\n"
-     ]
-    }
-   ],
-   "source": [
-    "# Drop tables\n",
-    "%sql DROP TABLE IF EXISTS train_lt5, test_lt5, train_gte5, test_gte5\n",
-    "\n",
-    "# Save images to temporary directories and load into database\n",
-    "iloader.load_dataset_from_np(x_train_lt5, y_train_lt5, 'train_lt5', append=False)\n",
-    "iloader.load_dataset_from_np(x_test_lt5, y_test_lt5, 'test_lt5', append=False)\n",
-    "iloader.load_dataset_from_np(x_train_gte5, y_train_gte5, 'train_gte5', append=False)\n",
-    "iloader.load_dataset_from_np(x_test_gte5, y_test_gte5, 'test_gte5', append=False)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"image_preproc\"></a>\n",
-    "# 3. Call image preprocessor\n",
-    "\n",
-    "Transforms from one image per row to multiple images per row for batch optimization.  Also normalizes and one-hot encodes.\n",
-    "\n",
-    "Training dataset < 5"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 23,
-   "metadata": {
-    "scrolled": true
-   },
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>train_lt5</td>\n",
-       "        <td>train_lt5_packed</td>\n",
-       "        <td>y</td>\n",
-       "        <td>x</td>\n",
-       "        <td>text</td>\n",
-       "        <td>[u'0', u'1', u'2', u'3', u'4']</td>\n",
-       "        <td>1000</td>\n",
-       "        <td>255.0</td>\n",
-       "        <td>5</td>\n",
-       "        <td>all_segments</td>\n",
-       "        <td>all_segments</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'train_lt5', u'train_lt5_packed', u'y', u'x', u'text', [u'0', u'1', u'2', u'3', u'4'], 1000, 255.0, 5, 'all_segments', 'all_segments')]"
-      ]
-     },
-     "execution_count": 23,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS train_lt5_packed, train_lt5_packed_summary;\n",
-    "\n",
-    "SELECT madlib.training_preprocessor_dl('train_lt5',               -- Source table\n",
-    "                                       'train_lt5_packed',        -- Output table\n",
-    "                                       'y',                       -- Dependent variable\n",
-    "                                       'x',                       -- Independent variable\n",
-    "                                        1000,                     -- Buffer size\n",
-    "                                        255                       -- Normalizing constant\n",
-    "                                        );\n",
-    "\n",
-    "SELECT * FROM train_lt5_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Test dataset < 5"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 24,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>test_lt5</td>\n",
-       "        <td>test_lt5_packed</td>\n",
-       "        <td>y</td>\n",
-       "        <td>x</td>\n",
-       "        <td>text</td>\n",
-       "        <td>[u'0', u'1', u'2', u'3', u'4']</td>\n",
-       "        <td>2570</td>\n",
-       "        <td>255.0</td>\n",
-       "        <td>5</td>\n",
-       "        <td>all_segments</td>\n",
-       "        <td>all_segments</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'test_lt5', u'test_lt5_packed', u'y', u'x', u'text', [u'0', u'1', u'2', u'3', u'4'], 2570, 255.0, 5, 'all_segments', 'all_segments')]"
-      ]
-     },
-     "execution_count": 24,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS test_lt5_packed, test_lt5_packed_summary;\n",
-    "\n",
-    "SELECT madlib.validation_preprocessor_dl('test_lt5',                -- Source table\n",
-    "                                         'test_lt5_packed',         -- Output table\n",
-    "                                         'y',                       -- Dependent variable\n",
-    "                                         'x',                       -- Independent variable\n",
-    "                                         'train_lt5_packed'         -- Training preproc table\n",
-    "                                        );\n",
-    "\n",
-    "SELECT * FROM test_lt5_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Training dataset >= 5"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 25,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>train_gte5</td>\n",
-       "        <td>train_gte5_packed</td>\n",
-       "        <td>y</td>\n",
-       "        <td>x</td>\n",
-       "        <td>text</td>\n",
-       "        <td>[u'0', u'1', u'2', u'3', u'4']</td>\n",
-       "        <td>1000</td>\n",
-       "        <td>255.0</td>\n",
-       "        <td>5</td>\n",
-       "        <td>all_segments</td>\n",
-       "        <td>all_segments</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'train_gte5', u'train_gte5_packed', u'y', u'x', u'text', [u'0', u'1', u'2', u'3', u'4'], 1000, 255.0, 5, 'all_segments', 'all_segments')]"
-      ]
-     },
-     "execution_count": 25,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS train_gte5_packed, train_gte5_packed_summary;\n",
-    "\n",
-    "SELECT madlib.training_preprocessor_dl('train_gte5',              -- Source table\n",
-    "                                       'train_gte5_packed',       -- Output table\n",
-    "                                       'y',                       -- Dependent variable\n",
-    "                                       'x',                       -- Independent variable\n",
-    "                                        1000,                     -- Buffer size\n",
-    "                                        255                       -- Normalizing constant\n",
-    "                                        );\n",
-    "\n",
-    "SELECT * FROM train_gte5_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Test dataset >= 5"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 26,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>test_gte5</td>\n",
-       "        <td>test_gte5_packed</td>\n",
-       "        <td>y</td>\n",
-       "        <td>x</td>\n",
-       "        <td>text</td>\n",
-       "        <td>[u'0', u'1', u'2', u'3', u'4']</td>\n",
-       "        <td>2431</td>\n",
-       "        <td>255.0</td>\n",
-       "        <td>5</td>\n",
-       "        <td>all_segments</td>\n",
-       "        <td>all_segments</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'test_gte5', u'test_gte5_packed', u'y', u'x', u'text', [u'0', u'1', u'2', u'3', u'4'], 2431, 255.0, 5, 'all_segments', 'all_segments')]"
-      ]
-     },
-     "execution_count": 26,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS test_gte5_packed, test_gte5_packed_summary;\n",
-    "\n",
-    "SELECT madlib.validation_preprocessor_dl('test_gte5',             -- Source table\n",
-    "                                         'test_gte5_packed',      -- Output table\n",
-    "                                         'y',                     -- Dependent variable\n",
-    "                                         'x',                     -- Independent variable\n",
-    "                                         'train_gte5_packed'      -- Training preproc table\n",
-    "                                        );\n",
-    "\n",
-    "SELECT * FROM test_gte5_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"define_and_load_model\"></a>\n",
-    "# 4. Define and load model architecture\n",
-    "\n",
-    "Model with feature and classification layers trainable"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 27,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "conv2d_1 (Conv2D)            (None, 26, 26, 32)        320       \n",
-      "_________________________________________________________________\n",
-      "activation_1 (Activation)    (None, 26, 26, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "conv2d_2 (Conv2D)            (None, 24, 24, 32)        9248      \n",
-      "_________________________________________________________________\n",
-      "activation_2 (Activation)    (None, 24, 24, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_1 (MaxPooling2 (None, 12, 12, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "dropout_1 (Dropout)          (None, 12, 12, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "flatten_1 (Flatten)          (None, 4608)              0         \n",
-      "_________________________________________________________________\n",
-      "dense_1 (Dense)              (None, 128)               589952    \n",
-      "_________________________________________________________________\n",
-      "activation_3 (Activation)    (None, 128)               0         \n",
-      "_________________________________________________________________\n",
-      "dropout_2 (Dropout)          (None, 128)               0         \n",
-      "_________________________________________________________________\n",
-      "dense_2 (Dense)              (None, 5)                 645       \n",
-      "_________________________________________________________________\n",
-      "activation_4 (Activation)    (None, 5)                 0         \n",
-      "=================================================================\n",
-      "Total params: 600,165\n",
-      "Trainable params: 600,165\n",
-      "Non-trainable params: 0\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "# define two groups of layers: feature (convolutions) and classification (dense)\n",
-    "feature_layers = [\n",
-    "    Conv2D(filters, kernel_size,\n",
-    "           padding='valid',\n",
-    "           input_shape=input_shape),\n",
-    "    Activation('relu'),\n",
-    "    Conv2D(filters, kernel_size),\n",
-    "    Activation('relu'),\n",
-    "    MaxPooling2D(pool_size=pool_size),\n",
-    "    Dropout(0.25),\n",
-    "    Flatten(),\n",
-    "]\n",
-    "\n",
-    "classification_layers = [\n",
-    "    Dense(128),\n",
-    "    Activation('relu'),\n",
-    "    Dropout(0.5),\n",
-    "    Dense(num_classes),\n",
-    "    Activation('softmax')\n",
-    "]\n",
-    "\n",
-    "# create complete model\n",
-    "model = Sequential(feature_layers + classification_layers)\n",
-    "\n",
-    "model.summary()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Load into model architecture table using psycopg2"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 29,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_id</th>\n",
-       "        <th>name</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>feature + classification layers trainable</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, u'feature + classification layers trainable')]"
-      ]
-     },
-     "execution_count": 29,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "import psycopg2 as p2\n",
-    "#conn = p2.connect('postgresql://gpadmin@35.239.240.26:5432/madlib')\n",
-    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
-    "cur = conn.cursor()\n",
-    "\n",
-    "%sql DROP TABLE IF EXISTS model_arch_library;\n",
-    "query = \"SELECT madlib.load_keras_model('model_arch_library', %s, NULL, %s)\"\n",
-    "cur.execute(query,[model.to_json(), \"feature + classification layers trainable\"])\n",
-    "conn.commit()\n",
-    "\n",
-    "# check model loaded OK\n",
-    "%sql SELECT model_id, name FROM model_arch_library;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Model with feature layers frozen"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 30,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "conv2d_1 (Conv2D)            (None, 26, 26, 32)        320       \n",
-      "_________________________________________________________________\n",
-      "activation_1 (Activation)    (None, 26, 26, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "conv2d_2 (Conv2D)            (None, 24, 24, 32)        9248      \n",
-      "_________________________________________________________________\n",
-      "activation_2 (Activation)    (None, 24, 24, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_1 (MaxPooling2 (None, 12, 12, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "dropout_1 (Dropout)          (None, 12, 12, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "flatten_1 (Flatten)          (None, 4608)              0         \n",
-      "_________________________________________________________________\n",
-      "dense_1 (Dense)              (None, 128)               589952    \n",
-      "_________________________________________________________________\n",
-      "activation_3 (Activation)    (None, 128)               0         \n",
-      "_________________________________________________________________\n",
-      "dropout_2 (Dropout)          (None, 128)               0         \n",
-      "_________________________________________________________________\n",
-      "dense_2 (Dense)              (None, 5)                 645       \n",
-      "_________________________________________________________________\n",
-      "activation_4 (Activation)    (None, 5)                 0         \n",
-      "=================================================================\n",
-      "Total params: 600,165\n",
-      "Trainable params: 590,597\n",
-      "Non-trainable params: 9,568\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "# freeze feature layers\n",
-    "for l in feature_layers:\n",
-    "    l.trainable = False\n",
-    "\n",
-    "model.summary()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Load into transfer model architecture table using psycopg2"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 31,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "2 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_id</th>\n",
-       "        <th>name</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>feature + classification layers trainable</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>only classification layers trainable</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, u'feature + classification layers trainable'),\n",
-       " (2, u'only classification layers trainable')]"
-      ]
-     },
-     "execution_count": 31,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "cur.execute(query,[model.to_json(), \"only classification layers trainable\"])\n",
-    "conn.commit()\n",
-    "\n",
-    "# check model loaded OK\n",
-    "%sql SELECT model_id, name FROM model_arch_library ORDER BY model_id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"train\"></a>\n",
-    "# 5.  Train\n",
-    "Train the model for 5-digit classification [0..4]  "
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 32,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>madlib_keras_fit</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td></td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[('',)]"
-      ]
-     },
-     "execution_count": 32,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS mnist_model, mnist_model_summary;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_fit('train_lt5_packed',    -- source table\n",
-    "                               'mnist_model',         -- model output table\n",
-    "                               'model_arch_library',  -- model arch table\n",
-    "                                1,                    -- model arch id\n",
-    "                                $$ loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']$$,  -- compile_params\n",
-    "                                $$ batch_size=128, epochs=1 $$,  -- fit_params\n",
-    "                                5                     -- num_iterations\n",
-    "                              );"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View the model summary:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 33,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>model</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>validation_table</th>\n",
-       "        <th>metrics_compute_frequency</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>madlib_version</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "        <th>metrics_iters</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>train_lt5_packed</td>\n",
-       "        <td>mnist_model</td>\n",
-       "        <td>y</td>\n",
-       "        <td>x</td>\n",
-       "        <td>model_arch_library</td>\n",
-       "        <td>1</td>\n",
-       "        <td> loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']</td>\n",
-       "        <td> batch_size=128, epochs=1 </td>\n",
-       "        <td>5</td>\n",
-       "        <td>None</td>\n",
-       "        <td>5</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>2344.43066406</td>\n",
-       "        <td>2019-12-18 22:08:33.212149</td>\n",
-       "        <td>2019-12-18 22:10:33.354468</td>\n",
-       "        <td>[120.142242193222]</td>\n",
-       "        <td>1.17-dev</td>\n",
-       "        <td>5</td>\n",
-       "        <td>[u'0', u'1', u'2', u'3', u'4']</td>\n",
-       "        <td>text</td>\n",
-       "        <td>255.0</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.994607150555</td>\n",
-       "        <td>0.0173618607223</td>\n",
-       "        <td>[0.994607150554657]</td>\n",
-       "        <td>[0.0173618607223034]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>[5]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'train_lt5_packed', u'mnist_model', u'y', u'x', u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']\", u' batch_size=128, epochs=1 ', 5, None, 5, None, None, u'madlib_keras', 2344.43066406, datetime.datetime(2019, 12, 18, 22, 8, 33, 212149), datetime.datetime(2019, 12, 18, 22, 10, 33, 354468), [120.142242193222], u'1.17-dev', 5, [u'0', u'1', u'2', u'3', u'4'], u'text', 255.0, [u'accuracy'], 0.994607150555, 0.0173618607223, [0.994607150554657], [0.0173618607223034], None, None, None, None, [5])]"
-      ]
-     },
-     "execution_count": 33,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM mnist_model_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Evaluate using test data"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 34,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>loss</th>\n",
-       "        <th>metric</th>\n",
-       "        <th>metrics_type</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>0.0107044409961</td>\n",
-       "        <td>0.995719015598</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(0.0107044409960508, 0.995719015598297, [u'accuracy'])]"
-      ]
-     },
-     "execution_count": 34,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS mnist_validate;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_evaluate('mnist_model',      -- model\n",
-    "                                   'test_lt5_packed',   -- test table\n",
-    "                                   'mnist_validate'     -- output table\n",
-    "                                   );\n",
-    "\n",
-    "SELECT * FROM mnist_validate;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"transfer_learning\"></a>\n",
-    "# 6. Transfer learning"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Use UPDATE to load trained weights from previous run into the model library table:"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 39,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "[]"
-      ]
-     },
-     "execution_count": 39,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "UPDATE model_arch_library\n",
-    "SET model_weights = mnist_model.model_weights\n",
-    "FROM mnist_model\n",
-    "WHERE model_arch_library.model_id = 2;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Transfer: train dense layers for new classification task [5..9]"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 40,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>madlib_keras_fit</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td></td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[('',)]"
-      ]
-     },
-     "execution_count": 40,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS mnist_transfer_model, mnist_transfer_model_summary;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_fit('train_gte5_packed',   -- source table\n",
-    "                               'mnist_transfer_model',-- model output table\n",
-    "                               'model_arch_library',  -- model arch table\n",
-    "                                2,                    -- model arch id\n",
-    "                                $$ loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']$$,  -- compile_params\n",
-    "                                $$ batch_size=128, epochs=1 $$,  -- fit_params\n",
-    "                                5                     -- num_iterations\n",
-    "                              );"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "View the model summary"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 41,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>model</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>validation_table</th>\n",
-       "        <th>metrics_compute_frequency</th>\n",
-       "        <th>name</th>\n",
-       "        <th>description</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>madlib_version</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "        <th>metrics_iters</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>train_gte5_packed</td>\n",
-       "        <td>mnist_transfer_model</td>\n",
-       "        <td>y</td>\n",
-       "        <td>x</td>\n",
-       "        <td>model_arch_library</td>\n",
-       "        <td>2</td>\n",
-       "        <td> loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']</td>\n",
-       "        <td> batch_size=128, epochs=1 </td>\n",
-       "        <td>5</td>\n",
-       "        <td>None</td>\n",
-       "        <td>5</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>2344.43066406</td>\n",
-       "        <td>2019-12-18 22:12:24.949576</td>\n",
-       "        <td>2019-12-18 22:13:10.273382</td>\n",
-       "        <td>[45.3237240314484]</td>\n",
-       "        <td>1.17-dev</td>\n",
-       "        <td>5</td>\n",
-       "        <td>[u'0', u'1', u'2', u'3', u'4']</td>\n",
-       "        <td>text</td>\n",
-       "        <td>255.0</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.991259694099</td>\n",
-       "        <td>0.0289842225611</td>\n",
-       "        <td>[0.991259694099426]</td>\n",
-       "        <td>[0.028984222561121]</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>None</td>\n",
-       "        <td>[5]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'train_gte5_packed', u'mnist_transfer_model', u'y', u'x', u'model_arch_library', 2, u\" loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']\", u' batch_size=128, epochs=1 ', 5, None, 5, None, None, u'madlib_keras', 2344.43066406, datetime.datetime(2019, 12, 18, 22, 12, 24, 949576), datetime.datetime(2019, 12, 18, 22, 13, 10, 273382), [45.3237240314484], u'1.17-dev', 5, [u'0', u'1', u'2', u'3', u'4'], u'text', 255.0, [u'accuracy'], 0.991259694099, 0.0289842225611, [0.991259694099426], [0.028984222561121], None, None, None, None, [5])]"
-      ]
-     },
-     "execution_count": 41,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM mnist_transfer_model_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Evaluate using test data"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 42,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>loss</th>\n",
-       "        <th>metric</th>\n",
-       "        <th>metrics_type</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>0.0291066691279</td>\n",
-       "        <td>0.990536928177</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(0.0291066691279411, 0.99053692817688, [u'accuracy'])]"
-      ]
-     },
-     "execution_count": 42,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS mnist_transfer_validate;\n",
-    "\n",
-    "SELECT madlib.madlib_keras_evaluate('mnist_transfer_model',      -- model\n",
-    "                                   'test_gte5_packed',           -- test table\n",
-    "                                   'mnist_transfer_validate'     -- output table\n",
-    "                                   );\n",
-    "\n",
-    "SELECT * FROM mnist_transfer_validate;"
-   ]
-  }
- ],
- "metadata": {
-  "kernelspec": {
-   "display_name": "Python 2",
-   "language": "python",
-   "name": "python2"
-  },
-  "language_info": {
-   "codemirror_mode": {
-    "name": "ipython",
-    "version": 2
-   },
-   "file_extension": ".py",
-   "mimetype": "text/x-python",
-   "name": "python",
-   "nbconvert_exporter": "python",
-   "pygments_lexer": "ipython2",
-   "version": "2.7.10"
-  }
- },
- "nbformat": 4,
- "nbformat_minor": 1
-}
diff --git a/community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Define-custom-functions-v1-checkpoint.ipynb b/community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Define-custom-functions-v1-checkpoint.ipynb
new file mode 100755
index 0000000..f1d301b
--- /dev/null
+++ b/community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Define-custom-functions-v1-checkpoint.ipynb
@@ -0,0 +1,531 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Define custom functions\n",
+    "\n",
+    "This function loads custom Python functions into a table for use by deep learning algorithms.\n",
+    "\n",
+    "Custom functions can be useful if, for example, you need loss functions or metrics that are not built into the standard libraries. The functions to be loaded must be in the form of serialized Python objects created using Dill, which extends Python's pickle module to the majority of the built-in Python types.\n",
+    "\n",
+    "Custom functions are also used to return top k categorical accuracy rate in the case that you want a different k value than the default from Keras. This module includes a helper function to create the custom function automatically for a specified k.\n",
+    "\n",
+    "This method was added in MADlib 1.18.0.\n",
+    "\n",
+    "## <em>Warning</em>\n",
+    "<em>For security reasons there are controls on custom functions in MADlib. You must be a superuser to create custom functions because they could theoretically allow execution of any untrusted Python code. Regular users with MADlib USAGE permission can use existing custom functions but cannot create new ones or update existing ones.</em>\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#load_psycopg2\">1. Load object using psycopg2</a>\n",
+    "\n",
+    "<a href=\"#load_plpython\">2. Load object using a PL/Python function</a>\n",
+    "\n",
+    "<a href=\"#delete_object\">3. Delete object</a>\n",
+    "\n",
+    "<a href=\"#top_k\">4. Top k accuracy function</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_psycopg2\"></a>\n",
+    "# 1. Load object using psycopg2\n",
+    "Psycopg is a PostgreSQL database adapter for the Python programming language. Note need to use the psycopg2.Binary() method to pass as bytes."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# import database connector psycopg2 and create connection cursor\n",
+    "import psycopg2 as p2\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
+    "cur = conn.cursor()\n",
+    "\n",
+    "# import Dill and define functions\n",
+    "import dill\n",
+    "\n",
+    "# custom loss\n",
+    "def squared_error(y_true, y_pred):\n",
+    "    import keras.backend as K \n",
+    "    return K.square(y_pred - y_true)\n",
+    "pb_squared_error=dill.dumps(squared_error)\n",
+    "\n",
+    "# custom metric\n",
+    "def rmse(y_true, y_pred):\n",
+    "    import keras.backend as K \n",
+    "    return K.sqrt(K.mean(K.square(y_pred - y_true), axis=-1))\n",
+    "pb_rmse=dill.dumps(rmse)\n",
+    "\n",
+    "# call load function\n",
+    "cur.execute(\"DROP TABLE IF EXISTS madlib.custom_function_table\")\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'squared_error', 'squared error')\", [p2.Binary(pb_squared_error)])\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'rmse', 'root mean square error')\", [p2.Binary(pb_rmse)])\n",
+    "conn.commit()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>squared_error</td>\n",
+       "        <td>squared error</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>rmse</td>\n",
+       "        <td>root mean square error</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'squared_error', u'squared error'),\n",
+       " (2, u'rmse', u'root mean square error')]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_plpython\"></a>\n",
+    "# 2. Load object using a PL/Python function"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "CREATE OR REPLACE FUNCTION custom_function_squared_error()\n",
+    "RETURNS BYTEA AS\n",
+    "$$\n",
+    "import dill\n",
+    "def squared_error(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.square(y_pred - y_true)\n",
+    "pb_squared_error=dill.dumps(squared_error)\n",
+    "return pb_squared_error\n",
+    "$$ language plpythonu;\n",
+    "CREATE OR REPLACE FUNCTION custom_function_rmse()\n",
+    "RETURNS BYTEA AS\n",
+    "$$\n",
+    "import dill\n",
+    "def rmse(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.sqrt(K.mean(K.square(y_pred - y_true), axis=-1))\n",
+    "pb_rmse=dill.dumps(rmse)\n",
+    "return pb_rmse\n",
+    "$$ language plpythonu;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>load_custom_function</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS madlib.custom_function_table;\n",
+    "SELECT madlib.load_custom_function('custom_function_table', \n",
+    "                                   custom_function_squared_error(), \n",
+    "                                   'squared_error', \n",
+    "                                   'squared error');\n",
+    "\n",
+    "SELECT madlib.load_custom_function('custom_function_table', \n",
+    "                                   custom_function_rmse(), \n",
+    "                                   'rmse', \n",
+    "                                   'root mean square error');"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>squared_error</td>\n",
+       "        <td>squared error</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>rmse</td>\n",
+       "        <td>root mean square error</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'squared_error', u'squared error'),\n",
+       " (2, u'rmse', u'root mean square error')]"
+      ]
+     },
+     "execution_count": 8,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"delete_object\"></a>\n",
+    "# 3. Delete object\n",
+    "Delete by id:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>rmse</td>\n",
+       "        <td>root mean square error</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2, u'rmse', u'root mean square error')]"
+      ]
+     },
+     "execution_count": 9,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.delete_custom_function( 'custom_function_table', 1);\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Delete by name:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>delete_custom_function</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.delete_custom_function( 'custom_function_table', 'rmse');"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Since this was the last object in the table, if you delete it then the table will also be dropped."
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"top_k\"></a>\n",
+    "# 4. Top k accuracy function\n",
+    "Load top 3 accuracy function followed by a top 10 accuracy function:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>top_3_accuracy</td>\n",
+       "        <td>returns top_3_accuracy</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>top_10_accuracy</td>\n",
+       "        <td>returns top_10_accuracy</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'top_3_accuracy', u'returns top_3_accuracy'),\n",
+       " (2, u'top_10_accuracy', u'returns top_10_accuracy')]"
+      ]
+     },
+     "execution_count": 11,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS madlib.custom_function_table;\n",
+    "\n",
+    "SELECT madlib.load_top_k_accuracy_function('custom_function_table',\n",
+    "                                           3);\n",
+    "\n",
+    "SELECT madlib.load_top_k_accuracy_function('custom_function_table',\n",
+    "                                           10);\n",
+    "\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Load-model-architecture-v2.ipynb b/community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Define-model-architecture-v2-checkpoint.ipynb
similarity index 68%
copy from community-artifacts/Deep-learning/Load-model-architecture-v2.ipynb
copy to community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Define-model-architecture-v2-checkpoint.ipynb
index 8aa3716..b823f09 100644
--- a/community-artifacts/Deep-learning/Load-model-architecture-v2.ipynb
+++ b/community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Define-model-architecture-v2-checkpoint.ipynb
@@ -4,10 +4,16 @@
    "cell_type": "markdown",
    "metadata": {},
    "source": [
-    "# Load model architecture\n",
-    "This utility function loads model architectures and weights into a table for use by deep learning algorithms in Keras.  \n",
+    "# Define model architecture\n",
+    "This function loads model architectures and weights into a table for use by deep learning algorithms.\n",
     "\n",
-    "The model architecture loader was added in MADlib 1.16.\n",
+    "Model architecture is in JSON form and model weights are in the form of PostgreSQL binary data types (bytea). If the output table already exists, a new row is inserted into the table so it can act as a repository for multiple model architectures and weights.\n",
+    "\n",
+    "There is also a function to delete a model from the table.\n",
+    "\n",
+    "MADlib's deep learning methods are designed to use the TensorFlow package and its built in Keras functions. To ensure consistency, please use tensorflow.keras objects (models, layers, etc.) instead of importing Keras and using its objects.\n",
+    "\n",
+    "The model architecture loader was added in MADlib 1.16 and updated after that.\n",
     "\n",
     "## Table of contents\n",
     "\n",
@@ -25,17 +31,15 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 1,
+   "execution_count": 34,
    "metadata": {},
    "outputs": [
     {
-     "name": "stderr",
+     "name": "stdout",
      "output_type": "stream",
      "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
+      "The sql extension is already loaded. To reload it, use:\n",
+      "  %reload_ext sql\n"
      ]
     }
    ],
@@ -45,24 +49,10 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 5,
+   "execution_count": 35,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 5,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
     "# Greenplum Database 5.x on GCP - via tunnel\n",
     "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
@@ -72,7 +62,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
+   "execution_count": 36,
    "metadata": {},
    "outputs": [
     {
@@ -90,15 +80,15 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
-     "execution_count": 6,
+     "execution_count": 36,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -120,28 +110,13 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 37,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "Using TensorFlow backend.\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Couldn't import dot_parser, loading of dot files will not be possible.\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
-    "import keras\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense"
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
    ]
   },
   {
@@ -153,21 +128,22 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 8,
+   "execution_count": 38,
    "metadata": {},
    "outputs": [
     {
      "name": "stdout",
      "output_type": "stream",
      "text": [
+      "Model: \"sequential_3\"\n",
       "_________________________________________________________________\n",
       "Layer (type)                 Output Shape              Param #   \n",
       "=================================================================\n",
-      "dense_1 (Dense)              (None, 10)                50        \n",
+      "dense_9 (Dense)              (None, 10)                50        \n",
       "_________________________________________________________________\n",
-      "dense_2 (Dense)              (None, 10)                110       \n",
+      "dense_10 (Dense)             (None, 10)                110       \n",
       "_________________________________________________________________\n",
-      "dense_3 (Dense)              (None, 3)                 33        \n",
+      "dense_11 (Dense)             (None, 3)                 33        \n",
       "=================================================================\n",
       "Total params: 193\n",
       "Trainable params: 193\n",
@@ -182,21 +158,21 @@
     "model.add(Dense(10, activation='relu'))\n",
     "model.add(Dense(3, activation='softmax'))\n",
     "    \n",
-    "model.summary()"
+    "model.summary();"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 9,
+   "execution_count": 39,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_9\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_10\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_11\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_3\"}, \"backend\": \"tensorflow\"}'"
       ]
      },
-     "execution_count": 9,
+     "execution_count": 39,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -225,7 +201,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 10,
+   "execution_count": 40,
    "metadata": {},
    "outputs": [
     {
@@ -255,15 +231,15 @@
        "        <td>None</td>\n",
        "        <td>Sophie</td>\n",
        "        <td>A simple model</td>\n",
-       "        <td>__madlib_temp_19839392_1576692433_56744839__</td>\n",
+       "        <td>__madlib_temp_27065614_1614901189_16021319__</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_19839392_1576692433_56744839__')]"
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_27065614_1614901189_16021319__')]"
       ]
      },
-     "execution_count": 10,
+     "execution_count": 40,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -294,7 +270,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 11,
+   "execution_count": 41,
    "metadata": {},
    "outputs": [
     {
@@ -323,7 +299,7 @@
        "        <td>None</td>\n",
        "        <td>Maria</td>\n",
        "        <td>Also a simple model</td>\n",
-       "        <td>__madlib_temp_36064316_1576692433_8110861__</td>\n",
+       "        <td>__madlib_temp_87665369_1614901189_11144097__</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
@@ -331,16 +307,16 @@
        "        <td>None</td>\n",
        "        <td>Sophie</td>\n",
        "        <td>A simple model</td>\n",
-       "        <td>__madlib_temp_19839392_1576692433_56744839__</td>\n",
+       "        <td>__madlib_temp_27065614_1614901189_16021319__</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'Also a simple model', u'__madlib_temp_36064316_1576692433_8110861__'),\n",
-       " (1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_19839392_1576692433_56744839__')]"
+       "[(2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'Also a simple model', u'__madlib_temp_87665369_1614901189_11144097__'),\n",
+       " (1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_27065614_1614901189_16021319__')]"
       ]
      },
-     "execution_count": 11,
+     "execution_count": 41,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -376,7 +352,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 12,
+   "execution_count": 42,
    "metadata": {},
    "outputs": [
     {
@@ -384,7 +360,7 @@
      "output_type": "stream",
      "text": [
       "1 rows affected.\n",
-      "1 rows affected.\n"
+      "2 rows affected.\n"
      ]
     },
     {
@@ -392,18 +368,31 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>count</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "        <td>False</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1L,)]"
+       "[(1, u'Sophie', u'A simple model', False),\n",
+       " (2, u'Maria', u'Also a simple model', True)]"
       ]
      },
-     "execution_count": 12,
+     "execution_count": 42,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -416,7 +405,7 @@
     "WHERE model_arch_library.model_id = 2;\n",
     "\n",
     "-- Check weights loaded OK\n",
-    "SELECT COUNT(*) FROM model_arch_library WHERE model_weights IS NOT NULL;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   },
   {
@@ -430,7 +419,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 13,
+   "execution_count": 43,
    "metadata": {},
    "outputs": [
     {
@@ -438,7 +427,6 @@
      "output_type": "stream",
      "text": [
       "Done.\n",
-      "1 rows affected.\n",
       "1 rows affected.\n"
      ]
     },
@@ -447,18 +435,18 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>count</th>\n",
+       "        <th>load_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>2</td>\n",
+       "        <td></td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(2L,)]"
+       "[('',)]"
       ]
      },
-     "execution_count": 13,
+     "execution_count": 43,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -467,8 +455,8 @@
     "%%sql\n",
     "CREATE OR REPLACE FUNCTION load_weights() RETURNS VOID AS\n",
     "$$\n",
-    "from keras.layers import *\n",
-    "from keras import Sequential\n",
+    "from tensorflow.keras.layers import *\n",
+    "from tensorflow.keras import Sequential\n",
     "import numpy as np\n",
     "import plpy\n",
     "\n",
@@ -493,15 +481,12 @@
     "$$ language plpythonu;\n",
     "\n",
     "-- Call load function\n",
-    "SELECT load_weights();\n",
-    "\n",
-    "-- Check weights loaded OK\n",
-    "SELECT COUNT(*) FROM model_arch_library WHERE model_weights IS NOT NULL;"
+    "SELECT load_weights();"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 14,
+   "execution_count": 44,
    "metadata": {},
    "outputs": [
     {
@@ -518,33 +503,43 @@
        "    <tr>\n",
        "        <th>model_id</th>\n",
        "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
        "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "        <td>False</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>3</td>\n",
        "        <td>Ella</td>\n",
+       "        <td>Model x</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, u'Sophie'), (2, u'Maria'), (3, u'Ella')]"
+       "[(1, u'Sophie', u'A simple model', False),\n",
+       " (2, u'Maria', u'Also a simple model', True),\n",
+       " (3, u'Ella', u'Model x', True)]"
       ]
      },
-     "execution_count": 14,
+     "execution_count": 44,
      "metadata": {},
      "output_type": "execute_result"
     }
    ],
    "source": [
     "%%sql\n",
-    "SELECT model_id, name from model_arch_library ORDER BY model_id;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   },
   {
@@ -560,45 +555,16 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 15,
+   "execution_count": 45,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(2L,)]"
-      ]
-     },
-     "execution_count": 15,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
     "import psycopg2 as p2\n",
-    "#conn = p2.connect('postgresql://gpadmin@35.239.240.26:5432/madlib')\n",
-    "conn = p2.connect('postgresql://fmcquillan@localhost:5432/madlib')\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
     "cur = conn.cursor()\n",
     "\n",
-    "from keras.layers import *\n",
-    "from keras import Sequential\n",
+    "from tensorflow.keras.layers import *\n",
+    "from tensorflow.keras import Sequential\n",
     "import numpy as np\n",
     "\n",
     "# create model\n",
@@ -615,22 +581,19 @@
     "\n",
     "query = \"SELECT madlib.load_keras_model('model_arch_library', %s,%s,%s,%s)\"\n",
     "cur.execute(query,[model.to_json(), weights_bytea, \"Grace\", \"Model y\"])\n",
-    "conn.commit()\n",
-    "\n",
-    "# check weights loaded OK\n",
-    "%sql SELECT COUNT(*) FROM model_arch_library WHERE model_weights IS NOT NULL;"
+    "conn.commit()"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 16,
+   "execution_count": 46,
    "metadata": {},
    "outputs": [
     {
      "name": "stdout",
      "output_type": "stream",
      "text": [
-      "3 rows affected.\n"
+      "4 rows affected.\n"
      ]
     },
     {
@@ -640,33 +603,50 @@
        "    <tr>\n",
        "        <th>model_id</th>\n",
        "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
        "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "        <td>False</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>3</td>\n",
        "        <td>Ella</td>\n",
+       "        <td>Model x</td>\n",
+       "        <td>True</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>Grace</td>\n",
+       "        <td>Model y</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, u'Sophie'), (2, u'Maria'), (3, u'Ella')]"
+       "[(1, u'Sophie', u'A simple model', False),\n",
+       " (2, u'Maria', u'Also a simple model', True),\n",
+       " (3, u'Ella', u'Model x', True),\n",
+       " (4, u'Grace', u'Model y', True)]"
       ]
      },
-     "execution_count": 16,
+     "execution_count": 46,
      "metadata": {},
      "output_type": "execute_result"
     }
    ],
    "source": [
     "%%sql\n",
-    "SELECT model_id, name from model_arch_library ORDER BY model_id;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   },
   {
@@ -679,7 +659,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 17,
+   "execution_count": 47,
    "metadata": {},
    "outputs": [
     {
@@ -687,7 +667,7 @@
      "output_type": "stream",
      "text": [
       "1 rows affected.\n",
-      "2 rows affected.\n"
+      "3 rows affected.\n"
      ]
     },
     {
@@ -697,22 +677,36 @@
        "    <tr>\n",
        "        <th>model_id</th>\n",
        "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>3</td>\n",
        "        <td>Ella</td>\n",
+       "        <td>Model x</td>\n",
+       "        <td>True</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>Grace</td>\n",
+       "        <td>Model y</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(2, u'Maria'), (3, u'Ella')]"
+       "[(2, u'Maria', u'Also a simple model', True),\n",
+       " (3, u'Ella', u'Model x', True),\n",
+       " (4, u'Grace', u'Model y', True)]"
       ]
      },
-     "execution_count": 17,
+     "execution_count": 47,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -722,7 +716,7 @@
     "SELECT madlib.delete_keras_model('model_arch_library',   -- Output table\n",
     "                                  1                      -- Model id\n",
     "                                );\n",
-    "SELECT model_id, name from model_arch_library ORDER BY model_id;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   }
  ],
@@ -742,7 +736,7 @@
    "name": "python",
    "nbconvert_exporter": "python",
    "pygments_lexer": "ipython2",
-   "version": "2.7.10"
+   "version": "2.7.16"
   }
  },
  "nbformat": 4,
diff --git a/community-artifacts/Deep-learning/Preprocessor-for-images-distribution-rules-v1.ipynb b/community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Preprocessor-for-images-distribution-rules-v1-checkpoint.ipynb
similarity index 98%
copy from community-artifacts/Deep-learning/Preprocessor-for-images-distribution-rules-v1.ipynb
copy to community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Preprocessor-for-images-distribution-rules-v1-checkpoint.ipynb
index b457303..0ae2b4c 100644
--- a/community-artifacts/Deep-learning/Preprocessor-for-images-distribution-rules-v1.ipynb
+++ b/community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Preprocessor-for-images-distribution-rules-v1-checkpoint.ipynb
@@ -1198,7 +1198,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1465,7 +1465,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1657,7 +1657,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1805,7 +1805,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1938,7 +1938,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_val_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_val_packed ORDER BY __dist_key__;"
    ]
   }
  ],
diff --git a/community-artifacts/Deep-learning/Preprocessor-for-images-v2.ipynb b/community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Preprocessor-for-images-v2-checkpoint.ipynb
similarity index 60%
copy from community-artifacts/Deep-learning/Preprocessor-for-images-v2.ipynb
copy to community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Preprocessor-for-images-v2-checkpoint.ipynb
index cb76d1e..5fd5a69 100644
--- a/community-artifacts/Deep-learning/Preprocessor-for-images-v2.ipynb
+++ b/community-artifacts/Deep-learning/Model-preparation/.ipynb_checkpoints/Preprocessor-for-images-v2-checkpoint.ipynb
@@ -5,11 +5,11 @@
    "metadata": {},
    "source": [
     "# Preprocessor for image data\n",
-    "This is a mini-batch preprocessor utility for image data:\n",
+    "This preprocessor prepares training data for deep learning.\n",
     "* training_preprocessor_dl() for training datasets\n",
     "* validation_preprocessor_dl() for validation datasets\n",
     "\n",
-    "Note that there is a separate mini-batch preprocessor utility for general use cases\n",
+    "Note that there is a separate mini-batch preprocessor utility for non deep learning use cases\n",
     "http://madlib.apache.org/docs/latest/group__grp__minibatch__preprocessing.html\n",
     "\n",
     "The preprocessor for image data was added in MADlib 1.16.\n",
@@ -39,42 +39,17 @@
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
     "%load_ext sql"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 2,
+   "execution_count": 3,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
     "# Greenplum Database 5.x on GCP - via tunnel\n",
     "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
@@ -84,7 +59,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 3,
+   "execution_count": 5,
    "metadata": {},
    "outputs": [
     {
@@ -102,15 +77,15 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-85-g4bac900, cmake configuration time: Wed Mar  3 20:37:11 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-85-g4bac900, cmake configuration time: Wed Mar  3 20:37:11 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
-     "execution_count": 3,
+     "execution_count": 5,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -132,7 +107,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 4,
+   "execution_count": 6,
    "metadata": {},
    "outputs": [
     {
@@ -153,271 +128,271 @@
        "        <th>species</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[152, 186, 35], [102, 145, 138]], [[40, 249, 108], [175, 207, 70]]]</td>\n",
+       "        <td>[[[17, 201, 110], [175, 136, 179]], [[102, 57, 24], [110, 199, 64]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[205, 85, 56], [209, 11, 117]], [[86, 82, 41], [226, 192, 132]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[234, 110, 251], [147, 18, 158]], [[55, 79, 14], [140, 50, 143]]]</td>\n",
+       "        <td>[[[209, 227, 160], [86, 88, 177]], [[31, 198, 96], [167, 122, 198]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[146, 52, 167], [210, 33, 116]], [[38, 89, 69], [50, 207, 155]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[247, 125, 68], [124, 196, 20]], [[95, 100, 107], [183, 21, 138]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[117, 49, 248], [59, 18, 137]], [[110, 186, 91], [143, 46, 129]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[115, 179, 183], [14, 54, 175]], [[138, 122, 42], [79, 142, 137]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[249, 65, 200], [131, 191, 61]], [[180, 182, 119], [199, 63, 230]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[154, 117, 174], [27, 94, 33]], [[206, 21, 46], [4, 196, 185]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[238, 8, 12], [120, 187, 4]], [[184, 130, 135], [119, 191, 59]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[179, 202, 20], [219, 198, 173]], [[149, 233, 18], [38, 115, 59]]]</td>\n",
+       "        <td>[[[55, 2, 109], [28, 130, 7]], [[146, 48, 34], [240, 81, 240]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[223, 234, 239], [37, 253, 217]], [[147, 248, 108], [166, 150, 162]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[164, 46, 39], [51, 130, 218]], [[253, 150, 181], [195, 66, 75]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[85, 113, 32], [144, 145, 255]], [[122, 127, 36], [118, 88, 183]]]</td>\n",
+       "        <td>[[[128, 244, 200], [57, 113, 182]], [[64, 125, 46], [251, 129, 230]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[195, 93, 4], [102, 81, 168]], [[148, 120, 219], [21, 82, 217]]]</td>\n",
+       "        <td>[[[8, 93, 61], [67, 139, 115]], [[69, 248, 144], [199, 255, 33]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[8, 156, 237], [82, 72, 66]], [[196, 104, 210], [84, 103, 75]]]</td>\n",
+       "        <td>[[[33, 17, 73], [17, 21, 201]], [[5, 222, 1], [118, 148, 66]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[139, 194, 43], [66, 48, 239]], [[159, 52, 84], [240, 220, 232]]]</td>\n",
+       "        <td>[[[194, 61, 116], [168, 187, 124]], [[6, 247, 192], [145, 106, 5]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[183, 253, 187], [144, 168, 194]], [[44, 150, 21], [116, 216, 216]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[170, 44, 68], [245, 256, 207]], [[183, 43, 17], [231, 25, 176]]]</td>\n",
+       "        <td>[[[250, 204, 135], [27, 196, 168]], [[44, 12, 185], [65, 213, 190]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[110, 160, 246], [85, 9, 173]], [[82, 195, 61], [251, 134, 105]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[154, 222, 104], [114, 186, 18]], [[159, 254, 7], [158, 205, 190]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[222, 165, 227], [142, 191, 80]], [[46, 182, 165], [55, 99, 248]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[161, 243, 128], [10, 131, 26]], [[232, 235, 141], [162, 253, 43]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[4, 202, 109], [194, 147, 75]], [[103, 117, 217], [39, 197, 8]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[107, 63, 64], [99, 57, 224]], [[86, 185, 234], [216, 212, 210]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[96, 116, 192], [140, 21, 196]], [[85, 130, 135], [232, 206, 238]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[167, 20, 35], [174, 241, 142]], [[237, 48, 241], [38, 16, 70]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[251, 31, 179], [205, 226, 19]], [[65, 162, 159], [86, 103, 244]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[237, 220, 166], [219, 58, 77]], [[239, 93, 251], [224, 235, 232]]]</td>\n",
+       "        <td>[[[215, 52, 179], [25, 39, 117]], [[86, 155, 29], [16, 24, 35]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[219, 14, 33], [34, 237, 28]], [[64, 160, 232], [34, 180, 41]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[83, 127, 43], [71, 87, 24]], [[35, 253, 243], [93, 74, 227]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[69, 195, 165], [45, 212, 129]], [[59, 245, 162], [40, 16, 226]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[248, 5, 124], [34, 201, 206]], [[161, 244, 21], [248, 13, 57]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[0, 150, 63], [227, 80, 132]], [[166, 245, 176], [121, 118, 235]]]</td>\n",
+       "        <td>[[[215, 180, 113], [220, 61, 107]], [[168, 196, 134], [108, 108, 178]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[104, 42, 37], [143, 227, 111]], [[96, 135, 172], [12, 207, 100]]]</td>\n",
+       "        <td>[[[38, 244, 77], [228, 19, 36]], [[24, 198, 60], [63, 59, 146]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[221, 150, 126], [143, 129, 93]], [[92, 235, 60], [174, 100, 100]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[216, 163, 35], [249, 33, 139]], [[35, 70, 26], [6, 181, 122]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[97, 134, 93], [198, 94, 57]], [[92, 219, 200], [221, 56, 35]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[116, 210, 44], [216, 129, 4]], [[123, 164, 253], [156, 47, 32]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[73, 39, 151], [196, 180, 248]], [[74, 16, 190], [168, 74, 26]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[18, 246, 187], [53, 190, 47]], [[7, 234, 8], [136, 238, 131]]]</td>\n",
+       "        <td>[[[89, 162, 242], [124, 169, 202]], [[48, 26, 166], [109, 134, 78]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[235, 31, 91], [11, 1, 164]], [[49, 152, 103], [229, 144, 177]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[78, 89, 63], [104, 220, 81]], [[94, 151, 134], [28, 199, 141]]]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[206, 21, 244], [81, 65, 223]], [[112, 155, 234], [113, 63, 27]]]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[166, 1, 152], [88, 246, 230]], [[176, 54, 78], [140, 135, 172]]]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[13, 200, 234], [155, 207, 185]], [[176, 195, 10], [240, 162, 122]]]</td>\n",
+       "        <td>[[[12, 185, 157], [191, 49, 195]], [[178, 126, 167], [197, 162, 191]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[140, 235, 202], [167, 244, 113]], [[168, 140, 200], [158, 114, 121]]]</td>\n",
+       "        <td>[[[222, 254, 199], [112, 217, 32]], [[18, 203, 156], [187, 148, 204]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[192, 5, 91], [108, 41, 104]], [[52, 19, 3], [3, 204, 178]]]</td>\n",
+       "        <td>[[[58, 56, 91], [136, 105, 103]], [[65, 6, 38], [114, 201, 216]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[214, 162, 103], [80, 46, 243]], [[60, 248, 154], [47, 105, 65]]]</td>\n",
+       "        <td>[[[111, 157, 147], [46, 41, 113]], [[44, 240, 226], [5, 15, 244]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[49, 223, 45], [170, 179, 237]], [[175, 14, 89], [216, 118, 141]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[121, 144, 183], [43, 86, 141]], [[205, 189, 221], [251, 176, 25]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[74, 72, 92], [139, 3, 141]], [[106, 48, 55], [29, 30, 230]]]</td>\n",
+       "        <td>[[[171, 175, 100], [119, 132, 158]], [[175, 224, 37], [24, 71, 102]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[119, 190, 161], [4, 168, 25]], [[148, 95, 68], [234, 236, 17]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[201, 13, 87], [226, 256, 161]], [[42, 92, 44], [45, 233, 150]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[33, 179, 122], [7, 222, 241]], [[196, 127, 246], [108, 152, 138]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[220, 116, 183], [237, 27, 128]], [[250, 115, 98], [250, 19, 140]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[64, 184, 64], [214, 21, 96]], [[137, 143, 103], [103, 129, 43]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[118, 151, 126], [1, 99, 90]], [[117, 26, 71], [144, 154, 65]]]</td>\n",
+       "        <td>[[[174, 243, 194], [14, 219, 228]], [[86, 254, 177], [214, 92, 119]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[252, 59, 22], [136, 146, 86]], [[64, 209, 43], [85, 49, 181]]]</td>\n",
+       "        <td>[[[24, 120, 130], [256, 167, 172]], [[142, 93, 141], [165, 156, 239]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[81, 253, 127], [77, 53, 45]], [[64, 246, 59], [27, 219, 145]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[140, 103, 118], [4, 127, 142]], [[124, 1, 142], [35, 173, 28]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[58, 193, 28], [41, 201, 109]], [[38, 72, 186], [90, 116, 250]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[176, 21, 44], [65, 47, 184]], [[168, 165, 187], [39, 50, 55]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[192, 90, 212], [220, 218, 14]], [[157, 246, 55], [102, 99, 93]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[152, 28, 101], [195, 2, 220]], [[91, 128, 220], [189, 218, 81]]]</td>\n",
+       "        <td>[[[29, 183, 34], [23, 8, 210]], [[44, 51, 19], [91, 235, 187]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[166, 226, 50], [222, 9, 242]], [[56, 222, 206], [18, 236, 108]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[35, 210, 106], [127, 127, 134]], [[55, 162, 157], [62, 115, 201]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[134, 36, 93], [65, 36, 4]], [[35, 86, 225], [44, 73, 25]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[23, 42, 246], [130, 49, 24]], [[84, 155, 152], [212, 34, 206]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[191, 13, 233], [136, 126, 111]], [[173, 220, 176], [209, 223, 211]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[192, 255, 112], [217, 8, 134]], [[3, 254, 9], [53, 22, 93]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[174, 48, 241], [124, 166, 176]], [[136, 142, 56], [7, 253, 229]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[173, 181, 193], [127, 220, 130]], [[126, 76, 91], [135, 210, 94]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[219, 147, 155], [56, 99, 72]], [[104, 84, 196], [14, 4, 77]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[60, 83, 153], [33, 54, 70]], [[214, 247, 197], [179, 121, 67]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[212, 202, 209], [50, 78, 172]], [[196, 233, 227], [39, 49, 76]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[246, 89, 127], [66, 245, 187]], [[150, 142, 220], [203, 212, 178]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[153, 101, 60], [220, 100, 15]], [[166, 52, 65], [245, 224, 5]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[195, 44, 15], [15, 167, 4]], [[104, 38, 71], [94, 225, 220]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[189, 168, 192], [112, 107, 89]], [[213, 166, 54], [56, 181, 220]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[246, 208, 77], [251, 174, 16]], [[39, 189, 31], [206, 193, 135]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[8, 229, 214], [228, 209, 147]], [[140, 146, 3], [247, 235, 215]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[33, 16, 82], [252, 124, 72]], [[205, 201, 68], [123, 217, 107]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[248, 57, 249], [127, 46, 1]], [[100, 3, 229], [54, 150, 113]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[([[[152, 186, 35], [102, 145, 138]], [[40, 249, 108], [175, 207, 70]]], u'cat'),\n",
-       " ([[[234, 110, 251], [147, 18, 158]], [[55, 79, 14], [140, 50, 143]]], u'cat'),\n",
-       " ([[[179, 202, 20], [219, 198, 173]], [[149, 233, 18], [38, 115, 59]]], u'cat'),\n",
-       " ([[[223, 234, 239], [37, 253, 217]], [[147, 248, 108], [166, 150, 162]]], u'bird'),\n",
-       " ([[[164, 46, 39], [51, 130, 218]], [[253, 150, 181], [195, 66, 75]]], u'bird'),\n",
-       " ([[[85, 113, 32], [144, 145, 255]], [[122, 127, 36], [118, 88, 183]]], u'dog'),\n",
-       " ([[[195, 93, 4], [102, 81, 168]], [[148, 120, 219], [21, 82, 217]]], u'bird'),\n",
-       " ([[[8, 156, 237], [82, 72, 66]], [[196, 104, 210], [84, 103, 75]]], u'bird'),\n",
-       " ([[[139, 194, 43], [66, 48, 239]], [[159, 52, 84], [240, 220, 232]]], u'dog'),\n",
-       " ([[[183, 253, 187], [144, 168, 194]], [[44, 150, 21], [116, 216, 216]]], u'bird'),\n",
-       " ([[[170, 44, 68], [245, 256, 207]], [[183, 43, 17], [231, 25, 176]]], u'cat'),\n",
-       " ([[[110, 160, 246], [85, 9, 173]], [[82, 195, 61], [251, 134, 105]]], u'dog'),\n",
-       " ([[[154, 222, 104], [114, 186, 18]], [[159, 254, 7], [158, 205, 190]]], u'bird'),\n",
-       " ([[[222, 165, 227], [142, 191, 80]], [[46, 182, 165], [55, 99, 248]]], u'bird'),\n",
-       " ([[[161, 243, 128], [10, 131, 26]], [[232, 235, 141], [162, 253, 43]]], u'dog'),\n",
-       " ([[[4, 202, 109], [194, 147, 75]], [[103, 117, 217], [39, 197, 8]]], u'bird'),\n",
-       " ([[[107, 63, 64], [99, 57, 224]], [[86, 185, 234], [216, 212, 210]]], u'bird'),\n",
-       " ([[[96, 116, 192], [140, 21, 196]], [[85, 130, 135], [232, 206, 238]]], u'dog'),\n",
-       " ([[[167, 20, 35], [174, 241, 142]], [[237, 48, 241], [38, 16, 70]]], u'bird'),\n",
-       " ([[[251, 31, 179], [205, 226, 19]], [[65, 162, 159], [86, 103, 244]]], u'bird'),\n",
-       " ([[[237, 220, 166], [219, 58, 77]], [[239, 93, 251], [224, 235, 232]]], u'cat'),\n",
-       " ([[[219, 14, 33], [34, 237, 28]], [[64, 160, 232], [34, 180, 41]]], u'bird'),\n",
-       " ([[[83, 127, 43], [71, 87, 24]], [[35, 253, 243], [93, 74, 227]]], u'bird'),\n",
-       " ([[[69, 195, 165], [45, 212, 129]], [[59, 245, 162], [40, 16, 226]]], u'bird'),\n",
-       " ([[[248, 5, 124], [34, 201, 206]], [[161, 244, 21], [248, 13, 57]]], u'bird'),\n",
-       " ([[[0, 150, 63], [227, 80, 132]], [[166, 245, 176], [121, 118, 235]]], u'dog'),\n",
-       " ([[[104, 42, 37], [143, 227, 111]], [[96, 135, 172], [12, 207, 100]]], u'bird'),\n",
-       " ([[[221, 150, 126], [143, 129, 93]], [[92, 235, 60], [174, 100, 100]]], u'bird'),\n",
-       " ([[[216, 163, 35], [249, 33, 139]], [[35, 70, 26], [6, 181, 122]]], u'dog'),\n",
-       " ([[[97, 134, 93], [198, 94, 57]], [[92, 219, 200], [221, 56, 35]]], u'bird'),\n",
-       " ([[[116, 210, 44], [216, 129, 4]], [[123, 164, 253], [156, 47, 32]]], u'bird'),\n",
-       " ([[[73, 39, 151], [196, 180, 248]], [[74, 16, 190], [168, 74, 26]]], u'dog'),\n",
-       " ([[[18, 246, 187], [53, 190, 47]], [[7, 234, 8], [136, 238, 131]]], u'cat'),\n",
-       " ([[[235, 31, 91], [11, 1, 164]], [[49, 152, 103], [229, 144, 177]]], u'bird'),\n",
-       " ([[[78, 89, 63], [104, 220, 81]], [[94, 151, 134], [28, 199, 141]]], u'cat'),\n",
-       " ([[[206, 21, 244], [81, 65, 223]], [[112, 155, 234], [113, 63, 27]]], u'cat'),\n",
-       " ([[[166, 1, 152], [88, 246, 230]], [[176, 54, 78], [140, 135, 172]]], u'cat'),\n",
-       " ([[[13, 200, 234], [155, 207, 185]], [[176, 195, 10], [240, 162, 122]]], u'dog'),\n",
-       " ([[[140, 235, 202], [167, 244, 113]], [[168, 140, 200], [158, 114, 121]]], u'bird'),\n",
-       " ([[[192, 5, 91], [108, 41, 104]], [[52, 19, 3], [3, 204, 178]]], u'bird'),\n",
-       " ([[[214, 162, 103], [80, 46, 243]], [[60, 248, 154], [47, 105, 65]]], u'bird'),\n",
-       " ([[[49, 223, 45], [170, 179, 237]], [[175, 14, 89], [216, 118, 141]]], u'bird'),\n",
-       " ([[[121, 144, 183], [43, 86, 141]], [[205, 189, 221], [251, 176, 25]]], u'bird'),\n",
-       " ([[[74, 72, 92], [139, 3, 141]], [[106, 48, 55], [29, 30, 230]]], u'cat'),\n",
-       " ([[[119, 190, 161], [4, 168, 25]], [[148, 95, 68], [234, 236, 17]]], u'dog'),\n",
-       " ([[[201, 13, 87], [226, 256, 161]], [[42, 92, 44], [45, 233, 150]]], u'dog'),\n",
-       " ([[[33, 179, 122], [7, 222, 241]], [[196, 127, 246], [108, 152, 138]]], u'bird'),\n",
-       " ([[[220, 116, 183], [237, 27, 128]], [[250, 115, 98], [250, 19, 140]]], u'dog'),\n",
-       " ([[[64, 184, 64], [214, 21, 96]], [[137, 143, 103], [103, 129, 43]]], u'bird'),\n",
-       " ([[[118, 151, 126], [1, 99, 90]], [[117, 26, 71], [144, 154, 65]]], u'cat'),\n",
-       " ([[[252, 59, 22], [136, 146, 86]], [[64, 209, 43], [85, 49, 181]]], u'bird'),\n",
-       " ([[[152, 28, 101], [195, 2, 220]], [[91, 128, 220], [189, 218, 81]]], u'bird')]"
+       "[([[[17, 201, 110], [175, 136, 179]], [[102, 57, 24], [110, 199, 64]]], u'bird'),\n",
+       " ([[[205, 85, 56], [209, 11, 117]], [[86, 82, 41], [226, 192, 132]]], u'cat'),\n",
+       " ([[[209, 227, 160], [86, 88, 177]], [[31, 198, 96], [167, 122, 198]]], u'bird'),\n",
+       " ([[[146, 52, 167], [210, 33, 116]], [[38, 89, 69], [50, 207, 155]]], u'dog'),\n",
+       " ([[[247, 125, 68], [124, 196, 20]], [[95, 100, 107], [183, 21, 138]]], u'dog'),\n",
+       " ([[[117, 49, 248], [59, 18, 137]], [[110, 186, 91], [143, 46, 129]]], u'bird'),\n",
+       " ([[[115, 179, 183], [14, 54, 175]], [[138, 122, 42], [79, 142, 137]]], u'bird'),\n",
+       " ([[[249, 65, 200], [131, 191, 61]], [[180, 182, 119], [199, 63, 230]]], u'dog'),\n",
+       " ([[[154, 117, 174], [27, 94, 33]], [[206, 21, 46], [4, 196, 185]]], u'dog'),\n",
+       " ([[[238, 8, 12], [120, 187, 4]], [[184, 130, 135], [119, 191, 59]]], u'cat'),\n",
+       " ([[[55, 2, 109], [28, 130, 7]], [[146, 48, 34], [240, 81, 240]]], u'cat'),\n",
+       " ([[[128, 244, 200], [57, 113, 182]], [[64, 125, 46], [251, 129, 230]]], u'dog'),\n",
+       " ([[[8, 93, 61], [67, 139, 115]], [[69, 248, 144], [199, 255, 33]]], u'bird'),\n",
+       " ([[[33, 17, 73], [17, 21, 201]], [[5, 222, 1], [118, 148, 66]]], u'bird'),\n",
+       " ([[[194, 61, 116], [168, 187, 124]], [[6, 247, 192], [145, 106, 5]]], u'dog'),\n",
+       " ([[[250, 204, 135], [27, 196, 168]], [[44, 12, 185], [65, 213, 190]]], u'cat'),\n",
+       " ([[[215, 52, 179], [25, 39, 117]], [[86, 155, 29], [16, 24, 35]]], u'cat'),\n",
+       " ([[[215, 180, 113], [220, 61, 107]], [[168, 196, 134], [108, 108, 178]]], u'dog'),\n",
+       " ([[[38, 244, 77], [228, 19, 36]], [[24, 198, 60], [63, 59, 146]]], u'bird'),\n",
+       " ([[[89, 162, 242], [124, 169, 202]], [[48, 26, 166], [109, 134, 78]]], u'cat'),\n",
+       " ([[[12, 185, 157], [191, 49, 195]], [[178, 126, 167], [197, 162, 191]]], u'dog'),\n",
+       " ([[[222, 254, 199], [112, 217, 32]], [[18, 203, 156], [187, 148, 204]]], u'bird'),\n",
+       " ([[[58, 56, 91], [136, 105, 103]], [[65, 6, 38], [114, 201, 216]]], u'bird'),\n",
+       " ([[[111, 157, 147], [46, 41, 113]], [[44, 240, 226], [5, 15, 244]]], u'bird'),\n",
+       " ([[[171, 175, 100], [119, 132, 158]], [[175, 224, 37], [24, 71, 102]]], u'cat'),\n",
+       " ([[[174, 243, 194], [14, 219, 228]], [[86, 254, 177], [214, 92, 119]]], u'cat'),\n",
+       " ([[[24, 120, 130], [256, 167, 172]], [[142, 93, 141], [165, 156, 239]]], u'cat'),\n",
+       " ([[[81, 253, 127], [77, 53, 45]], [[64, 246, 59], [27, 219, 145]]], u'cat'),\n",
+       " ([[[140, 103, 118], [4, 127, 142]], [[124, 1, 142], [35, 173, 28]]], u'dog'),\n",
+       " ([[[58, 193, 28], [41, 201, 109]], [[38, 72, 186], [90, 116, 250]]], u'cat'),\n",
+       " ([[[176, 21, 44], [65, 47, 184]], [[168, 165, 187], [39, 50, 55]]], u'cat'),\n",
+       " ([[[192, 90, 212], [220, 218, 14]], [[157, 246, 55], [102, 99, 93]]], u'bird'),\n",
+       " ([[[29, 183, 34], [23, 8, 210]], [[44, 51, 19], [91, 235, 187]]], u'bird'),\n",
+       " ([[[166, 226, 50], [222, 9, 242]], [[56, 222, 206], [18, 236, 108]]], u'cat'),\n",
+       " ([[[35, 210, 106], [127, 127, 134]], [[55, 162, 157], [62, 115, 201]]], u'dog'),\n",
+       " ([[[134, 36, 93], [65, 36, 4]], [[35, 86, 225], [44, 73, 25]]], u'cat'),\n",
+       " ([[[23, 42, 246], [130, 49, 24]], [[84, 155, 152], [212, 34, 206]]], u'dog'),\n",
+       " ([[[191, 13, 233], [136, 126, 111]], [[173, 220, 176], [209, 223, 211]]], u'cat'),\n",
+       " ([[[192, 255, 112], [217, 8, 134]], [[3, 254, 9], [53, 22, 93]]], u'bird'),\n",
+       " ([[[174, 48, 241], [124, 166, 176]], [[136, 142, 56], [7, 253, 229]]], u'bird'),\n",
+       " ([[[173, 181, 193], [127, 220, 130]], [[126, 76, 91], [135, 210, 94]]], u'dog'),\n",
+       " ([[[219, 147, 155], [56, 99, 72]], [[104, 84, 196], [14, 4, 77]]], u'dog'),\n",
+       " ([[[60, 83, 153], [33, 54, 70]], [[214, 247, 197], [179, 121, 67]]], u'bird'),\n",
+       " ([[[212, 202, 209], [50, 78, 172]], [[196, 233, 227], [39, 49, 76]]], u'dog'),\n",
+       " ([[[246, 89, 127], [66, 245, 187]], [[150, 142, 220], [203, 212, 178]]], u'bird'),\n",
+       " ([[[153, 101, 60], [220, 100, 15]], [[166, 52, 65], [245, 224, 5]]], u'bird'),\n",
+       " ([[[195, 44, 15], [15, 167, 4]], [[104, 38, 71], [94, 225, 220]]], u'bird'),\n",
+       " ([[[189, 168, 192], [112, 107, 89]], [[213, 166, 54], [56, 181, 220]]], u'dog'),\n",
+       " ([[[246, 208, 77], [251, 174, 16]], [[39, 189, 31], [206, 193, 135]]], u'bird'),\n",
+       " ([[[8, 229, 214], [228, 209, 147]], [[140, 146, 3], [247, 235, 215]]], u'dog'),\n",
+       " ([[[33, 16, 82], [252, 124, 72]], [[205, 201, 68], [123, 217, 107]]], u'cat'),\n",
+       " ([[[248, 57, 249], [127, 46, 1]], [[100, 3, 229], [54, 150, 113]]], u'bird')]"
       ]
      },
-     "execution_count": 4,
+     "execution_count": 6,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -463,7 +438,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 5,
+   "execution_count": 8,
    "metadata": {},
    "outputs": [
     {
@@ -480,8 +455,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -500,7 +475,7 @@
        "[([26, 2, 2, 3], [26, 3], 0), ([26, 2, 2, 3], [26, 3], 1)]"
       ]
      },
-     "execution_count": 5,
+     "execution_count": 8,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -517,7 +492,7 @@
     "                                        255                   -- Normalizing constant\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -531,7 +506,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
+   "execution_count": 9,
    "metadata": {},
    "outputs": [
     {
@@ -551,7 +526,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -561,23 +536,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 26, 255.0, 3, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 26, 255.0, [3], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 6,
+     "execution_count": 9,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -599,7 +574,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 10,
    "metadata": {},
    "outputs": [
     {
@@ -616,8 +591,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -636,7 +611,7 @@
        "[([26, 2, 2, 3], [26, 3], 0), ([26, 2, 2, 3], [26, 3], 1)]"
       ]
      },
-     "execution_count": 7,
+     "execution_count": 10,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -644,6 +619,7 @@
    "source": [
     "%%sql\n",
     "DROP TABLE IF EXISTS val_image_data_packed, val_image_data_packed_summary;\n",
+    "\n",
     "SELECT madlib.validation_preprocessor_dl(\n",
     "      'image_data',             -- Source table\n",
     "      'val_image_data_packed',  -- Output table\n",
@@ -652,7 +628,8 @@
     "      'image_data_packed',      -- From training preprocessor step\n",
     "      NULL                      -- Buffer size\n",
     "      ); \n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
+    "\n",
+    "SELECT rgb_shape, species_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -664,7 +641,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 8,
+   "execution_count": 11,
    "metadata": {},
    "outputs": [
     {
@@ -684,7 +661,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -694,23 +671,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>val_image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'val_image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 26, 255.0, 3, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'val_image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 26, 255.0, [3], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 8,
+     "execution_count": 11,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -731,7 +708,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 9,
+   "execution_count": 12,
    "metadata": {},
    "outputs": [
     {
@@ -752,271 +729,271 @@
        "        <th>species</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[19, 126, 250, 219, 119, 255, 86, 152, 200, 36, 57, 188]</td>\n",
+       "        <td>[168, 228, 110, 3, 51, 104, 192, 23, 120, 249, 96, 99]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[20, 145, 109, 135, 149, 100, 39, 66, 124, 102, 77, 140]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[125, 32, 244, 23, 201, 156, 251, 55, 159, 47, 160, 95]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[49, 201, 114, 38, 201, 8, 101, 172, 88, 233, 82, 78]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[203, 196, 132, 57, 220, 151, 183, 214, 113, 46, 213, 200]</td>\n",
+       "        <td>[24, 88, 166, 123, 193, 186, 12, 46, 65, 161, 145, 104]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[157, 236, 255, 90, 38, 48, 35, 152, 86, 236, 160, 187]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[248, 164, 234, 70, 61, 181, 10, 193, 238, 229, 88, 165]</td>\n",
+       "        <td>[14, 206, 47, 154, 85, 172, 186, 73, 196, 131, 229, 191]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[201, 210, 145, 145, 152, 46, 125, 151, 135, 163, 199, 170]</td>\n",
+       "        <td>[131, 238, 90, 227, 51, 114, 59, 217, 237, 252, 147, 248]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[29, 150, 219, 216, 46, 211, 124, 24, 25, 186, 205, 35]</td>\n",
+       "        <td>[211, 153, 187, 59, 123, 200, 10, 171, 98, 95, 87, 28]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[187, 8, 211, 95, 196, 156, 50, 84, 45, 202, 130, 170]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[9, 77, 40, 179, 136, 69, 74, 98, 29, 120, 53, 153]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[78, 83, 93, 113, 206, 23, 121, 160, 119, 61, 60, 168]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[105, 114, 19, 19, 211, 28, 96, 251, 208, 232, 64, 25]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[93, 145, 128, 246, 33, 206, 73, 126, 63, 22, 150, 184]</td>\n",
+       "        <td>[26, 159, 140, 217, 89, 15, 199, 179, 242, 250, 37, 45]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[12, 245, 243, 181, 134, 92, 39, 153, 112, 250, 181, 208]</td>\n",
+       "        <td>[18, 41, 102, 10, 82, 57, 163, 13, 116, 30, 213, 126]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[133, 184, 53, 158, 3, 145, 47, 130, 135, 81, 80, 208]</td>\n",
+       "        <td>[56, 221, 31, 84, 132, 58, 243, 16, 19, 76, 31, 218]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[143, 230, 101, 71, 156, 113, 61, 143, 37, 195, 235, 76]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[91, 70, 17, 43, 59, 150, 227, 111, 53, 229, 0, 100]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[136, 181, 184, 87, 132, 71, 61, 232, 143, 218, 89, 203]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[126, 142, 84, 203, 234, 175, 17, 251, 217, 75, 145, 188]</td>\n",
+       "        <td>[17, 212, 36, 62, 167, 54, 103, 13, 64, 185, 70, 227]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[198, 162, 187, 42, 9, 67, 223, 193, 154, 99, 9, 215]</td>\n",
-       "        <td>cat</td>\n",
+       "        <td>[186, 1, 155, 56, 201, 211, 21, 233, 38, 153, 34, 25]</td>\n",
+       "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[151, 177, 164, 98, 25, 35, 240, 109, 237, 218, 28, 254]</td>\n",
+       "        <td>[53, 101, 200, 15, 101, 217, 227, 137, 23, 138, 191, 126]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[255, 54, 220, 226, 252, 150, 227, 151, 207, 172, 105, 227]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[144, 124, 183, 169, 37, 237, 14, 237, 252, 115, 198, 222]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[246, 73, 102, 178, 4, 45, 84, 191, 87, 93, 2, 54]</td>\n",
+       "        <td>[222, 104, 188, 92, 254, 187, 146, 219, 157, 142, 113, 128]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[156, 153, 39, 115, 228, 190, 35, 136, 32, 61, 171, 16]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[152, 234, 198, 149, 191, 188, 222, 37, 110, 226, 82, 194]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[169, 31, 163, 222, 61, 62, 119, 100, 177, 91, 34, 213]</td>\n",
+       "        <td>[64, 44, 142, 35, 193, 30, 159, 120, 199, 196, 101, 213]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[67, 17, 141, 83, 188, 37, 61, 130, 187, 252, 62, 153]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[172, 123, 115, 110, 28, 28, 140, 191, 250, 202, 253, 113]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[225, 113, 99, 228, 109, 158, 250, 245, 47, 79, 52, 1]</td>\n",
+       "        <td>[96, 72, 120, 63, 69, 86, 167, 0, 177, 165, 187, 67]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[137, 50, 48, 110, 202, 76, 211, 142, 78, 174, 232, 206]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[166, 168, 219, 125, 201, 188, 238, 44, 160, 92, 202, 153]</td>\n",
+       "        <td>[88, 210, 241, 216, 246, 48, 4, 132, 83, 197, 162, 242]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[249, 233, 133, 249, 100, 14, 43, 147, 124, 246, 223, 78]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[45, 253, 108, 251, 135, 18, 163, 98, 143, 108, 30, 126]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[190, 217, 97, 87, 41, 90, 64, 174, 84, 164, 188, 127]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[56, 117, 22, 134, 249, 67, 130, 101, 62, 9, 119, 225]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[6, 78, 138, 132, 230, 72, 93, 71, 159, 134, 161, 223]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[245, 131, 240, 116, 186, 40, 233, 209, 174, 226, 20, 48]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[82, 57, 189, 52, 165, 195, 129, 46, 71, 103, 118, 163]</td>\n",
+       "        <td>[105, 182, 162, 62, 104, 2, 134, 223, 65, 203, 53, 231]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[21, 41, 79, 244, 93, 68, 120, 78, 184, 50, 117, 161]</td>\n",
+       "        <td>[230, 140, 134, 42, 12, 223, 251, 252, 183, 241, 44, 188]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[127, 129, 24, 113, 190, 129, 40, 96, 191, 143, 98, 69]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[162, 16, 163, 137, 219, 137, 21, 97, 179, 33, 64, 174]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[35, 131, 23, 83, 201, 105, 140, 134, 157, 48, 73, 30]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[144, 133, 213, 51, 51, 234, 93, 130, 222, 186, 198, 86]</td>\n",
+       "        <td>[247, 159, 74, 179, 21, 201, 51, 45, 58, 241, 175, 98]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[126, 136, 125, 31, 139, 160, 161, 162, 242, 106, 11, 126]</td>\n",
+       "        <td>[110, 241, 179, 179, 96, 85, 195, 3, 222, 158, 140, 244]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[168, 174, 58, 198, 13, 202, 75, 226, 254, 126, 204, 90]</td>\n",
+       "        <td>[63, 21, 63, 237, 50, 54, 140, 124, 233, 162, 69, 28]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[170, 20, 197, 1, 28, 67, 137, 153, 97, 20, 57, 3]</td>\n",
-       "        <td>bird</td>\n",
+       "        <td>[94, 111, 234, 231, 203, 73, 118, 97, 57, 254, 209, 131]</td>\n",
+       "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[43, 109, 193, 169, 94, 105, 88, 152, 46, 101, 98, 121]</td>\n",
+       "        <td>[246, 73, 151, 78, 201, 43, 59, 1, 215, 155, 138, 63]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[46, 186, 18, 158, 254, 111, 13, 232, 86, 216, 49, 204]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[95, 247, 19, 186, 247, 189, 206, 188, 190, 234, 254, 70]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[96, 90, 188, 98, 16, 231, 207, 209, 145, 45, 58, 232]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[104, 77, 39, 226, 148, 134, 217, 166, 64, 207, 99, 14]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[33, 248, 137, 103, 124, 233, 194, 56, 75, 210, 32, 27]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[176, 72, 221, 152, 12, 70, 229, 51, 39, 121, 185, 0]</td>\n",
+       "        <td>[106, 202, 9, 238, 104, 256, 55, 255, 78, 0, 42, 137]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[249, 207, 131, 7, 90, 164, 255, 228, 11, 123, 205, 205]</td>\n",
+       "        <td>[1, 35, 139, 64, 121, 185, 250, 139, 87, 248, 250, 100]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[25, 160, 211, 51, 67, 131, 123, 33, 28, 135, 102, 1]</td>\n",
+       "        <td>[81, 59, 17, 29, 116, 124, 231, 125, 105, 79, 124, 160]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[202, 160, 119, 83, 161, 120, 118, 44, 183, 239, 230, 177]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[61, 169, 117, 160, 136, 197, 220, 153, 226, 79, 21, 201]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[142, 122, 115, 142, 154, 108, 93, 29, 115, 184, 193, 114]</td>\n",
+       "        <td>[126, 23, 73, 30, 100, 19, 191, 219, 102, 96, 83, 220]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[204, 237, 105, 153, 161, 129, 57, 116, 181, 124, 247, 47]</td>\n",
+       "        <td>[10, 203, 113, 187, 70, 174, 99, 186, 78, 235, 128, 42]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[98, 122, 154, 42, 70, 24, 66, 143, 54, 166, 161, 245]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[7, 84, 211, 227, 224, 221, 174, 82, 152, 244, 255, 251]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[78, 230, 46, 120, 106, 144, 241, 4, 186, 55, 28, 252]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[82, 162, 103, 71, 35, 110, 156, 246, 81, 124, 211, 255]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[106, 243, 205, 101, 161, 26, 75, 207, 146, 181, 94, 132]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[24, 187, 213, 20, 129, 39, 182, 232, 110, 217, 86, 10]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[168, 134, 161, 167, 83, 12, 154, 32, 113, 58, 58, 188]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[205, 113, 103, 80, 42, 128, 11, 255, 148, 140, 39, 74]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[149, 34, 203, 159, 241, 114, 37, 146, 25, 120, 158, 179]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 237, 210, 202, 246, 159, 59, 94, 239, 101, 221, 250]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[113, 134, 139, 187, 250, 32, 222, 197, 192, 206, 55, 229]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[81, 93, 255, 4, 244, 13, 241, 198, 215, 231, 101, 18]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[84, 120, 34, 78, 220, 147, 212, 103, 79, 206, 136, 44]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[71, 251, 203, 44, 91, 28, 136, 90, 31, 124, 103, 16]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[62, 248, 167, 81, 60, 251, 200, 95, 72, 164, 242, 28]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[65, 235, 147, 109, 126, 219, 103, 73, 6, 195, 101, 143]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[([19, 126, 250, 219, 119, 255, 86, 152, 200, 36, 57, 188], u'cat'),\n",
-       " ([49, 201, 114, 38, 201, 8, 101, 172, 88, 233, 82, 78], u'dog'),\n",
-       " ([203, 196, 132, 57, 220, 151, 183, 214, 113, 46, 213, 200], u'bird'),\n",
-       " ([157, 236, 255, 90, 38, 48, 35, 152, 86, 236, 160, 187], u'dog'),\n",
-       " ([248, 164, 234, 70, 61, 181, 10, 193, 238, 229, 88, 165], u'bird'),\n",
-       " ([201, 210, 145, 145, 152, 46, 125, 151, 135, 163, 199, 170], u'cat'),\n",
-       " ([29, 150, 219, 216, 46, 211, 124, 24, 25, 186, 205, 35], u'dog'),\n",
-       " ([187, 8, 211, 95, 196, 156, 50, 84, 45, 202, 130, 170], u'dog'),\n",
-       " ([9, 77, 40, 179, 136, 69, 74, 98, 29, 120, 53, 153], u'dog'),\n",
-       " ([78, 83, 93, 113, 206, 23, 121, 160, 119, 61, 60, 168], u'dog'),\n",
-       " ([105, 114, 19, 19, 211, 28, 96, 251, 208, 232, 64, 25], u'cat'),\n",
-       " ([93, 145, 128, 246, 33, 206, 73, 126, 63, 22, 150, 184], u'bird'),\n",
-       " ([12, 245, 243, 181, 134, 92, 39, 153, 112, 250, 181, 208], u'bird'),\n",
-       " ([133, 184, 53, 158, 3, 145, 47, 130, 135, 81, 80, 208], u'bird'),\n",
-       " ([143, 230, 101, 71, 156, 113, 61, 143, 37, 195, 235, 76], u'dog'),\n",
-       " ([91, 70, 17, 43, 59, 150, 227, 111, 53, 229, 0, 100], u'dog'),\n",
-       " ([136, 181, 184, 87, 132, 71, 61, 232, 143, 218, 89, 203], u'dog'),\n",
-       " ([126, 142, 84, 203, 234, 175, 17, 251, 217, 75, 145, 188], u'bird'),\n",
-       " ([198, 162, 187, 42, 9, 67, 223, 193, 154, 99, 9, 215], u'cat'),\n",
-       " ([151, 177, 164, 98, 25, 35, 240, 109, 237, 218, 28, 254], u'bird'),\n",
-       " ([246, 73, 102, 178, 4, 45, 84, 191, 87, 93, 2, 54], u'cat'),\n",
-       " ([156, 153, 39, 115, 228, 190, 35, 136, 32, 61, 171, 16], u'dog'),\n",
-       " ([152, 234, 198, 149, 191, 188, 222, 37, 110, 226, 82, 194], u'dog'),\n",
-       " ([169, 31, 163, 222, 61, 62, 119, 100, 177, 91, 34, 213], u'bird'),\n",
-       " ([67, 17, 141, 83, 188, 37, 61, 130, 187, 252, 62, 153], u'cat'),\n",
-       " ([172, 123, 115, 110, 28, 28, 140, 191, 250, 202, 253, 113], u'cat'),\n",
-       " ([225, 113, 99, 228, 109, 158, 250, 245, 47, 79, 52, 1], u'dog'),\n",
-       " ([137, 50, 48, 110, 202, 76, 211, 142, 78, 174, 232, 206], u'dog'),\n",
-       " ([166, 168, 219, 125, 201, 188, 238, 44, 160, 92, 202, 153], u'cat'),\n",
-       " ([249, 233, 133, 249, 100, 14, 43, 147, 124, 246, 223, 78], u'dog'),\n",
-       " ([45, 253, 108, 251, 135, 18, 163, 98, 143, 108, 30, 126], u'dog'),\n",
-       " ([190, 217, 97, 87, 41, 90, 64, 174, 84, 164, 188, 127], u'cat'),\n",
-       " ([56, 117, 22, 134, 249, 67, 130, 101, 62, 9, 119, 225], u'dog'),\n",
-       " ([6, 78, 138, 132, 230, 72, 93, 71, 159, 134, 161, 223], u'cat'),\n",
-       " ([245, 131, 240, 116, 186, 40, 233, 209, 174, 226, 20, 48], u'cat'),\n",
-       " ([82, 57, 189, 52, 165, 195, 129, 46, 71, 103, 118, 163], u'bird'),\n",
-       " ([21, 41, 79, 244, 93, 68, 120, 78, 184, 50, 117, 161], u'cat'),\n",
-       " ([35, 131, 23, 83, 201, 105, 140, 134, 157, 48, 73, 30], u'dog'),\n",
-       " ([144, 133, 213, 51, 51, 234, 93, 130, 222, 186, 198, 86], u'cat'),\n",
-       " ([126, 136, 125, 31, 139, 160, 161, 162, 242, 106, 11, 126], u'bird'),\n",
-       " ([168, 174, 58, 198, 13, 202, 75, 226, 254, 126, 204, 90], u'bird'),\n",
-       " ([170, 20, 197, 1, 28, 67, 137, 153, 97, 20, 57, 3], u'bird'),\n",
-       " ([43, 109, 193, 169, 94, 105, 88, 152, 46, 101, 98, 121], u'cat'),\n",
-       " ([95, 247, 19, 186, 247, 189, 206, 188, 190, 234, 254, 70], u'dog'),\n",
-       " ([96, 90, 188, 98, 16, 231, 207, 209, 145, 45, 58, 232], u'bird'),\n",
-       " ([104, 77, 39, 226, 148, 134, 217, 166, 64, 207, 99, 14], u'dog'),\n",
-       " ([33, 248, 137, 103, 124, 233, 194, 56, 75, 210, 32, 27], u'dog'),\n",
-       " ([176, 72, 221, 152, 12, 70, 229, 51, 39, 121, 185, 0], u'cat'),\n",
-       " ([249, 207, 131, 7, 90, 164, 255, 228, 11, 123, 205, 205], u'bird'),\n",
-       " ([25, 160, 211, 51, 67, 131, 123, 33, 28, 135, 102, 1], u'bird'),\n",
-       " ([142, 122, 115, 142, 154, 108, 93, 29, 115, 184, 193, 114], u'dog'),\n",
-       " ([204, 237, 105, 153, 161, 129, 57, 116, 181, 124, 247, 47], u'dog')]"
+       "[([168, 228, 110, 3, 51, 104, 192, 23, 120, 249, 96, 99], u'dog'),\n",
+       " ([20, 145, 109, 135, 149, 100, 39, 66, 124, 102, 77, 140], u'dog'),\n",
+       " ([125, 32, 244, 23, 201, 156, 251, 55, 159, 47, 160, 95], u'cat'),\n",
+       " ([24, 88, 166, 123, 193, 186, 12, 46, 65, 161, 145, 104], u'bird'),\n",
+       " ([14, 206, 47, 154, 85, 172, 186, 73, 196, 131, 229, 191], u'bird'),\n",
+       " ([131, 238, 90, 227, 51, 114, 59, 217, 237, 252, 147, 248], u'cat'),\n",
+       " ([211, 153, 187, 59, 123, 200, 10, 171, 98, 95, 87, 28], u'dog'),\n",
+       " ([26, 159, 140, 217, 89, 15, 199, 179, 242, 250, 37, 45], u'bird'),\n",
+       " ([18, 41, 102, 10, 82, 57, 163, 13, 116, 30, 213, 126], u'bird'),\n",
+       " ([56, 221, 31, 84, 132, 58, 243, 16, 19, 76, 31, 218], u'bird'),\n",
+       " ([17, 212, 36, 62, 167, 54, 103, 13, 64, 185, 70, 227], u'bird'),\n",
+       " ([186, 1, 155, 56, 201, 211, 21, 233, 38, 153, 34, 25], u'dog'),\n",
+       " ([53, 101, 200, 15, 101, 217, 227, 137, 23, 138, 191, 126], u'dog'),\n",
+       " ([255, 54, 220, 226, 252, 150, 227, 151, 207, 172, 105, 227], u'dog'),\n",
+       " ([144, 124, 183, 169, 37, 237, 14, 237, 252, 115, 198, 222], u'bird'),\n",
+       " ([222, 104, 188, 92, 254, 187, 146, 219, 157, 142, 113, 128], u'cat'),\n",
+       " ([64, 44, 142, 35, 193, 30, 159, 120, 199, 196, 101, 213], u'bird'),\n",
+       " ([96, 72, 120, 63, 69, 86, 167, 0, 177, 165, 187, 67], u'dog'),\n",
+       " ([88, 210, 241, 216, 246, 48, 4, 132, 83, 197, 162, 242], u'cat'),\n",
+       " ([105, 182, 162, 62, 104, 2, 134, 223, 65, 203, 53, 231], u'bird'),\n",
+       " ([230, 140, 134, 42, 12, 223, 251, 252, 183, 241, 44, 188], u'dog'),\n",
+       " ([127, 129, 24, 113, 190, 129, 40, 96, 191, 143, 98, 69], u'dog'),\n",
+       " ([162, 16, 163, 137, 219, 137, 21, 97, 179, 33, 64, 174], u'cat'),\n",
+       " ([247, 159, 74, 179, 21, 201, 51, 45, 58, 241, 175, 98], u'cat'),\n",
+       " ([110, 241, 179, 179, 96, 85, 195, 3, 222, 158, 140, 244], u'bird'),\n",
+       " ([63, 21, 63, 237, 50, 54, 140, 124, 233, 162, 69, 28], u'bird'),\n",
+       " ([94, 111, 234, 231, 203, 73, 118, 97, 57, 254, 209, 131], u'dog'),\n",
+       " ([246, 73, 151, 78, 201, 43, 59, 1, 215, 155, 138, 63], u'dog'),\n",
+       " ([46, 186, 18, 158, 254, 111, 13, 232, 86, 216, 49, 204], u'cat'),\n",
+       " ([106, 202, 9, 238, 104, 256, 55, 255, 78, 0, 42, 137], u'cat'),\n",
+       " ([1, 35, 139, 64, 121, 185, 250, 139, 87, 248, 250, 100], u'bird'),\n",
+       " ([81, 59, 17, 29, 116, 124, 231, 125, 105, 79, 124, 160], u'cat'),\n",
+       " ([202, 160, 119, 83, 161, 120, 118, 44, 183, 239, 230, 177], u'dog'),\n",
+       " ([61, 169, 117, 160, 136, 197, 220, 153, 226, 79, 21, 201], u'bird'),\n",
+       " ([126, 23, 73, 30, 100, 19, 191, 219, 102, 96, 83, 220], u'dog'),\n",
+       " ([10, 203, 113, 187, 70, 174, 99, 186, 78, 235, 128, 42], u'dog'),\n",
+       " ([98, 122, 154, 42, 70, 24, 66, 143, 54, 166, 161, 245], u'dog'),\n",
+       " ([7, 84, 211, 227, 224, 221, 174, 82, 152, 244, 255, 251], u'bird'),\n",
+       " ([78, 230, 46, 120, 106, 144, 241, 4, 186, 55, 28, 252], u'bird'),\n",
+       " ([82, 162, 103, 71, 35, 110, 156, 246, 81, 124, 211, 255], u'bird'),\n",
+       " ([106, 243, 205, 101, 161, 26, 75, 207, 146, 181, 94, 132], u'bird'),\n",
+       " ([24, 187, 213, 20, 129, 39, 182, 232, 110, 217, 86, 10], u'bird'),\n",
+       " ([168, 134, 161, 167, 83, 12, 154, 32, 113, 58, 58, 188], u'cat'),\n",
+       " ([205, 113, 103, 80, 42, 128, 11, 255, 148, 140, 39, 74], u'dog'),\n",
+       " ([149, 34, 203, 159, 241, 114, 37, 146, 25, 120, 158, 179], u'dog'),\n",
+       " ([15, 237, 210, 202, 246, 159, 59, 94, 239, 101, 221, 250], u'dog'),\n",
+       " ([113, 134, 139, 187, 250, 32, 222, 197, 192, 206, 55, 229], u'dog'),\n",
+       " ([81, 93, 255, 4, 244, 13, 241, 198, 215, 231, 101, 18], u'cat'),\n",
+       " ([84, 120, 34, 78, 220, 147, 212, 103, 79, 206, 136, 44], u'dog'),\n",
+       " ([71, 251, 203, 44, 91, 28, 136, 90, 31, 124, 103, 16], u'cat'),\n",
+       " ([62, 248, 167, 81, 60, 251, 200, 95, 72, 164, 242, 28], u'cat'),\n",
+       " ([65, 235, 147, 109, 126, 219, 103, 73, 6, 195, 101, 143], u'cat')]"
       ]
      },
-     "execution_count": 9,
+     "execution_count": 12,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1058,7 +1035,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 10,
+   "execution_count": 13,
    "metadata": {},
    "outputs": [
     {
@@ -1075,8 +1052,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1095,7 +1072,7 @@
        "[([26, 12], [26, 3], 0), ([26, 12], [26, 3], 1)]"
       ]
      },
-     "execution_count": 10,
+     "execution_count": 13,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1112,7 +1089,7 @@
     "                                        255                   -- Normalizing constant\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -1127,7 +1104,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 11,
+   "execution_count": 14,
    "metadata": {},
    "outputs": [
     {
@@ -1144,8 +1121,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1164,7 +1141,7 @@
        "[([26, 12], [26, 3], 0), ([26, 12], [26, 3], 1)]"
       ]
      },
-     "execution_count": 11,
+     "execution_count": 14,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1182,7 +1159,7 @@
     "    NULL                      -- Buffer size\n",
     "    );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -1197,7 +1174,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 12,
+   "execution_count": 15,
    "metadata": {},
    "outputs": [
     {
@@ -1214,13 +1191,13 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[8, 12]</td>\n",
-       "        <td>[8, 3]</td>\n",
+       "        <td>[9, 12]</td>\n",
+       "        <td>[9, 3]</td>\n",
        "        <td>0</td>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1244,22 +1221,22 @@
        "        <td>4</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[8, 12]</td>\n",
-       "        <td>[8, 3]</td>\n",
+       "        <td>[7, 12]</td>\n",
+       "        <td>[7, 3]</td>\n",
        "        <td>5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[([8, 12], [8, 3], 0),\n",
+       "[([9, 12], [9, 3], 0),\n",
        " ([9, 12], [9, 3], 1),\n",
        " ([9, 12], [9, 3], 2),\n",
        " ([9, 12], [9, 3], 3),\n",
        " ([9, 12], [9, 3], 4),\n",
-       " ([8, 12], [8, 3], 5)]"
+       " ([7, 12], [7, 3], 5)]"
       ]
      },
-     "execution_count": 12,
+     "execution_count": 15,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1276,7 +1253,7 @@
     "                                        255                   -- Normalizing constant\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -1288,7 +1265,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 13,
+   "execution_count": 16,
    "metadata": {},
    "outputs": [
     {
@@ -1308,7 +1285,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -1318,23 +1295,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
-       "        <td>10</td>\n",
+       "        <td>9</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 10, 255.0, 3, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 9, 255.0, [3], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 13,
+     "execution_count": 16,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1356,7 +1333,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 14,
+   "execution_count": 18,
    "metadata": {},
    "outputs": [
     {
@@ -1373,8 +1350,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1393,7 +1370,7 @@
        "[([26, 12], [26, 5], 0), ([26, 12], [26, 5], 1)]"
       ]
      },
-     "execution_count": 14,
+     "execution_count": 18,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1408,15 +1385,15 @@
     "                                        'rgb',                -- Independent variable\n",
     "                                        NULL,                 -- Buffer size\n",
     "                                        255,                  -- Normalizing constant\n",
-    "                                        5                     -- Number of desired class values\n",
+    "                                        ARRAY[5]              -- Number of desired class values\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 15,
+   "execution_count": 19,
    "metadata": {},
    "outputs": [
     {
@@ -1436,7 +1413,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -1446,23 +1423,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog', None, None]</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>5</td>\n",
+       "        <td>[5]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog', None, None], 26, 255.0, 5, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog', None, None], 26, 255.0, [5], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 15,
+     "execution_count": 19,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1513,7 +1490,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 17,
+   "execution_count": 20,
    "metadata": {},
    "outputs": [
     {
@@ -1538,7 +1515,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -1548,23 +1525,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>[2, 3]</td>\n",
        "        <td>[0, 1]</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 26, 255.0, 3, [2, 3], [0, 1])]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 26, 255.0, [3], [2, 3], [0, 1])]"
       ]
      },
-     "execution_count": 17,
+     "execution_count": 20,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1610,7 +1587,7 @@
    "name": "python",
    "nbconvert_exporter": "python",
    "pygments_lexer": "ipython2",
-   "version": "2.7.10"
+   "version": "2.7.16"
   }
  },
  "nbformat": 4,
diff --git a/community-artifacts/Deep-learning/Model-preparation/Define-custom-functions-v1.ipynb b/community-artifacts/Deep-learning/Model-preparation/Define-custom-functions-v1.ipynb
new file mode 100755
index 0000000..f1d301b
--- /dev/null
+++ b/community-artifacts/Deep-learning/Model-preparation/Define-custom-functions-v1.ipynb
@@ -0,0 +1,531 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Define custom functions\n",
+    "\n",
+    "This function loads custom Python functions into a table for use by deep learning algorithms.\n",
+    "\n",
+    "Custom functions can be useful if, for example, you need loss functions or metrics that are not built into the standard libraries. The functions to be loaded must be in the form of serialized Python objects created using Dill, which extends Python's pickle module to the majority of the built-in Python types.\n",
+    "\n",
+    "Custom functions are also used to return top k categorical accuracy rate in the case that you want a different k value than the default from Keras. This module includes a helper function to create the custom function automatically for a specified k.\n",
+    "\n",
+    "This method was added in MADlib 1.18.0.\n",
+    "\n",
+    "## <em>Warning</em>\n",
+    "<em>For security reasons there are controls on custom functions in MADlib. You must be a superuser to create custom functions because they could theoretically allow execution of any untrusted Python code. Regular users with MADlib USAGE permission can use existing custom functions but cannot create new ones or update existing ones.</em>\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#load_psycopg2\">1. Load object using psycopg2</a>\n",
+    "\n",
+    "<a href=\"#load_plpython\">2. Load object using a PL/Python function</a>\n",
+    "\n",
+    "<a href=\"#delete_object\">3. Delete object</a>\n",
+    "\n",
+    "<a href=\"#top_k\">4. Top k accuracy function</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_psycopg2\"></a>\n",
+    "# 1. Load object using psycopg2\n",
+    "Psycopg is a PostgreSQL database adapter for the Python programming language. Note need to use the psycopg2.Binary() method to pass as bytes."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# import database connector psycopg2 and create connection cursor\n",
+    "import psycopg2 as p2\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
+    "cur = conn.cursor()\n",
+    "\n",
+    "# import Dill and define functions\n",
+    "import dill\n",
+    "\n",
+    "# custom loss\n",
+    "def squared_error(y_true, y_pred):\n",
+    "    import keras.backend as K \n",
+    "    return K.square(y_pred - y_true)\n",
+    "pb_squared_error=dill.dumps(squared_error)\n",
+    "\n",
+    "# custom metric\n",
+    "def rmse(y_true, y_pred):\n",
+    "    import keras.backend as K \n",
+    "    return K.sqrt(K.mean(K.square(y_pred - y_true), axis=-1))\n",
+    "pb_rmse=dill.dumps(rmse)\n",
+    "\n",
+    "# call load function\n",
+    "cur.execute(\"DROP TABLE IF EXISTS madlib.custom_function_table\")\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'squared_error', 'squared error')\", [p2.Binary(pb_squared_error)])\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'rmse', 'root mean square error')\", [p2.Binary(pb_rmse)])\n",
+    "conn.commit()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>squared_error</td>\n",
+       "        <td>squared error</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>rmse</td>\n",
+       "        <td>root mean square error</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'squared_error', u'squared error'),\n",
+       " (2, u'rmse', u'root mean square error')]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_plpython\"></a>\n",
+    "# 2. Load object using a PL/Python function"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "CREATE OR REPLACE FUNCTION custom_function_squared_error()\n",
+    "RETURNS BYTEA AS\n",
+    "$$\n",
+    "import dill\n",
+    "def squared_error(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.square(y_pred - y_true)\n",
+    "pb_squared_error=dill.dumps(squared_error)\n",
+    "return pb_squared_error\n",
+    "$$ language plpythonu;\n",
+    "CREATE OR REPLACE FUNCTION custom_function_rmse()\n",
+    "RETURNS BYTEA AS\n",
+    "$$\n",
+    "import dill\n",
+    "def rmse(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.sqrt(K.mean(K.square(y_pred - y_true), axis=-1))\n",
+    "pb_rmse=dill.dumps(rmse)\n",
+    "return pb_rmse\n",
+    "$$ language plpythonu;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>load_custom_function</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS madlib.custom_function_table;\n",
+    "SELECT madlib.load_custom_function('custom_function_table', \n",
+    "                                   custom_function_squared_error(), \n",
+    "                                   'squared_error', \n",
+    "                                   'squared error');\n",
+    "\n",
+    "SELECT madlib.load_custom_function('custom_function_table', \n",
+    "                                   custom_function_rmse(), \n",
+    "                                   'rmse', \n",
+    "                                   'root mean square error');"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>squared_error</td>\n",
+       "        <td>squared error</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>rmse</td>\n",
+       "        <td>root mean square error</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'squared_error', u'squared error'),\n",
+       " (2, u'rmse', u'root mean square error')]"
+      ]
+     },
+     "execution_count": 8,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"delete_object\"></a>\n",
+    "# 3. Delete object\n",
+    "Delete by id:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>rmse</td>\n",
+       "        <td>root mean square error</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2, u'rmse', u'root mean square error')]"
+      ]
+     },
+     "execution_count": 9,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.delete_custom_function( 'custom_function_table', 1);\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Delete by name:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>delete_custom_function</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.delete_custom_function( 'custom_function_table', 'rmse');"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Since this was the last object in the table, if you delete it then the table will also be dropped."
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"top_k\"></a>\n",
+    "# 4. Top k accuracy function\n",
+    "Load top 3 accuracy function followed by a top 10 accuracy function:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>top_3_accuracy</td>\n",
+       "        <td>returns top_3_accuracy</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>top_10_accuracy</td>\n",
+       "        <td>returns top_10_accuracy</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'top_3_accuracy', u'returns top_3_accuracy'),\n",
+       " (2, u'top_10_accuracy', u'returns top_10_accuracy')]"
+      ]
+     },
+     "execution_count": 11,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS madlib.custom_function_table;\n",
+    "\n",
+    "SELECT madlib.load_top_k_accuracy_function('custom_function_table',\n",
+    "                                           3);\n",
+    "\n",
+    "SELECT madlib.load_top_k_accuracy_function('custom_function_table',\n",
+    "                                           10);\n",
+    "\n",
+    "SELECT id, name, description FROM madlib.custom_function_table ORDER BY id;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Load-model-architecture-v2.ipynb b/community-artifacts/Deep-learning/Model-preparation/Define-model-architecture-v2.ipynb
similarity index 68%
copy from community-artifacts/Deep-learning/Load-model-architecture-v2.ipynb
copy to community-artifacts/Deep-learning/Model-preparation/Define-model-architecture-v2.ipynb
index 8aa3716..b823f09 100644
--- a/community-artifacts/Deep-learning/Load-model-architecture-v2.ipynb
+++ b/community-artifacts/Deep-learning/Model-preparation/Define-model-architecture-v2.ipynb
@@ -4,10 +4,16 @@
    "cell_type": "markdown",
    "metadata": {},
    "source": [
-    "# Load model architecture\n",
-    "This utility function loads model architectures and weights into a table for use by deep learning algorithms in Keras.  \n",
+    "# Define model architecture\n",
+    "This function loads model architectures and weights into a table for use by deep learning algorithms.\n",
     "\n",
-    "The model architecture loader was added in MADlib 1.16.\n",
+    "Model architecture is in JSON form and model weights are in the form of PostgreSQL binary data types (bytea). If the output table already exists, a new row is inserted into the table so it can act as a repository for multiple model architectures and weights.\n",
+    "\n",
+    "There is also a function to delete a model from the table.\n",
+    "\n",
+    "MADlib's deep learning methods are designed to use the TensorFlow package and its built in Keras functions. To ensure consistency, please use tensorflow.keras objects (models, layers, etc.) instead of importing Keras and using its objects.\n",
+    "\n",
+    "The model architecture loader was added in MADlib 1.16 and updated after that.\n",
     "\n",
     "## Table of contents\n",
     "\n",
@@ -25,17 +31,15 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 1,
+   "execution_count": 34,
    "metadata": {},
    "outputs": [
     {
-     "name": "stderr",
+     "name": "stdout",
      "output_type": "stream",
      "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
+      "The sql extension is already loaded. To reload it, use:\n",
+      "  %reload_ext sql\n"
      ]
     }
    ],
@@ -45,24 +49,10 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 5,
+   "execution_count": 35,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 5,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
     "# Greenplum Database 5.x on GCP - via tunnel\n",
     "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
@@ -72,7 +62,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
+   "execution_count": 36,
    "metadata": {},
    "outputs": [
     {
@@ -90,15 +80,15 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
-     "execution_count": 6,
+     "execution_count": 36,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -120,28 +110,13 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 37,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "Using TensorFlow backend.\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Couldn't import dot_parser, loading of dot files will not be possible.\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
-    "import keras\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense"
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
    ]
   },
   {
@@ -153,21 +128,22 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 8,
+   "execution_count": 38,
    "metadata": {},
    "outputs": [
     {
      "name": "stdout",
      "output_type": "stream",
      "text": [
+      "Model: \"sequential_3\"\n",
       "_________________________________________________________________\n",
       "Layer (type)                 Output Shape              Param #   \n",
       "=================================================================\n",
-      "dense_1 (Dense)              (None, 10)                50        \n",
+      "dense_9 (Dense)              (None, 10)                50        \n",
       "_________________________________________________________________\n",
-      "dense_2 (Dense)              (None, 10)                110       \n",
+      "dense_10 (Dense)             (None, 10)                110       \n",
       "_________________________________________________________________\n",
-      "dense_3 (Dense)              (None, 3)                 33        \n",
+      "dense_11 (Dense)             (None, 3)                 33        \n",
       "=================================================================\n",
       "Total params: 193\n",
       "Trainable params: 193\n",
@@ -182,21 +158,21 @@
     "model.add(Dense(10, activation='relu'))\n",
     "model.add(Dense(3, activation='softmax'))\n",
     "    \n",
-    "model.summary()"
+    "model.summary();"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 9,
+   "execution_count": 39,
    "metadata": {},
    "outputs": [
     {
      "data": {
       "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_9\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_10\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_11\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_3\"}, \"backend\": \"tensorflow\"}'"
       ]
      },
-     "execution_count": 9,
+     "execution_count": 39,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -225,7 +201,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 10,
+   "execution_count": 40,
    "metadata": {},
    "outputs": [
     {
@@ -255,15 +231,15 @@
        "        <td>None</td>\n",
        "        <td>Sophie</td>\n",
        "        <td>A simple model</td>\n",
-       "        <td>__madlib_temp_19839392_1576692433_56744839__</td>\n",
+       "        <td>__madlib_temp_27065614_1614901189_16021319__</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_19839392_1576692433_56744839__')]"
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_27065614_1614901189_16021319__')]"
       ]
      },
-     "execution_count": 10,
+     "execution_count": 40,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -294,7 +270,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 11,
+   "execution_count": 41,
    "metadata": {},
    "outputs": [
     {
@@ -323,7 +299,7 @@
        "        <td>None</td>\n",
        "        <td>Maria</td>\n",
        "        <td>Also a simple model</td>\n",
-       "        <td>__madlib_temp_36064316_1576692433_8110861__</td>\n",
+       "        <td>__madlib_temp_87665369_1614901189_11144097__</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
@@ -331,16 +307,16 @@
        "        <td>None</td>\n",
        "        <td>Sophie</td>\n",
        "        <td>A simple model</td>\n",
-       "        <td>__madlib_temp_19839392_1576692433_56744839__</td>\n",
+       "        <td>__madlib_temp_27065614_1614901189_16021319__</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'Also a simple model', u'__madlib_temp_36064316_1576692433_8110861__'),\n",
-       " (1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_19839392_1576692433_56744839__')]"
+       "[(2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'Also a simple model', u'__madlib_temp_87665369_1614901189_11144097__'),\n",
+       " (1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'A simple model', u'__madlib_temp_27065614_1614901189_16021319__')]"
       ]
      },
-     "execution_count": 11,
+     "execution_count": 41,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -376,7 +352,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 12,
+   "execution_count": 42,
    "metadata": {},
    "outputs": [
     {
@@ -384,7 +360,7 @@
      "output_type": "stream",
      "text": [
       "1 rows affected.\n",
-      "1 rows affected.\n"
+      "2 rows affected.\n"
      ]
     },
     {
@@ -392,18 +368,31 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>count</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "        <td>False</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1L,)]"
+       "[(1, u'Sophie', u'A simple model', False),\n",
+       " (2, u'Maria', u'Also a simple model', True)]"
       ]
      },
-     "execution_count": 12,
+     "execution_count": 42,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -416,7 +405,7 @@
     "WHERE model_arch_library.model_id = 2;\n",
     "\n",
     "-- Check weights loaded OK\n",
-    "SELECT COUNT(*) FROM model_arch_library WHERE model_weights IS NOT NULL;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   },
   {
@@ -430,7 +419,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 13,
+   "execution_count": 43,
    "metadata": {},
    "outputs": [
     {
@@ -438,7 +427,6 @@
      "output_type": "stream",
      "text": [
       "Done.\n",
-      "1 rows affected.\n",
       "1 rows affected.\n"
      ]
     },
@@ -447,18 +435,18 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>count</th>\n",
+       "        <th>load_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>2</td>\n",
+       "        <td></td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(2L,)]"
+       "[('',)]"
       ]
      },
-     "execution_count": 13,
+     "execution_count": 43,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -467,8 +455,8 @@
     "%%sql\n",
     "CREATE OR REPLACE FUNCTION load_weights() RETURNS VOID AS\n",
     "$$\n",
-    "from keras.layers import *\n",
-    "from keras import Sequential\n",
+    "from tensorflow.keras.layers import *\n",
+    "from tensorflow.keras import Sequential\n",
     "import numpy as np\n",
     "import plpy\n",
     "\n",
@@ -493,15 +481,12 @@
     "$$ language plpythonu;\n",
     "\n",
     "-- Call load function\n",
-    "SELECT load_weights();\n",
-    "\n",
-    "-- Check weights loaded OK\n",
-    "SELECT COUNT(*) FROM model_arch_library WHERE model_weights IS NOT NULL;"
+    "SELECT load_weights();"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 14,
+   "execution_count": 44,
    "metadata": {},
    "outputs": [
     {
@@ -518,33 +503,43 @@
        "    <tr>\n",
        "        <th>model_id</th>\n",
        "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
        "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "        <td>False</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>3</td>\n",
        "        <td>Ella</td>\n",
+       "        <td>Model x</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, u'Sophie'), (2, u'Maria'), (3, u'Ella')]"
+       "[(1, u'Sophie', u'A simple model', False),\n",
+       " (2, u'Maria', u'Also a simple model', True),\n",
+       " (3, u'Ella', u'Model x', True)]"
       ]
      },
-     "execution_count": 14,
+     "execution_count": 44,
      "metadata": {},
      "output_type": "execute_result"
     }
    ],
    "source": [
     "%%sql\n",
-    "SELECT model_id, name from model_arch_library ORDER BY model_id;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   },
   {
@@ -560,45 +555,16 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 15,
+   "execution_count": 45,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(2L,)]"
-      ]
-     },
-     "execution_count": 15,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
     "import psycopg2 as p2\n",
-    "#conn = p2.connect('postgresql://gpadmin@35.239.240.26:5432/madlib')\n",
-    "conn = p2.connect('postgresql://fmcquillan@localhost:5432/madlib')\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
     "cur = conn.cursor()\n",
     "\n",
-    "from keras.layers import *\n",
-    "from keras import Sequential\n",
+    "from tensorflow.keras.layers import *\n",
+    "from tensorflow.keras import Sequential\n",
     "import numpy as np\n",
     "\n",
     "# create model\n",
@@ -615,22 +581,19 @@
     "\n",
     "query = \"SELECT madlib.load_keras_model('model_arch_library', %s,%s,%s,%s)\"\n",
     "cur.execute(query,[model.to_json(), weights_bytea, \"Grace\", \"Model y\"])\n",
-    "conn.commit()\n",
-    "\n",
-    "# check weights loaded OK\n",
-    "%sql SELECT COUNT(*) FROM model_arch_library WHERE model_weights IS NOT NULL;"
+    "conn.commit()"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 16,
+   "execution_count": 46,
    "metadata": {},
    "outputs": [
     {
      "name": "stdout",
      "output_type": "stream",
      "text": [
-      "3 rows affected.\n"
+      "4 rows affected.\n"
      ]
     },
     {
@@ -640,33 +603,50 @@
        "    <tr>\n",
        "        <th>model_id</th>\n",
        "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>1</td>\n",
        "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "        <td>False</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>3</td>\n",
        "        <td>Ella</td>\n",
+       "        <td>Model x</td>\n",
+       "        <td>True</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>Grace</td>\n",
+       "        <td>Model y</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(1, u'Sophie'), (2, u'Maria'), (3, u'Ella')]"
+       "[(1, u'Sophie', u'A simple model', False),\n",
+       " (2, u'Maria', u'Also a simple model', True),\n",
+       " (3, u'Ella', u'Model x', True),\n",
+       " (4, u'Grace', u'Model y', True)]"
       ]
      },
-     "execution_count": 16,
+     "execution_count": 46,
      "metadata": {},
      "output_type": "execute_result"
     }
    ],
    "source": [
     "%%sql\n",
-    "SELECT model_id, name from model_arch_library ORDER BY model_id;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   },
   {
@@ -679,7 +659,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 17,
+   "execution_count": 47,
    "metadata": {},
    "outputs": [
     {
@@ -687,7 +667,7 @@
      "output_type": "stream",
      "text": [
       "1 rows affected.\n",
-      "2 rows affected.\n"
+      "3 rows affected.\n"
      ]
     },
     {
@@ -697,22 +677,36 @@
        "    <tr>\n",
        "        <th>model_id</th>\n",
        "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>has_model_weights</th>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>2</td>\n",
        "        <td>Maria</td>\n",
+       "        <td>Also a simple model</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "    <tr>\n",
        "        <td>3</td>\n",
        "        <td>Ella</td>\n",
+       "        <td>Model x</td>\n",
+       "        <td>True</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>Grace</td>\n",
+       "        <td>Model y</td>\n",
+       "        <td>True</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(2, u'Maria'), (3, u'Ella')]"
+       "[(2, u'Maria', u'Also a simple model', True),\n",
+       " (3, u'Ella', u'Model x', True),\n",
+       " (4, u'Grace', u'Model y', True)]"
       ]
      },
-     "execution_count": 17,
+     "execution_count": 47,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -722,7 +716,7 @@
     "SELECT madlib.delete_keras_model('model_arch_library',   -- Output table\n",
     "                                  1                      -- Model id\n",
     "                                );\n",
-    "SELECT model_id, name from model_arch_library ORDER BY model_id;"
+    "SELECT model_id, name, description, (model_weights IS NOT NULL) AS has_model_weights FROM model_arch_library ORDER BY model_id;"
    ]
   }
  ],
@@ -742,7 +736,7 @@
    "name": "python",
    "nbconvert_exporter": "python",
    "pygments_lexer": "ipython2",
-   "version": "2.7.10"
+   "version": "2.7.16"
   }
  },
  "nbformat": 4,
diff --git a/community-artifacts/Deep-learning/Preprocessor-for-images-distribution-rules-v1.ipynb b/community-artifacts/Deep-learning/Model-preparation/Preprocessor-for-images-distribution-rules-v1.ipynb
similarity index 98%
copy from community-artifacts/Deep-learning/Preprocessor-for-images-distribution-rules-v1.ipynb
copy to community-artifacts/Deep-learning/Model-preparation/Preprocessor-for-images-distribution-rules-v1.ipynb
index b457303..0ae2b4c 100644
--- a/community-artifacts/Deep-learning/Preprocessor-for-images-distribution-rules-v1.ipynb
+++ b/community-artifacts/Deep-learning/Model-preparation/Preprocessor-for-images-distribution-rules-v1.ipynb
@@ -1198,7 +1198,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1465,7 +1465,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1657,7 +1657,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1805,7 +1805,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_train_packed ORDER BY __dist_key__;"
    ]
   },
   {
@@ -1938,7 +1938,7 @@
    ],
    "source": [
     "%%sql\n",
-    "SELECT __dist_key__, independent_var_shape, dependent_var_shape, buffer_id FROM image_data_val_packed ORDER BY __dist_key__;"
+    "SELECT __dist_key__, x_shape, y_shape, buffer_id FROM image_data_val_packed ORDER BY __dist_key__;"
    ]
   }
  ],
diff --git a/community-artifacts/Deep-learning/Preprocessor-for-images-v2.ipynb b/community-artifacts/Deep-learning/Model-preparation/Preprocessor-for-images-v2.ipynb
similarity index 61%
rename from community-artifacts/Deep-learning/Preprocessor-for-images-v2.ipynb
rename to community-artifacts/Deep-learning/Model-preparation/Preprocessor-for-images-v2.ipynb
index cb76d1e..5fd5a69 100644
--- a/community-artifacts/Deep-learning/Preprocessor-for-images-v2.ipynb
+++ b/community-artifacts/Deep-learning/Model-preparation/Preprocessor-for-images-v2.ipynb
@@ -5,11 +5,11 @@
    "metadata": {},
    "source": [
     "# Preprocessor for image data\n",
-    "This is a mini-batch preprocessor utility for image data:\n",
+    "This preprocessor prepares training data for deep learning.\n",
     "* training_preprocessor_dl() for training datasets\n",
     "* validation_preprocessor_dl() for validation datasets\n",
     "\n",
-    "Note that there is a separate mini-batch preprocessor utility for general use cases\n",
+    "Note that there is a separate mini-batch preprocessor utility for non deep learning use cases\n",
     "http://madlib.apache.org/docs/latest/group__grp__minibatch__preprocessing.html\n",
     "\n",
     "The preprocessor for image data was added in MADlib 1.16.\n",
@@ -39,42 +39,17 @@
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
     "%load_ext sql"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 2,
+   "execution_count": 3,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
     "# Greenplum Database 5.x on GCP - via tunnel\n",
     "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
@@ -84,7 +59,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 3,
+   "execution_count": 5,
    "metadata": {},
    "outputs": [
     {
@@ -102,15 +77,15 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-85-g4bac900, cmake configuration time: Wed Mar  3 20:37:11 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-85-g4bac900, cmake configuration time: Wed Mar  3 20:37:11 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
-     "execution_count": 3,
+     "execution_count": 5,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -132,7 +107,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 4,
+   "execution_count": 6,
    "metadata": {},
    "outputs": [
     {
@@ -153,271 +128,271 @@
        "        <th>species</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[152, 186, 35], [102, 145, 138]], [[40, 249, 108], [175, 207, 70]]]</td>\n",
+       "        <td>[[[17, 201, 110], [175, 136, 179]], [[102, 57, 24], [110, 199, 64]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[205, 85, 56], [209, 11, 117]], [[86, 82, 41], [226, 192, 132]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[234, 110, 251], [147, 18, 158]], [[55, 79, 14], [140, 50, 143]]]</td>\n",
+       "        <td>[[[209, 227, 160], [86, 88, 177]], [[31, 198, 96], [167, 122, 198]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[146, 52, 167], [210, 33, 116]], [[38, 89, 69], [50, 207, 155]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[247, 125, 68], [124, 196, 20]], [[95, 100, 107], [183, 21, 138]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[117, 49, 248], [59, 18, 137]], [[110, 186, 91], [143, 46, 129]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[115, 179, 183], [14, 54, 175]], [[138, 122, 42], [79, 142, 137]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[249, 65, 200], [131, 191, 61]], [[180, 182, 119], [199, 63, 230]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[154, 117, 174], [27, 94, 33]], [[206, 21, 46], [4, 196, 185]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[238, 8, 12], [120, 187, 4]], [[184, 130, 135], [119, 191, 59]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[179, 202, 20], [219, 198, 173]], [[149, 233, 18], [38, 115, 59]]]</td>\n",
+       "        <td>[[[55, 2, 109], [28, 130, 7]], [[146, 48, 34], [240, 81, 240]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[223, 234, 239], [37, 253, 217]], [[147, 248, 108], [166, 150, 162]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[164, 46, 39], [51, 130, 218]], [[253, 150, 181], [195, 66, 75]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[85, 113, 32], [144, 145, 255]], [[122, 127, 36], [118, 88, 183]]]</td>\n",
+       "        <td>[[[128, 244, 200], [57, 113, 182]], [[64, 125, 46], [251, 129, 230]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[195, 93, 4], [102, 81, 168]], [[148, 120, 219], [21, 82, 217]]]</td>\n",
+       "        <td>[[[8, 93, 61], [67, 139, 115]], [[69, 248, 144], [199, 255, 33]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[8, 156, 237], [82, 72, 66]], [[196, 104, 210], [84, 103, 75]]]</td>\n",
+       "        <td>[[[33, 17, 73], [17, 21, 201]], [[5, 222, 1], [118, 148, 66]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[139, 194, 43], [66, 48, 239]], [[159, 52, 84], [240, 220, 232]]]</td>\n",
+       "        <td>[[[194, 61, 116], [168, 187, 124]], [[6, 247, 192], [145, 106, 5]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[183, 253, 187], [144, 168, 194]], [[44, 150, 21], [116, 216, 216]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[170, 44, 68], [245, 256, 207]], [[183, 43, 17], [231, 25, 176]]]</td>\n",
+       "        <td>[[[250, 204, 135], [27, 196, 168]], [[44, 12, 185], [65, 213, 190]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[110, 160, 246], [85, 9, 173]], [[82, 195, 61], [251, 134, 105]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[154, 222, 104], [114, 186, 18]], [[159, 254, 7], [158, 205, 190]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[222, 165, 227], [142, 191, 80]], [[46, 182, 165], [55, 99, 248]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[161, 243, 128], [10, 131, 26]], [[232, 235, 141], [162, 253, 43]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[4, 202, 109], [194, 147, 75]], [[103, 117, 217], [39, 197, 8]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[107, 63, 64], [99, 57, 224]], [[86, 185, 234], [216, 212, 210]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[96, 116, 192], [140, 21, 196]], [[85, 130, 135], [232, 206, 238]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[167, 20, 35], [174, 241, 142]], [[237, 48, 241], [38, 16, 70]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[251, 31, 179], [205, 226, 19]], [[65, 162, 159], [86, 103, 244]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[237, 220, 166], [219, 58, 77]], [[239, 93, 251], [224, 235, 232]]]</td>\n",
+       "        <td>[[[215, 52, 179], [25, 39, 117]], [[86, 155, 29], [16, 24, 35]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[219, 14, 33], [34, 237, 28]], [[64, 160, 232], [34, 180, 41]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[83, 127, 43], [71, 87, 24]], [[35, 253, 243], [93, 74, 227]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[69, 195, 165], [45, 212, 129]], [[59, 245, 162], [40, 16, 226]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[248, 5, 124], [34, 201, 206]], [[161, 244, 21], [248, 13, 57]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[0, 150, 63], [227, 80, 132]], [[166, 245, 176], [121, 118, 235]]]</td>\n",
+       "        <td>[[[215, 180, 113], [220, 61, 107]], [[168, 196, 134], [108, 108, 178]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[104, 42, 37], [143, 227, 111]], [[96, 135, 172], [12, 207, 100]]]</td>\n",
+       "        <td>[[[38, 244, 77], [228, 19, 36]], [[24, 198, 60], [63, 59, 146]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[221, 150, 126], [143, 129, 93]], [[92, 235, 60], [174, 100, 100]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[216, 163, 35], [249, 33, 139]], [[35, 70, 26], [6, 181, 122]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[97, 134, 93], [198, 94, 57]], [[92, 219, 200], [221, 56, 35]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[116, 210, 44], [216, 129, 4]], [[123, 164, 253], [156, 47, 32]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[73, 39, 151], [196, 180, 248]], [[74, 16, 190], [168, 74, 26]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[18, 246, 187], [53, 190, 47]], [[7, 234, 8], [136, 238, 131]]]</td>\n",
+       "        <td>[[[89, 162, 242], [124, 169, 202]], [[48, 26, 166], [109, 134, 78]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[235, 31, 91], [11, 1, 164]], [[49, 152, 103], [229, 144, 177]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[78, 89, 63], [104, 220, 81]], [[94, 151, 134], [28, 199, 141]]]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[206, 21, 244], [81, 65, 223]], [[112, 155, 234], [113, 63, 27]]]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[166, 1, 152], [88, 246, 230]], [[176, 54, 78], [140, 135, 172]]]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[13, 200, 234], [155, 207, 185]], [[176, 195, 10], [240, 162, 122]]]</td>\n",
+       "        <td>[[[12, 185, 157], [191, 49, 195]], [[178, 126, 167], [197, 162, 191]]]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[140, 235, 202], [167, 244, 113]], [[168, 140, 200], [158, 114, 121]]]</td>\n",
+       "        <td>[[[222, 254, 199], [112, 217, 32]], [[18, 203, 156], [187, 148, 204]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[192, 5, 91], [108, 41, 104]], [[52, 19, 3], [3, 204, 178]]]</td>\n",
+       "        <td>[[[58, 56, 91], [136, 105, 103]], [[65, 6, 38], [114, 201, 216]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[214, 162, 103], [80, 46, 243]], [[60, 248, 154], [47, 105, 65]]]</td>\n",
+       "        <td>[[[111, 157, 147], [46, 41, 113]], [[44, 240, 226], [5, 15, 244]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[49, 223, 45], [170, 179, 237]], [[175, 14, 89], [216, 118, 141]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[121, 144, 183], [43, 86, 141]], [[205, 189, 221], [251, 176, 25]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[74, 72, 92], [139, 3, 141]], [[106, 48, 55], [29, 30, 230]]]</td>\n",
+       "        <td>[[[171, 175, 100], [119, 132, 158]], [[175, 224, 37], [24, 71, 102]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[119, 190, 161], [4, 168, 25]], [[148, 95, 68], [234, 236, 17]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[201, 13, 87], [226, 256, 161]], [[42, 92, 44], [45, 233, 150]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[33, 179, 122], [7, 222, 241]], [[196, 127, 246], [108, 152, 138]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[220, 116, 183], [237, 27, 128]], [[250, 115, 98], [250, 19, 140]]]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[64, 184, 64], [214, 21, 96]], [[137, 143, 103], [103, 129, 43]]]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[[[118, 151, 126], [1, 99, 90]], [[117, 26, 71], [144, 154, 65]]]</td>\n",
+       "        <td>[[[174, 243, 194], [14, 219, 228]], [[86, 254, 177], [214, 92, 119]]]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[252, 59, 22], [136, 146, 86]], [[64, 209, 43], [85, 49, 181]]]</td>\n",
+       "        <td>[[[24, 120, 130], [256, 167, 172]], [[142, 93, 141], [165, 156, 239]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[81, 253, 127], [77, 53, 45]], [[64, 246, 59], [27, 219, 145]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[140, 103, 118], [4, 127, 142]], [[124, 1, 142], [35, 173, 28]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[58, 193, 28], [41, 201, 109]], [[38, 72, 186], [90, 116, 250]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[176, 21, 44], [65, 47, 184]], [[168, 165, 187], [39, 50, 55]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[192, 90, 212], [220, 218, 14]], [[157, 246, 55], [102, 99, 93]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[[[152, 28, 101], [195, 2, 220]], [[91, 128, 220], [189, 218, 81]]]</td>\n",
+       "        <td>[[[29, 183, 34], [23, 8, 210]], [[44, 51, 19], [91, 235, 187]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[166, 226, 50], [222, 9, 242]], [[56, 222, 206], [18, 236, 108]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[35, 210, 106], [127, 127, 134]], [[55, 162, 157], [62, 115, 201]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[134, 36, 93], [65, 36, 4]], [[35, 86, 225], [44, 73, 25]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[23, 42, 246], [130, 49, 24]], [[84, 155, 152], [212, 34, 206]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[191, 13, 233], [136, 126, 111]], [[173, 220, 176], [209, 223, 211]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[192, 255, 112], [217, 8, 134]], [[3, 254, 9], [53, 22, 93]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[174, 48, 241], [124, 166, 176]], [[136, 142, 56], [7, 253, 229]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[173, 181, 193], [127, 220, 130]], [[126, 76, 91], [135, 210, 94]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[219, 147, 155], [56, 99, 72]], [[104, 84, 196], [14, 4, 77]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[60, 83, 153], [33, 54, 70]], [[214, 247, 197], [179, 121, 67]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[212, 202, 209], [50, 78, 172]], [[196, 233, 227], [39, 49, 76]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[246, 89, 127], [66, 245, 187]], [[150, 142, 220], [203, 212, 178]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[153, 101, 60], [220, 100, 15]], [[166, 52, 65], [245, 224, 5]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[195, 44, 15], [15, 167, 4]], [[104, 38, 71], [94, 225, 220]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[189, 168, 192], [112, 107, 89]], [[213, 166, 54], [56, 181, 220]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[246, 208, 77], [251, 174, 16]], [[39, 189, 31], [206, 193, 135]]]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[8, 229, 214], [228, 209, 147]], [[140, 146, 3], [247, 235, 215]]]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[33, 16, 82], [252, 124, 72]], [[205, 201, 68], [123, 217, 107]]]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[[[248, 57, 249], [127, 46, 1]], [[100, 3, 229], [54, 150, 113]]]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[([[[152, 186, 35], [102, 145, 138]], [[40, 249, 108], [175, 207, 70]]], u'cat'),\n",
-       " ([[[234, 110, 251], [147, 18, 158]], [[55, 79, 14], [140, 50, 143]]], u'cat'),\n",
-       " ([[[179, 202, 20], [219, 198, 173]], [[149, 233, 18], [38, 115, 59]]], u'cat'),\n",
-       " ([[[223, 234, 239], [37, 253, 217]], [[147, 248, 108], [166, 150, 162]]], u'bird'),\n",
-       " ([[[164, 46, 39], [51, 130, 218]], [[253, 150, 181], [195, 66, 75]]], u'bird'),\n",
-       " ([[[85, 113, 32], [144, 145, 255]], [[122, 127, 36], [118, 88, 183]]], u'dog'),\n",
-       " ([[[195, 93, 4], [102, 81, 168]], [[148, 120, 219], [21, 82, 217]]], u'bird'),\n",
-       " ([[[8, 156, 237], [82, 72, 66]], [[196, 104, 210], [84, 103, 75]]], u'bird'),\n",
-       " ([[[139, 194, 43], [66, 48, 239]], [[159, 52, 84], [240, 220, 232]]], u'dog'),\n",
-       " ([[[183, 253, 187], [144, 168, 194]], [[44, 150, 21], [116, 216, 216]]], u'bird'),\n",
-       " ([[[170, 44, 68], [245, 256, 207]], [[183, 43, 17], [231, 25, 176]]], u'cat'),\n",
-       " ([[[110, 160, 246], [85, 9, 173]], [[82, 195, 61], [251, 134, 105]]], u'dog'),\n",
-       " ([[[154, 222, 104], [114, 186, 18]], [[159, 254, 7], [158, 205, 190]]], u'bird'),\n",
-       " ([[[222, 165, 227], [142, 191, 80]], [[46, 182, 165], [55, 99, 248]]], u'bird'),\n",
-       " ([[[161, 243, 128], [10, 131, 26]], [[232, 235, 141], [162, 253, 43]]], u'dog'),\n",
-       " ([[[4, 202, 109], [194, 147, 75]], [[103, 117, 217], [39, 197, 8]]], u'bird'),\n",
-       " ([[[107, 63, 64], [99, 57, 224]], [[86, 185, 234], [216, 212, 210]]], u'bird'),\n",
-       " ([[[96, 116, 192], [140, 21, 196]], [[85, 130, 135], [232, 206, 238]]], u'dog'),\n",
-       " ([[[167, 20, 35], [174, 241, 142]], [[237, 48, 241], [38, 16, 70]]], u'bird'),\n",
-       " ([[[251, 31, 179], [205, 226, 19]], [[65, 162, 159], [86, 103, 244]]], u'bird'),\n",
-       " ([[[237, 220, 166], [219, 58, 77]], [[239, 93, 251], [224, 235, 232]]], u'cat'),\n",
-       " ([[[219, 14, 33], [34, 237, 28]], [[64, 160, 232], [34, 180, 41]]], u'bird'),\n",
-       " ([[[83, 127, 43], [71, 87, 24]], [[35, 253, 243], [93, 74, 227]]], u'bird'),\n",
-       " ([[[69, 195, 165], [45, 212, 129]], [[59, 245, 162], [40, 16, 226]]], u'bird'),\n",
-       " ([[[248, 5, 124], [34, 201, 206]], [[161, 244, 21], [248, 13, 57]]], u'bird'),\n",
-       " ([[[0, 150, 63], [227, 80, 132]], [[166, 245, 176], [121, 118, 235]]], u'dog'),\n",
-       " ([[[104, 42, 37], [143, 227, 111]], [[96, 135, 172], [12, 207, 100]]], u'bird'),\n",
-       " ([[[221, 150, 126], [143, 129, 93]], [[92, 235, 60], [174, 100, 100]]], u'bird'),\n",
-       " ([[[216, 163, 35], [249, 33, 139]], [[35, 70, 26], [6, 181, 122]]], u'dog'),\n",
-       " ([[[97, 134, 93], [198, 94, 57]], [[92, 219, 200], [221, 56, 35]]], u'bird'),\n",
-       " ([[[116, 210, 44], [216, 129, 4]], [[123, 164, 253], [156, 47, 32]]], u'bird'),\n",
-       " ([[[73, 39, 151], [196, 180, 248]], [[74, 16, 190], [168, 74, 26]]], u'dog'),\n",
-       " ([[[18, 246, 187], [53, 190, 47]], [[7, 234, 8], [136, 238, 131]]], u'cat'),\n",
-       " ([[[235, 31, 91], [11, 1, 164]], [[49, 152, 103], [229, 144, 177]]], u'bird'),\n",
-       " ([[[78, 89, 63], [104, 220, 81]], [[94, 151, 134], [28, 199, 141]]], u'cat'),\n",
-       " ([[[206, 21, 244], [81, 65, 223]], [[112, 155, 234], [113, 63, 27]]], u'cat'),\n",
-       " ([[[166, 1, 152], [88, 246, 230]], [[176, 54, 78], [140, 135, 172]]], u'cat'),\n",
-       " ([[[13, 200, 234], [155, 207, 185]], [[176, 195, 10], [240, 162, 122]]], u'dog'),\n",
-       " ([[[140, 235, 202], [167, 244, 113]], [[168, 140, 200], [158, 114, 121]]], u'bird'),\n",
-       " ([[[192, 5, 91], [108, 41, 104]], [[52, 19, 3], [3, 204, 178]]], u'bird'),\n",
-       " ([[[214, 162, 103], [80, 46, 243]], [[60, 248, 154], [47, 105, 65]]], u'bird'),\n",
-       " ([[[49, 223, 45], [170, 179, 237]], [[175, 14, 89], [216, 118, 141]]], u'bird'),\n",
-       " ([[[121, 144, 183], [43, 86, 141]], [[205, 189, 221], [251, 176, 25]]], u'bird'),\n",
-       " ([[[74, 72, 92], [139, 3, 141]], [[106, 48, 55], [29, 30, 230]]], u'cat'),\n",
-       " ([[[119, 190, 161], [4, 168, 25]], [[148, 95, 68], [234, 236, 17]]], u'dog'),\n",
-       " ([[[201, 13, 87], [226, 256, 161]], [[42, 92, 44], [45, 233, 150]]], u'dog'),\n",
-       " ([[[33, 179, 122], [7, 222, 241]], [[196, 127, 246], [108, 152, 138]]], u'bird'),\n",
-       " ([[[220, 116, 183], [237, 27, 128]], [[250, 115, 98], [250, 19, 140]]], u'dog'),\n",
-       " ([[[64, 184, 64], [214, 21, 96]], [[137, 143, 103], [103, 129, 43]]], u'bird'),\n",
-       " ([[[118, 151, 126], [1, 99, 90]], [[117, 26, 71], [144, 154, 65]]], u'cat'),\n",
-       " ([[[252, 59, 22], [136, 146, 86]], [[64, 209, 43], [85, 49, 181]]], u'bird'),\n",
-       " ([[[152, 28, 101], [195, 2, 220]], [[91, 128, 220], [189, 218, 81]]], u'bird')]"
+       "[([[[17, 201, 110], [175, 136, 179]], [[102, 57, 24], [110, 199, 64]]], u'bird'),\n",
+       " ([[[205, 85, 56], [209, 11, 117]], [[86, 82, 41], [226, 192, 132]]], u'cat'),\n",
+       " ([[[209, 227, 160], [86, 88, 177]], [[31, 198, 96], [167, 122, 198]]], u'bird'),\n",
+       " ([[[146, 52, 167], [210, 33, 116]], [[38, 89, 69], [50, 207, 155]]], u'dog'),\n",
+       " ([[[247, 125, 68], [124, 196, 20]], [[95, 100, 107], [183, 21, 138]]], u'dog'),\n",
+       " ([[[117, 49, 248], [59, 18, 137]], [[110, 186, 91], [143, 46, 129]]], u'bird'),\n",
+       " ([[[115, 179, 183], [14, 54, 175]], [[138, 122, 42], [79, 142, 137]]], u'bird'),\n",
+       " ([[[249, 65, 200], [131, 191, 61]], [[180, 182, 119], [199, 63, 230]]], u'dog'),\n",
+       " ([[[154, 117, 174], [27, 94, 33]], [[206, 21, 46], [4, 196, 185]]], u'dog'),\n",
+       " ([[[238, 8, 12], [120, 187, 4]], [[184, 130, 135], [119, 191, 59]]], u'cat'),\n",
+       " ([[[55, 2, 109], [28, 130, 7]], [[146, 48, 34], [240, 81, 240]]], u'cat'),\n",
+       " ([[[128, 244, 200], [57, 113, 182]], [[64, 125, 46], [251, 129, 230]]], u'dog'),\n",
+       " ([[[8, 93, 61], [67, 139, 115]], [[69, 248, 144], [199, 255, 33]]], u'bird'),\n",
+       " ([[[33, 17, 73], [17, 21, 201]], [[5, 222, 1], [118, 148, 66]]], u'bird'),\n",
+       " ([[[194, 61, 116], [168, 187, 124]], [[6, 247, 192], [145, 106, 5]]], u'dog'),\n",
+       " ([[[250, 204, 135], [27, 196, 168]], [[44, 12, 185], [65, 213, 190]]], u'cat'),\n",
+       " ([[[215, 52, 179], [25, 39, 117]], [[86, 155, 29], [16, 24, 35]]], u'cat'),\n",
+       " ([[[215, 180, 113], [220, 61, 107]], [[168, 196, 134], [108, 108, 178]]], u'dog'),\n",
+       " ([[[38, 244, 77], [228, 19, 36]], [[24, 198, 60], [63, 59, 146]]], u'bird'),\n",
+       " ([[[89, 162, 242], [124, 169, 202]], [[48, 26, 166], [109, 134, 78]]], u'cat'),\n",
+       " ([[[12, 185, 157], [191, 49, 195]], [[178, 126, 167], [197, 162, 191]]], u'dog'),\n",
+       " ([[[222, 254, 199], [112, 217, 32]], [[18, 203, 156], [187, 148, 204]]], u'bird'),\n",
+       " ([[[58, 56, 91], [136, 105, 103]], [[65, 6, 38], [114, 201, 216]]], u'bird'),\n",
+       " ([[[111, 157, 147], [46, 41, 113]], [[44, 240, 226], [5, 15, 244]]], u'bird'),\n",
+       " ([[[171, 175, 100], [119, 132, 158]], [[175, 224, 37], [24, 71, 102]]], u'cat'),\n",
+       " ([[[174, 243, 194], [14, 219, 228]], [[86, 254, 177], [214, 92, 119]]], u'cat'),\n",
+       " ([[[24, 120, 130], [256, 167, 172]], [[142, 93, 141], [165, 156, 239]]], u'cat'),\n",
+       " ([[[81, 253, 127], [77, 53, 45]], [[64, 246, 59], [27, 219, 145]]], u'cat'),\n",
+       " ([[[140, 103, 118], [4, 127, 142]], [[124, 1, 142], [35, 173, 28]]], u'dog'),\n",
+       " ([[[58, 193, 28], [41, 201, 109]], [[38, 72, 186], [90, 116, 250]]], u'cat'),\n",
+       " ([[[176, 21, 44], [65, 47, 184]], [[168, 165, 187], [39, 50, 55]]], u'cat'),\n",
+       " ([[[192, 90, 212], [220, 218, 14]], [[157, 246, 55], [102, 99, 93]]], u'bird'),\n",
+       " ([[[29, 183, 34], [23, 8, 210]], [[44, 51, 19], [91, 235, 187]]], u'bird'),\n",
+       " ([[[166, 226, 50], [222, 9, 242]], [[56, 222, 206], [18, 236, 108]]], u'cat'),\n",
+       " ([[[35, 210, 106], [127, 127, 134]], [[55, 162, 157], [62, 115, 201]]], u'dog'),\n",
+       " ([[[134, 36, 93], [65, 36, 4]], [[35, 86, 225], [44, 73, 25]]], u'cat'),\n",
+       " ([[[23, 42, 246], [130, 49, 24]], [[84, 155, 152], [212, 34, 206]]], u'dog'),\n",
+       " ([[[191, 13, 233], [136, 126, 111]], [[173, 220, 176], [209, 223, 211]]], u'cat'),\n",
+       " ([[[192, 255, 112], [217, 8, 134]], [[3, 254, 9], [53, 22, 93]]], u'bird'),\n",
+       " ([[[174, 48, 241], [124, 166, 176]], [[136, 142, 56], [7, 253, 229]]], u'bird'),\n",
+       " ([[[173, 181, 193], [127, 220, 130]], [[126, 76, 91], [135, 210, 94]]], u'dog'),\n",
+       " ([[[219, 147, 155], [56, 99, 72]], [[104, 84, 196], [14, 4, 77]]], u'dog'),\n",
+       " ([[[60, 83, 153], [33, 54, 70]], [[214, 247, 197], [179, 121, 67]]], u'bird'),\n",
+       " ([[[212, 202, 209], [50, 78, 172]], [[196, 233, 227], [39, 49, 76]]], u'dog'),\n",
+       " ([[[246, 89, 127], [66, 245, 187]], [[150, 142, 220], [203, 212, 178]]], u'bird'),\n",
+       " ([[[153, 101, 60], [220, 100, 15]], [[166, 52, 65], [245, 224, 5]]], u'bird'),\n",
+       " ([[[195, 44, 15], [15, 167, 4]], [[104, 38, 71], [94, 225, 220]]], u'bird'),\n",
+       " ([[[189, 168, 192], [112, 107, 89]], [[213, 166, 54], [56, 181, 220]]], u'dog'),\n",
+       " ([[[246, 208, 77], [251, 174, 16]], [[39, 189, 31], [206, 193, 135]]], u'bird'),\n",
+       " ([[[8, 229, 214], [228, 209, 147]], [[140, 146, 3], [247, 235, 215]]], u'dog'),\n",
+       " ([[[33, 16, 82], [252, 124, 72]], [[205, 201, 68], [123, 217, 107]]], u'cat'),\n",
+       " ([[[248, 57, 249], [127, 46, 1]], [[100, 3, 229], [54, 150, 113]]], u'bird')]"
       ]
      },
-     "execution_count": 4,
+     "execution_count": 6,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -463,7 +438,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 5,
+   "execution_count": 8,
    "metadata": {},
    "outputs": [
     {
@@ -480,8 +455,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -500,7 +475,7 @@
        "[([26, 2, 2, 3], [26, 3], 0), ([26, 2, 2, 3], [26, 3], 1)]"
       ]
      },
-     "execution_count": 5,
+     "execution_count": 8,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -517,7 +492,7 @@
     "                                        255                   -- Normalizing constant\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -531,7 +506,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
+   "execution_count": 9,
    "metadata": {},
    "outputs": [
     {
@@ -551,7 +526,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -561,23 +536,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 26, 255.0, 3, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 26, 255.0, [3], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 6,
+     "execution_count": 9,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -599,7 +574,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 10,
    "metadata": {},
    "outputs": [
     {
@@ -616,8 +591,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -636,7 +611,7 @@
        "[([26, 2, 2, 3], [26, 3], 0), ([26, 2, 2, 3], [26, 3], 1)]"
       ]
      },
-     "execution_count": 7,
+     "execution_count": 10,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -644,6 +619,7 @@
    "source": [
     "%%sql\n",
     "DROP TABLE IF EXISTS val_image_data_packed, val_image_data_packed_summary;\n",
+    "\n",
     "SELECT madlib.validation_preprocessor_dl(\n",
     "      'image_data',             -- Source table\n",
     "      'val_image_data_packed',  -- Output table\n",
@@ -652,7 +628,8 @@
     "      'image_data_packed',      -- From training preprocessor step\n",
     "      NULL                      -- Buffer size\n",
     "      ); \n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
+    "\n",
+    "SELECT rgb_shape, species_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -664,7 +641,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 8,
+   "execution_count": 11,
    "metadata": {},
    "outputs": [
     {
@@ -684,7 +661,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -694,23 +671,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>val_image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'val_image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 26, 255.0, 3, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'val_image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 26, 255.0, [3], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 8,
+     "execution_count": 11,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -731,7 +708,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 9,
+   "execution_count": 12,
    "metadata": {},
    "outputs": [
     {
@@ -752,271 +729,271 @@
        "        <th>species</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[19, 126, 250, 219, 119, 255, 86, 152, 200, 36, 57, 188]</td>\n",
+       "        <td>[168, 228, 110, 3, 51, 104, 192, 23, 120, 249, 96, 99]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[20, 145, 109, 135, 149, 100, 39, 66, 124, 102, 77, 140]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[125, 32, 244, 23, 201, 156, 251, 55, 159, 47, 160, 95]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[49, 201, 114, 38, 201, 8, 101, 172, 88, 233, 82, 78]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[203, 196, 132, 57, 220, 151, 183, 214, 113, 46, 213, 200]</td>\n",
+       "        <td>[24, 88, 166, 123, 193, 186, 12, 46, 65, 161, 145, 104]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[157, 236, 255, 90, 38, 48, 35, 152, 86, 236, 160, 187]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[248, 164, 234, 70, 61, 181, 10, 193, 238, 229, 88, 165]</td>\n",
+       "        <td>[14, 206, 47, 154, 85, 172, 186, 73, 196, 131, 229, 191]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[201, 210, 145, 145, 152, 46, 125, 151, 135, 163, 199, 170]</td>\n",
+       "        <td>[131, 238, 90, 227, 51, 114, 59, 217, 237, 252, 147, 248]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[29, 150, 219, 216, 46, 211, 124, 24, 25, 186, 205, 35]</td>\n",
+       "        <td>[211, 153, 187, 59, 123, 200, 10, 171, 98, 95, 87, 28]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[187, 8, 211, 95, 196, 156, 50, 84, 45, 202, 130, 170]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[9, 77, 40, 179, 136, 69, 74, 98, 29, 120, 53, 153]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[78, 83, 93, 113, 206, 23, 121, 160, 119, 61, 60, 168]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[105, 114, 19, 19, 211, 28, 96, 251, 208, 232, 64, 25]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[93, 145, 128, 246, 33, 206, 73, 126, 63, 22, 150, 184]</td>\n",
+       "        <td>[26, 159, 140, 217, 89, 15, 199, 179, 242, 250, 37, 45]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[12, 245, 243, 181, 134, 92, 39, 153, 112, 250, 181, 208]</td>\n",
+       "        <td>[18, 41, 102, 10, 82, 57, 163, 13, 116, 30, 213, 126]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[133, 184, 53, 158, 3, 145, 47, 130, 135, 81, 80, 208]</td>\n",
+       "        <td>[56, 221, 31, 84, 132, 58, 243, 16, 19, 76, 31, 218]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[143, 230, 101, 71, 156, 113, 61, 143, 37, 195, 235, 76]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[91, 70, 17, 43, 59, 150, 227, 111, 53, 229, 0, 100]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[136, 181, 184, 87, 132, 71, 61, 232, 143, 218, 89, 203]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[126, 142, 84, 203, 234, 175, 17, 251, 217, 75, 145, 188]</td>\n",
+       "        <td>[17, 212, 36, 62, 167, 54, 103, 13, 64, 185, 70, 227]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[198, 162, 187, 42, 9, 67, 223, 193, 154, 99, 9, 215]</td>\n",
-       "        <td>cat</td>\n",
+       "        <td>[186, 1, 155, 56, 201, 211, 21, 233, 38, 153, 34, 25]</td>\n",
+       "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[151, 177, 164, 98, 25, 35, 240, 109, 237, 218, 28, 254]</td>\n",
+       "        <td>[53, 101, 200, 15, 101, 217, 227, 137, 23, 138, 191, 126]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[255, 54, 220, 226, 252, 150, 227, 151, 207, 172, 105, 227]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[144, 124, 183, 169, 37, 237, 14, 237, 252, 115, 198, 222]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[246, 73, 102, 178, 4, 45, 84, 191, 87, 93, 2, 54]</td>\n",
+       "        <td>[222, 104, 188, 92, 254, 187, 146, 219, 157, 142, 113, 128]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[156, 153, 39, 115, 228, 190, 35, 136, 32, 61, 171, 16]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[152, 234, 198, 149, 191, 188, 222, 37, 110, 226, 82, 194]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[169, 31, 163, 222, 61, 62, 119, 100, 177, 91, 34, 213]</td>\n",
+       "        <td>[64, 44, 142, 35, 193, 30, 159, 120, 199, 196, 101, 213]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[67, 17, 141, 83, 188, 37, 61, 130, 187, 252, 62, 153]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[172, 123, 115, 110, 28, 28, 140, 191, 250, 202, 253, 113]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[225, 113, 99, 228, 109, 158, 250, 245, 47, 79, 52, 1]</td>\n",
+       "        <td>[96, 72, 120, 63, 69, 86, 167, 0, 177, 165, 187, 67]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[137, 50, 48, 110, 202, 76, 211, 142, 78, 174, 232, 206]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[166, 168, 219, 125, 201, 188, 238, 44, 160, 92, 202, 153]</td>\n",
+       "        <td>[88, 210, 241, 216, 246, 48, 4, 132, 83, 197, 162, 242]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[249, 233, 133, 249, 100, 14, 43, 147, 124, 246, 223, 78]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[45, 253, 108, 251, 135, 18, 163, 98, 143, 108, 30, 126]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[190, 217, 97, 87, 41, 90, 64, 174, 84, 164, 188, 127]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[56, 117, 22, 134, 249, 67, 130, 101, 62, 9, 119, 225]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[6, 78, 138, 132, 230, 72, 93, 71, 159, 134, 161, 223]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[245, 131, 240, 116, 186, 40, 233, 209, 174, 226, 20, 48]</td>\n",
-       "        <td>cat</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[82, 57, 189, 52, 165, 195, 129, 46, 71, 103, 118, 163]</td>\n",
+       "        <td>[105, 182, 162, 62, 104, 2, 134, 223, 65, 203, 53, 231]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[21, 41, 79, 244, 93, 68, 120, 78, 184, 50, 117, 161]</td>\n",
+       "        <td>[230, 140, 134, 42, 12, 223, 251, 252, 183, 241, 44, 188]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[127, 129, 24, 113, 190, 129, 40, 96, 191, 143, 98, 69]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[162, 16, 163, 137, 219, 137, 21, 97, 179, 33, 64, 174]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[35, 131, 23, 83, 201, 105, 140, 134, 157, 48, 73, 30]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[144, 133, 213, 51, 51, 234, 93, 130, 222, 186, 198, 86]</td>\n",
+       "        <td>[247, 159, 74, 179, 21, 201, 51, 45, 58, 241, 175, 98]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[126, 136, 125, 31, 139, 160, 161, 162, 242, 106, 11, 126]</td>\n",
+       "        <td>[110, 241, 179, 179, 96, 85, 195, 3, 222, 158, 140, 244]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[168, 174, 58, 198, 13, 202, 75, 226, 254, 126, 204, 90]</td>\n",
+       "        <td>[63, 21, 63, 237, 50, 54, 140, 124, 233, 162, 69, 28]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[170, 20, 197, 1, 28, 67, 137, 153, 97, 20, 57, 3]</td>\n",
-       "        <td>bird</td>\n",
+       "        <td>[94, 111, 234, 231, 203, 73, 118, 97, 57, 254, 209, 131]</td>\n",
+       "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[43, 109, 193, 169, 94, 105, 88, 152, 46, 101, 98, 121]</td>\n",
+       "        <td>[246, 73, 151, 78, 201, 43, 59, 1, 215, 155, 138, 63]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[46, 186, 18, 158, 254, 111, 13, 232, 86, 216, 49, 204]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[95, 247, 19, 186, 247, 189, 206, 188, 190, 234, 254, 70]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[96, 90, 188, 98, 16, 231, 207, 209, 145, 45, 58, 232]</td>\n",
-       "        <td>bird</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[104, 77, 39, 226, 148, 134, 217, 166, 64, 207, 99, 14]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[33, 248, 137, 103, 124, 233, 194, 56, 75, 210, 32, 27]</td>\n",
-       "        <td>dog</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[176, 72, 221, 152, 12, 70, 229, 51, 39, 121, 185, 0]</td>\n",
+       "        <td>[106, 202, 9, 238, 104, 256, 55, 255, 78, 0, 42, 137]</td>\n",
        "        <td>cat</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[249, 207, 131, 7, 90, 164, 255, 228, 11, 123, 205, 205]</td>\n",
+       "        <td>[1, 35, 139, 64, 121, 185, 250, 139, 87, 248, 250, 100]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[25, 160, 211, 51, 67, 131, 123, 33, 28, 135, 102, 1]</td>\n",
+       "        <td>[81, 59, 17, 29, 116, 124, 231, 125, 105, 79, 124, 160]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[202, 160, 119, 83, 161, 120, 118, 44, 183, 239, 230, 177]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[61, 169, 117, 160, 136, 197, 220, 153, 226, 79, 21, 201]</td>\n",
        "        <td>bird</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[142, 122, 115, 142, 154, 108, 93, 29, 115, 184, 193, 114]</td>\n",
+       "        <td>[126, 23, 73, 30, 100, 19, 191, 219, 102, 96, 83, 220]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[204, 237, 105, 153, 161, 129, 57, 116, 181, 124, 247, 47]</td>\n",
+       "        <td>[10, 203, 113, 187, 70, 174, 99, 186, 78, 235, 128, 42]</td>\n",
        "        <td>dog</td>\n",
        "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[98, 122, 154, 42, 70, 24, 66, 143, 54, 166, 161, 245]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[7, 84, 211, 227, 224, 221, 174, 82, 152, 244, 255, 251]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[78, 230, 46, 120, 106, 144, 241, 4, 186, 55, 28, 252]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[82, 162, 103, 71, 35, 110, 156, 246, 81, 124, 211, 255]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[106, 243, 205, 101, 161, 26, 75, 207, 146, 181, 94, 132]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[24, 187, 213, 20, 129, 39, 182, 232, 110, 217, 86, 10]</td>\n",
+       "        <td>bird</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[168, 134, 161, 167, 83, 12, 154, 32, 113, 58, 58, 188]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[205, 113, 103, 80, 42, 128, 11, 255, 148, 140, 39, 74]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[149, 34, 203, 159, 241, 114, 37, 146, 25, 120, 158, 179]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 237, 210, 202, 246, 159, 59, 94, 239, 101, 221, 250]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[113, 134, 139, 187, 250, 32, 222, 197, 192, 206, 55, 229]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[81, 93, 255, 4, 244, 13, 241, 198, 215, 231, 101, 18]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[84, 120, 34, 78, 220, 147, 212, 103, 79, 206, 136, 44]</td>\n",
+       "        <td>dog</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[71, 251, 203, 44, 91, 28, 136, 90, 31, 124, 103, 16]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[62, 248, 167, 81, 60, 251, 200, 95, 72, 164, 242, 28]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[65, 235, 147, 109, 126, 219, 103, 73, 6, 195, 101, 143]</td>\n",
+       "        <td>cat</td>\n",
+       "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[([19, 126, 250, 219, 119, 255, 86, 152, 200, 36, 57, 188], u'cat'),\n",
-       " ([49, 201, 114, 38, 201, 8, 101, 172, 88, 233, 82, 78], u'dog'),\n",
-       " ([203, 196, 132, 57, 220, 151, 183, 214, 113, 46, 213, 200], u'bird'),\n",
-       " ([157, 236, 255, 90, 38, 48, 35, 152, 86, 236, 160, 187], u'dog'),\n",
-       " ([248, 164, 234, 70, 61, 181, 10, 193, 238, 229, 88, 165], u'bird'),\n",
-       " ([201, 210, 145, 145, 152, 46, 125, 151, 135, 163, 199, 170], u'cat'),\n",
-       " ([29, 150, 219, 216, 46, 211, 124, 24, 25, 186, 205, 35], u'dog'),\n",
-       " ([187, 8, 211, 95, 196, 156, 50, 84, 45, 202, 130, 170], u'dog'),\n",
-       " ([9, 77, 40, 179, 136, 69, 74, 98, 29, 120, 53, 153], u'dog'),\n",
-       " ([78, 83, 93, 113, 206, 23, 121, 160, 119, 61, 60, 168], u'dog'),\n",
-       " ([105, 114, 19, 19, 211, 28, 96, 251, 208, 232, 64, 25], u'cat'),\n",
-       " ([93, 145, 128, 246, 33, 206, 73, 126, 63, 22, 150, 184], u'bird'),\n",
-       " ([12, 245, 243, 181, 134, 92, 39, 153, 112, 250, 181, 208], u'bird'),\n",
-       " ([133, 184, 53, 158, 3, 145, 47, 130, 135, 81, 80, 208], u'bird'),\n",
-       " ([143, 230, 101, 71, 156, 113, 61, 143, 37, 195, 235, 76], u'dog'),\n",
-       " ([91, 70, 17, 43, 59, 150, 227, 111, 53, 229, 0, 100], u'dog'),\n",
-       " ([136, 181, 184, 87, 132, 71, 61, 232, 143, 218, 89, 203], u'dog'),\n",
-       " ([126, 142, 84, 203, 234, 175, 17, 251, 217, 75, 145, 188], u'bird'),\n",
-       " ([198, 162, 187, 42, 9, 67, 223, 193, 154, 99, 9, 215], u'cat'),\n",
-       " ([151, 177, 164, 98, 25, 35, 240, 109, 237, 218, 28, 254], u'bird'),\n",
-       " ([246, 73, 102, 178, 4, 45, 84, 191, 87, 93, 2, 54], u'cat'),\n",
-       " ([156, 153, 39, 115, 228, 190, 35, 136, 32, 61, 171, 16], u'dog'),\n",
-       " ([152, 234, 198, 149, 191, 188, 222, 37, 110, 226, 82, 194], u'dog'),\n",
-       " ([169, 31, 163, 222, 61, 62, 119, 100, 177, 91, 34, 213], u'bird'),\n",
-       " ([67, 17, 141, 83, 188, 37, 61, 130, 187, 252, 62, 153], u'cat'),\n",
-       " ([172, 123, 115, 110, 28, 28, 140, 191, 250, 202, 253, 113], u'cat'),\n",
-       " ([225, 113, 99, 228, 109, 158, 250, 245, 47, 79, 52, 1], u'dog'),\n",
-       " ([137, 50, 48, 110, 202, 76, 211, 142, 78, 174, 232, 206], u'dog'),\n",
-       " ([166, 168, 219, 125, 201, 188, 238, 44, 160, 92, 202, 153], u'cat'),\n",
-       " ([249, 233, 133, 249, 100, 14, 43, 147, 124, 246, 223, 78], u'dog'),\n",
-       " ([45, 253, 108, 251, 135, 18, 163, 98, 143, 108, 30, 126], u'dog'),\n",
-       " ([190, 217, 97, 87, 41, 90, 64, 174, 84, 164, 188, 127], u'cat'),\n",
-       " ([56, 117, 22, 134, 249, 67, 130, 101, 62, 9, 119, 225], u'dog'),\n",
-       " ([6, 78, 138, 132, 230, 72, 93, 71, 159, 134, 161, 223], u'cat'),\n",
-       " ([245, 131, 240, 116, 186, 40, 233, 209, 174, 226, 20, 48], u'cat'),\n",
-       " ([82, 57, 189, 52, 165, 195, 129, 46, 71, 103, 118, 163], u'bird'),\n",
-       " ([21, 41, 79, 244, 93, 68, 120, 78, 184, 50, 117, 161], u'cat'),\n",
-       " ([35, 131, 23, 83, 201, 105, 140, 134, 157, 48, 73, 30], u'dog'),\n",
-       " ([144, 133, 213, 51, 51, 234, 93, 130, 222, 186, 198, 86], u'cat'),\n",
-       " ([126, 136, 125, 31, 139, 160, 161, 162, 242, 106, 11, 126], u'bird'),\n",
-       " ([168, 174, 58, 198, 13, 202, 75, 226, 254, 126, 204, 90], u'bird'),\n",
-       " ([170, 20, 197, 1, 28, 67, 137, 153, 97, 20, 57, 3], u'bird'),\n",
-       " ([43, 109, 193, 169, 94, 105, 88, 152, 46, 101, 98, 121], u'cat'),\n",
-       " ([95, 247, 19, 186, 247, 189, 206, 188, 190, 234, 254, 70], u'dog'),\n",
-       " ([96, 90, 188, 98, 16, 231, 207, 209, 145, 45, 58, 232], u'bird'),\n",
-       " ([104, 77, 39, 226, 148, 134, 217, 166, 64, 207, 99, 14], u'dog'),\n",
-       " ([33, 248, 137, 103, 124, 233, 194, 56, 75, 210, 32, 27], u'dog'),\n",
-       " ([176, 72, 221, 152, 12, 70, 229, 51, 39, 121, 185, 0], u'cat'),\n",
-       " ([249, 207, 131, 7, 90, 164, 255, 228, 11, 123, 205, 205], u'bird'),\n",
-       " ([25, 160, 211, 51, 67, 131, 123, 33, 28, 135, 102, 1], u'bird'),\n",
-       " ([142, 122, 115, 142, 154, 108, 93, 29, 115, 184, 193, 114], u'dog'),\n",
-       " ([204, 237, 105, 153, 161, 129, 57, 116, 181, 124, 247, 47], u'dog')]"
+       "[([168, 228, 110, 3, 51, 104, 192, 23, 120, 249, 96, 99], u'dog'),\n",
+       " ([20, 145, 109, 135, 149, 100, 39, 66, 124, 102, 77, 140], u'dog'),\n",
+       " ([125, 32, 244, 23, 201, 156, 251, 55, 159, 47, 160, 95], u'cat'),\n",
+       " ([24, 88, 166, 123, 193, 186, 12, 46, 65, 161, 145, 104], u'bird'),\n",
+       " ([14, 206, 47, 154, 85, 172, 186, 73, 196, 131, 229, 191], u'bird'),\n",
+       " ([131, 238, 90, 227, 51, 114, 59, 217, 237, 252, 147, 248], u'cat'),\n",
+       " ([211, 153, 187, 59, 123, 200, 10, 171, 98, 95, 87, 28], u'dog'),\n",
+       " ([26, 159, 140, 217, 89, 15, 199, 179, 242, 250, 37, 45], u'bird'),\n",
+       " ([18, 41, 102, 10, 82, 57, 163, 13, 116, 30, 213, 126], u'bird'),\n",
+       " ([56, 221, 31, 84, 132, 58, 243, 16, 19, 76, 31, 218], u'bird'),\n",
+       " ([17, 212, 36, 62, 167, 54, 103, 13, 64, 185, 70, 227], u'bird'),\n",
+       " ([186, 1, 155, 56, 201, 211, 21, 233, 38, 153, 34, 25], u'dog'),\n",
+       " ([53, 101, 200, 15, 101, 217, 227, 137, 23, 138, 191, 126], u'dog'),\n",
+       " ([255, 54, 220, 226, 252, 150, 227, 151, 207, 172, 105, 227], u'dog'),\n",
+       " ([144, 124, 183, 169, 37, 237, 14, 237, 252, 115, 198, 222], u'bird'),\n",
+       " ([222, 104, 188, 92, 254, 187, 146, 219, 157, 142, 113, 128], u'cat'),\n",
+       " ([64, 44, 142, 35, 193, 30, 159, 120, 199, 196, 101, 213], u'bird'),\n",
+       " ([96, 72, 120, 63, 69, 86, 167, 0, 177, 165, 187, 67], u'dog'),\n",
+       " ([88, 210, 241, 216, 246, 48, 4, 132, 83, 197, 162, 242], u'cat'),\n",
+       " ([105, 182, 162, 62, 104, 2, 134, 223, 65, 203, 53, 231], u'bird'),\n",
+       " ([230, 140, 134, 42, 12, 223, 251, 252, 183, 241, 44, 188], u'dog'),\n",
+       " ([127, 129, 24, 113, 190, 129, 40, 96, 191, 143, 98, 69], u'dog'),\n",
+       " ([162, 16, 163, 137, 219, 137, 21, 97, 179, 33, 64, 174], u'cat'),\n",
+       " ([247, 159, 74, 179, 21, 201, 51, 45, 58, 241, 175, 98], u'cat'),\n",
+       " ([110, 241, 179, 179, 96, 85, 195, 3, 222, 158, 140, 244], u'bird'),\n",
+       " ([63, 21, 63, 237, 50, 54, 140, 124, 233, 162, 69, 28], u'bird'),\n",
+       " ([94, 111, 234, 231, 203, 73, 118, 97, 57, 254, 209, 131], u'dog'),\n",
+       " ([246, 73, 151, 78, 201, 43, 59, 1, 215, 155, 138, 63], u'dog'),\n",
+       " ([46, 186, 18, 158, 254, 111, 13, 232, 86, 216, 49, 204], u'cat'),\n",
+       " ([106, 202, 9, 238, 104, 256, 55, 255, 78, 0, 42, 137], u'cat'),\n",
+       " ([1, 35, 139, 64, 121, 185, 250, 139, 87, 248, 250, 100], u'bird'),\n",
+       " ([81, 59, 17, 29, 116, 124, 231, 125, 105, 79, 124, 160], u'cat'),\n",
+       " ([202, 160, 119, 83, 161, 120, 118, 44, 183, 239, 230, 177], u'dog'),\n",
+       " ([61, 169, 117, 160, 136, 197, 220, 153, 226, 79, 21, 201], u'bird'),\n",
+       " ([126, 23, 73, 30, 100, 19, 191, 219, 102, 96, 83, 220], u'dog'),\n",
+       " ([10, 203, 113, 187, 70, 174, 99, 186, 78, 235, 128, 42], u'dog'),\n",
+       " ([98, 122, 154, 42, 70, 24, 66, 143, 54, 166, 161, 245], u'dog'),\n",
+       " ([7, 84, 211, 227, 224, 221, 174, 82, 152, 244, 255, 251], u'bird'),\n",
+       " ([78, 230, 46, 120, 106, 144, 241, 4, 186, 55, 28, 252], u'bird'),\n",
+       " ([82, 162, 103, 71, 35, 110, 156, 246, 81, 124, 211, 255], u'bird'),\n",
+       " ([106, 243, 205, 101, 161, 26, 75, 207, 146, 181, 94, 132], u'bird'),\n",
+       " ([24, 187, 213, 20, 129, 39, 182, 232, 110, 217, 86, 10], u'bird'),\n",
+       " ([168, 134, 161, 167, 83, 12, 154, 32, 113, 58, 58, 188], u'cat'),\n",
+       " ([205, 113, 103, 80, 42, 128, 11, 255, 148, 140, 39, 74], u'dog'),\n",
+       " ([149, 34, 203, 159, 241, 114, 37, 146, 25, 120, 158, 179], u'dog'),\n",
+       " ([15, 237, 210, 202, 246, 159, 59, 94, 239, 101, 221, 250], u'dog'),\n",
+       " ([113, 134, 139, 187, 250, 32, 222, 197, 192, 206, 55, 229], u'dog'),\n",
+       " ([81, 93, 255, 4, 244, 13, 241, 198, 215, 231, 101, 18], u'cat'),\n",
+       " ([84, 120, 34, 78, 220, 147, 212, 103, 79, 206, 136, 44], u'dog'),\n",
+       " ([71, 251, 203, 44, 91, 28, 136, 90, 31, 124, 103, 16], u'cat'),\n",
+       " ([62, 248, 167, 81, 60, 251, 200, 95, 72, 164, 242, 28], u'cat'),\n",
+       " ([65, 235, 147, 109, 126, 219, 103, 73, 6, 195, 101, 143], u'cat')]"
       ]
      },
-     "execution_count": 9,
+     "execution_count": 12,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1058,7 +1035,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 10,
+   "execution_count": 13,
    "metadata": {},
    "outputs": [
     {
@@ -1075,8 +1052,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1095,7 +1072,7 @@
        "[([26, 12], [26, 3], 0), ([26, 12], [26, 3], 1)]"
       ]
      },
-     "execution_count": 10,
+     "execution_count": 13,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1112,7 +1089,7 @@
     "                                        255                   -- Normalizing constant\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -1127,7 +1104,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 11,
+   "execution_count": 14,
    "metadata": {},
    "outputs": [
     {
@@ -1144,8 +1121,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1164,7 +1141,7 @@
        "[([26, 12], [26, 3], 0), ([26, 12], [26, 3], 1)]"
       ]
      },
-     "execution_count": 11,
+     "execution_count": 14,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1182,7 +1159,7 @@
     "    NULL                      -- Buffer size\n",
     "    );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM val_image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -1197,7 +1174,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 12,
+   "execution_count": 15,
    "metadata": {},
    "outputs": [
     {
@@ -1214,13 +1191,13 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[8, 12]</td>\n",
-       "        <td>[8, 3]</td>\n",
+       "        <td>[9, 12]</td>\n",
+       "        <td>[9, 3]</td>\n",
        "        <td>0</td>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1244,22 +1221,22 @@
        "        <td>4</td>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>[8, 12]</td>\n",
-       "        <td>[8, 3]</td>\n",
+       "        <td>[7, 12]</td>\n",
+       "        <td>[7, 3]</td>\n",
        "        <td>5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[([8, 12], [8, 3], 0),\n",
+       "[([9, 12], [9, 3], 0),\n",
        " ([9, 12], [9, 3], 1),\n",
        " ([9, 12], [9, 3], 2),\n",
        " ([9, 12], [9, 3], 3),\n",
        " ([9, 12], [9, 3], 4),\n",
-       " ([8, 12], [8, 3], 5)]"
+       " ([7, 12], [7, 3], 5)]"
       ]
      },
-     "execution_count": 12,
+     "execution_count": 15,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1276,7 +1253,7 @@
     "                                        255                   -- Normalizing constant\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
@@ -1288,7 +1265,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 13,
+   "execution_count": 16,
    "metadata": {},
    "outputs": [
     {
@@ -1308,7 +1285,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -1318,23 +1295,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
-       "        <td>10</td>\n",
+       "        <td>9</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 10, 255.0, 3, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 9, 255.0, [3], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 13,
+     "execution_count": 16,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1356,7 +1333,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 14,
+   "execution_count": 18,
    "metadata": {},
    "outputs": [
     {
@@ -1373,8 +1350,8 @@
       "text/html": [
        "<table>\n",
        "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
+       "        <th>rgb_shape</th>\n",
+       "        <th>species_shape</th>\n",
        "        <th>buffer_id</th>\n",
        "    </tr>\n",
        "    <tr>\n",
@@ -1393,7 +1370,7 @@
        "[([26, 12], [26, 5], 0), ([26, 12], [26, 5], 1)]"
       ]
      },
-     "execution_count": 14,
+     "execution_count": 18,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1408,15 +1385,15 @@
     "                                        'rgb',                -- Independent variable\n",
     "                                        NULL,                 -- Buffer size\n",
     "                                        255,                  -- Normalizing constant\n",
-    "                                        5                     -- Number of desired class values\n",
+    "                                        ARRAY[5]              -- Number of desired class values\n",
     "                                        );\n",
     "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
+    "SELECT rgb_shape, species_shape, buffer_id FROM image_data_packed ORDER BY buffer_id;"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 15,
+   "execution_count": 19,
    "metadata": {},
    "outputs": [
     {
@@ -1436,7 +1413,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -1446,23 +1423,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog', None, None]</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>5</td>\n",
+       "        <td>[5]</td>\n",
        "        <td>all_segments</td>\n",
        "        <td>all_segments</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog', None, None], 26, 255.0, 5, 'all_segments', 'all_segments')]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog', None, None], 26, 255.0, [5], 'all_segments', 'all_segments')]"
       ]
      },
-     "execution_count": 15,
+     "execution_count": 19,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1513,7 +1490,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 17,
+   "execution_count": 20,
    "metadata": {},
    "outputs": [
     {
@@ -1538,7 +1515,7 @@
        "        <th>dependent_varname</th>\n",
        "        <th>independent_varname</th>\n",
        "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
+       "        <th>species_class_values</th>\n",
        "        <th>buffer_size</th>\n",
        "        <th>normalizing_const</th>\n",
        "        <th>num_classes</th>\n",
@@ -1548,23 +1525,23 @@
        "    <tr>\n",
        "        <td>image_data</td>\n",
        "        <td>image_data_packed</td>\n",
-       "        <td>species</td>\n",
-       "        <td>rgb</td>\n",
-       "        <td>text</td>\n",
+       "        <td>[u'species']</td>\n",
+       "        <td>[u'rgb']</td>\n",
+       "        <td>[u'text']</td>\n",
        "        <td>[u'bird', u'cat', u'dog']</td>\n",
        "        <td>26</td>\n",
        "        <td>255.0</td>\n",
-       "        <td>3</td>\n",
+       "        <td>[3]</td>\n",
        "        <td>[2, 3]</td>\n",
        "        <td>[0, 1]</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'image_data', u'image_data_packed', u'species', u'rgb', u'text', [u'bird', u'cat', u'dog'], 26, 255.0, 3, [2, 3], [0, 1])]"
+       "[(u'image_data', u'image_data_packed', [u'species'], [u'rgb'], [u'text'], [u'bird', u'cat', u'dog'], 26, 255.0, [3], [2, 3], [0, 1])]"
       ]
      },
-     "execution_count": 17,
+     "execution_count": 20,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -1610,7 +1587,7 @@
    "name": "python",
    "nbconvert_exporter": "python",
    "pygments_lexer": "ipython2",
-   "version": "2.7.10"
+   "version": "2.7.16"
   }
  },
  "nbformat": 4,
diff --git a/community-artifacts/Deep-learning/Train-multiple-models/.ipynb_checkpoints/MADlib-Keras-model-selection-MLP-v1-checkpoint.ipynb b/community-artifacts/Deep-learning/Train-multiple-models/.ipynb_checkpoints/MADlib-Keras-model-selection-MLP-v1-checkpoint.ipynb
new file mode 100644
index 0000000..4ae9eae
--- /dev/null
+++ b/community-artifacts/Deep-learning/Train-multiple-models/.ipynb_checkpoints/MADlib-Keras-model-selection-MLP-v1-checkpoint.ipynb
@@ -0,0 +1,6279 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Model Selection for Multilayer Perceptron Using Keras and MADlib\n",
+    "\n",
+    "E2E classification example using MADlib calling a Keras MLP for different hyperparameters and model architectures.\n",
+    "\n",
+    "Deep learning works best on very large datasets, but that is not convenient for a quick introduction to the syntax.  So in this workbook we use the well known iris data set from https://archive.ics.uci.edu/ml/datasets/iris to help get you started.  It is similar to the example in user docs http://madlib.apache.org/docs/latest/index.html\n",
+    "\n",
+    "For more realistic examples please refer to the deep learning notebooks at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#class\">Classification</a>\n",
+    "\n",
+    "* <a href=\"#create_input_data\">1. Create input data</a>\n",
+    "\n",
+    "* <a href=\"#pp\">2. Call preprocessor for deep learning</a>\n",
+    "\n",
+    "* <a href=\"#load\">3. Define and load model architecture</a>\n",
+    "\n",
+    "* <a href=\"#def_mst\">4. Define and load model selection tuples</a>\n",
+    "\n",
+    "* <a href=\"#train\">5. Train</a>\n",
+    "\n",
+    "* <a href=\"#eval\">6. Evaluate</a>\n",
+    "\n",
+    "* <a href=\"#pred\">7. Predict</a>\n",
+    "\n",
+    "<a href=\"#class2\">Classification with Other Parameters</a>\n",
+    "\n",
+    "* <a href=\"#val_dataset\">1. Validation dataset</a>\n",
+    "\n",
+    "* <a href=\"#pred_prob\">2. Predict probabilities</a>\n",
+    "\n",
+    "* <a href=\"#warm_start\">3. Warm start</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class\"></a>\n",
+    "# Classification\n",
+    "\n",
+    "<a id=\"create_input_data\"></a>\n",
+    "# 1.  Create input data\n",
+    "\n",
+    "Load iris data set."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "150 rows affected.\n",
+      "150 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>attributes</th>\n",
+       "        <th>class_text</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>[Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>21</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>22</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>25</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>35</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>36</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>37</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>39</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>40</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>42</td>\n",
+       "        <td>[Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>44</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>45</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>46</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>47</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>48</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>[Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>50</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>[Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>53</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>58</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>59</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>62</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>65</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>69</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>79</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>86</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>87</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>88</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>91</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>95</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>97</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>[Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>105</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>106</td>\n",
+       "        <td>[Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>[Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>109</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>110</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>118</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>124</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>129</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>130</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>[Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>[Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>133</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>139</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>140</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>141</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>142</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (2, [Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (3, [Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (4, [Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (5, [Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (6, [Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (7, [Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (8, [Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (9, [Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (10, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (11, [Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (12, [Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (13, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (14, [Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (15, [Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (16, [Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (17, [Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (18, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (19, [Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (20, [Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (21, [Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (22, [Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (23, [Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (24, [Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')], u'Iris-setosa'),\n",
+       " (25, [Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (26, [Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (27, [Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (28, [Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (29, [Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (30, [Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (31, [Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (32, [Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (33, [Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (34, [Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (35, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (36, [Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (37, [Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (38, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (39, [Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (40, [Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (41, [Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (42, [Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (43, [Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (44, [Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')], u'Iris-setosa'),\n",
+       " (45, [Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (46, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (47, [Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (48, [Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (49, [Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (50, [Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (51, [Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (52, [Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (53, [Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (54, [Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (55, [Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (56, [Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (57, [Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (58, [Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (59, [Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (60, [Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (61, [Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (62, [Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (63, [Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (64, [Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (65, [Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (66, [Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (67, [Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (68, [Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (69, [Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (70, [Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (71, [Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')], u'Iris-versicolor'),\n",
+       " (72, [Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (73, [Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (74, [Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (75, [Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (76, [Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (77, [Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (78, [Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')], u'Iris-versicolor'),\n",
+       " (79, [Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (80, [Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (81, [Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (82, [Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (83, [Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (84, [Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (85, [Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (86, [Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (87, [Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (88, [Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (89, [Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (90, [Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (91, [Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (92, [Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (93, [Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (94, [Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (95, [Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (96, [Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (97, [Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (98, [Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (99, [Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (100, [Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (101, [Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (102, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (103, [Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (104, [Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (105, [Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (106, [Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (107, [Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')], u'Iris-virginica'),\n",
+       " (108, [Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (109, [Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (110, [Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (111, [Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (112, [Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (113, [Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (114, [Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (115, [Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (116, [Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (117, [Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (118, [Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (119, [Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (120, [Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (121, [Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (122, [Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (123, [Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (124, [Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (125, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (126, [Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (127, [Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (128, [Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (129, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (130, [Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')], u'Iris-virginica'),\n",
+       " (131, [Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (132, [Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (133, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (134, [Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (135, [Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')], u'Iris-virginica'),\n",
+       " (136, [Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (137, [Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (138, [Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (139, [Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (140, [Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (141, [Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (142, [Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (143, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (144, [Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (145, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (146, [Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (147, [Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (148, [Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (149, [Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (150, [Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')], u'Iris-virginica')]"
+      ]
+     },
+     "execution_count": 4,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql \n",
+    "DROP TABLE IF EXISTS iris_data;\n",
+    "\n",
+    "CREATE TABLE iris_data(\n",
+    "    id serial,\n",
+    "    attributes numeric[],\n",
+    "    class_text varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO iris_data(id, attributes, class_text) VALUES\n",
+    "(1,ARRAY[5.1,3.5,1.4,0.2],'Iris-setosa'),\n",
+    "(2,ARRAY[4.9,3.0,1.4,0.2],'Iris-setosa'),\n",
+    "(3,ARRAY[4.7,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(4,ARRAY[4.6,3.1,1.5,0.2],'Iris-setosa'),\n",
+    "(5,ARRAY[5.0,3.6,1.4,0.2],'Iris-setosa'),\n",
+    "(6,ARRAY[5.4,3.9,1.7,0.4],'Iris-setosa'),\n",
+    "(7,ARRAY[4.6,3.4,1.4,0.3],'Iris-setosa'),\n",
+    "(8,ARRAY[5.0,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(9,ARRAY[4.4,2.9,1.4,0.2],'Iris-setosa'),\n",
+    "(10,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(11,ARRAY[5.4,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(12,ARRAY[4.8,3.4,1.6,0.2],'Iris-setosa'),\n",
+    "(13,ARRAY[4.8,3.0,1.4,0.1],'Iris-setosa'),\n",
+    "(14,ARRAY[4.3,3.0,1.1,0.1],'Iris-setosa'),\n",
+    "(15,ARRAY[5.8,4.0,1.2,0.2],'Iris-setosa'),\n",
+    "(16,ARRAY[5.7,4.4,1.5,0.4],'Iris-setosa'),\n",
+    "(17,ARRAY[5.4,3.9,1.3,0.4],'Iris-setosa'),\n",
+    "(18,ARRAY[5.1,3.5,1.4,0.3],'Iris-setosa'),\n",
+    "(19,ARRAY[5.7,3.8,1.7,0.3],'Iris-setosa'),\n",
+    "(20,ARRAY[5.1,3.8,1.5,0.3],'Iris-setosa'),\n",
+    "(21,ARRAY[5.4,3.4,1.7,0.2],'Iris-setosa'),\n",
+    "(22,ARRAY[5.1,3.7,1.5,0.4],'Iris-setosa'),\n",
+    "(23,ARRAY[4.6,3.6,1.0,0.2],'Iris-setosa'),\n",
+    "(24,ARRAY[5.1,3.3,1.7,0.5],'Iris-setosa'),\n",
+    "(25,ARRAY[4.8,3.4,1.9,0.2],'Iris-setosa'),\n",
+    "(26,ARRAY[5.0,3.0,1.6,0.2],'Iris-setosa'),\n",
+    "(27,ARRAY[5.0,3.4,1.6,0.4],'Iris-setosa'),\n",
+    "(28,ARRAY[5.2,3.5,1.5,0.2],'Iris-setosa'),\n",
+    "(29,ARRAY[5.2,3.4,1.4,0.2],'Iris-setosa'),\n",
+    "(30,ARRAY[4.7,3.2,1.6,0.2],'Iris-setosa'),\n",
+    "(31,ARRAY[4.8,3.1,1.6,0.2],'Iris-setosa'),\n",
+    "(32,ARRAY[5.4,3.4,1.5,0.4],'Iris-setosa'),\n",
+    "(33,ARRAY[5.2,4.1,1.5,0.1],'Iris-setosa'),\n",
+    "(34,ARRAY[5.5,4.2,1.4,0.2],'Iris-setosa'),\n",
+    "(35,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(36,ARRAY[5.0,3.2,1.2,0.2],'Iris-setosa'),\n",
+    "(37,ARRAY[5.5,3.5,1.3,0.2],'Iris-setosa'),\n",
+    "(38,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(39,ARRAY[4.4,3.0,1.3,0.2],'Iris-setosa'),\n",
+    "(40,ARRAY[5.1,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(41,ARRAY[5.0,3.5,1.3,0.3],'Iris-setosa'),\n",
+    "(42,ARRAY[4.5,2.3,1.3,0.3],'Iris-setosa'),\n",
+    "(43,ARRAY[4.4,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(44,ARRAY[5.0,3.5,1.6,0.6],'Iris-setosa'),\n",
+    "(45,ARRAY[5.1,3.8,1.9,0.4],'Iris-setosa'),\n",
+    "(46,ARRAY[4.8,3.0,1.4,0.3],'Iris-setosa'),\n",
+    "(47,ARRAY[5.1,3.8,1.6,0.2],'Iris-setosa'),\n",
+    "(48,ARRAY[4.6,3.2,1.4,0.2],'Iris-setosa'),\n",
+    "(49,ARRAY[5.3,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(50,ARRAY[5.0,3.3,1.4,0.2],'Iris-setosa'),\n",
+    "(51,ARRAY[7.0,3.2,4.7,1.4],'Iris-versicolor'),\n",
+    "(52,ARRAY[6.4,3.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(53,ARRAY[6.9,3.1,4.9,1.5],'Iris-versicolor'),\n",
+    "(54,ARRAY[5.5,2.3,4.0,1.3],'Iris-versicolor'),\n",
+    "(55,ARRAY[6.5,2.8,4.6,1.5],'Iris-versicolor'),\n",
+    "(56,ARRAY[5.7,2.8,4.5,1.3],'Iris-versicolor'),\n",
+    "(57,ARRAY[6.3,3.3,4.7,1.6],'Iris-versicolor'),\n",
+    "(58,ARRAY[4.9,2.4,3.3,1.0],'Iris-versicolor'),\n",
+    "(59,ARRAY[6.6,2.9,4.6,1.3],'Iris-versicolor'),\n",
+    "(60,ARRAY[5.2,2.7,3.9,1.4],'Iris-versicolor'),\n",
+    "(61,ARRAY[5.0,2.0,3.5,1.0],'Iris-versicolor'),\n",
+    "(62,ARRAY[5.9,3.0,4.2,1.5],'Iris-versicolor'),\n",
+    "(63,ARRAY[6.0,2.2,4.0,1.0],'Iris-versicolor'),\n",
+    "(64,ARRAY[6.1,2.9,4.7,1.4],'Iris-versicolor'),\n",
+    "(65,ARRAY[5.6,2.9,3.6,1.3],'Iris-versicolor'),\n",
+    "(66,ARRAY[6.7,3.1,4.4,1.4],'Iris-versicolor'),\n",
+    "(67,ARRAY[5.6,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(68,ARRAY[5.8,2.7,4.1,1.0],'Iris-versicolor'),\n",
+    "(69,ARRAY[6.2,2.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(70,ARRAY[5.6,2.5,3.9,1.1],'Iris-versicolor'),\n",
+    "(71,ARRAY[5.9,3.2,4.8,1.8],'Iris-versicolor'),\n",
+    "(72,ARRAY[6.1,2.8,4.0,1.3],'Iris-versicolor'),\n",
+    "(73,ARRAY[6.3,2.5,4.9,1.5],'Iris-versicolor'),\n",
+    "(74,ARRAY[6.1,2.8,4.7,1.2],'Iris-versicolor'),\n",
+    "(75,ARRAY[6.4,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(76,ARRAY[6.6,3.0,4.4,1.4],'Iris-versicolor'),\n",
+    "(77,ARRAY[6.8,2.8,4.8,1.4],'Iris-versicolor'),\n",
+    "(78,ARRAY[6.7,3.0,5.0,1.7],'Iris-versicolor'),\n",
+    "(79,ARRAY[6.0,2.9,4.5,1.5],'Iris-versicolor'),\n",
+    "(80,ARRAY[5.7,2.6,3.5,1.0],'Iris-versicolor'),\n",
+    "(81,ARRAY[5.5,2.4,3.8,1.1],'Iris-versicolor'),\n",
+    "(82,ARRAY[5.5,2.4,3.7,1.0],'Iris-versicolor'),\n",
+    "(83,ARRAY[5.8,2.7,3.9,1.2],'Iris-versicolor'),\n",
+    "(84,ARRAY[6.0,2.7,5.1,1.6],'Iris-versicolor'),\n",
+    "(85,ARRAY[5.4,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(86,ARRAY[6.0,3.4,4.5,1.6],'Iris-versicolor'),\n",
+    "(87,ARRAY[6.7,3.1,4.7,1.5],'Iris-versicolor'),\n",
+    "(88,ARRAY[6.3,2.3,4.4,1.3],'Iris-versicolor'),\n",
+    "(89,ARRAY[5.6,3.0,4.1,1.3],'Iris-versicolor'),\n",
+    "(90,ARRAY[5.5,2.5,4.0,1.3],'Iris-versicolor'),\n",
+    "(91,ARRAY[5.5,2.6,4.4,1.2],'Iris-versicolor'),\n",
+    "(92,ARRAY[6.1,3.0,4.6,1.4],'Iris-versicolor'),\n",
+    "(93,ARRAY[5.8,2.6,4.0,1.2],'Iris-versicolor'),\n",
+    "(94,ARRAY[5.0,2.3,3.3,1.0],'Iris-versicolor'),\n",
+    "(95,ARRAY[5.6,2.7,4.2,1.3],'Iris-versicolor'),\n",
+    "(96,ARRAY[5.7,3.0,4.2,1.2],'Iris-versicolor'),\n",
+    "(97,ARRAY[5.7,2.9,4.2,1.3],'Iris-versicolor'),\n",
+    "(98,ARRAY[6.2,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(99,ARRAY[5.1,2.5,3.0,1.1],'Iris-versicolor'),\n",
+    "(100,ARRAY[5.7,2.8,4.1,1.3],'Iris-versicolor'),\n",
+    "(101,ARRAY[6.3,3.3,6.0,2.5],'Iris-virginica'),\n",
+    "(102,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(103,ARRAY[7.1,3.0,5.9,2.1],'Iris-virginica'),\n",
+    "(104,ARRAY[6.3,2.9,5.6,1.8],'Iris-virginica'),\n",
+    "(105,ARRAY[6.5,3.0,5.8,2.2],'Iris-virginica'),\n",
+    "(106,ARRAY[7.6,3.0,6.6,2.1],'Iris-virginica'),\n",
+    "(107,ARRAY[4.9,2.5,4.5,1.7],'Iris-virginica'),\n",
+    "(108,ARRAY[7.3,2.9,6.3,1.8],'Iris-virginica'),\n",
+    "(109,ARRAY[6.7,2.5,5.8,1.8],'Iris-virginica'),\n",
+    "(110,ARRAY[7.2,3.6,6.1,2.5],'Iris-virginica'),\n",
+    "(111,ARRAY[6.5,3.2,5.1,2.0],'Iris-virginica'),\n",
+    "(112,ARRAY[6.4,2.7,5.3,1.9],'Iris-virginica'),\n",
+    "(113,ARRAY[6.8,3.0,5.5,2.1],'Iris-virginica'),\n",
+    "(114,ARRAY[5.7,2.5,5.0,2.0],'Iris-virginica'),\n",
+    "(115,ARRAY[5.8,2.8,5.1,2.4],'Iris-virginica'),\n",
+    "(116,ARRAY[6.4,3.2,5.3,2.3],'Iris-virginica'),\n",
+    "(117,ARRAY[6.5,3.0,5.5,1.8],'Iris-virginica'),\n",
+    "(118,ARRAY[7.7,3.8,6.7,2.2],'Iris-virginica'),\n",
+    "(119,ARRAY[7.7,2.6,6.9,2.3],'Iris-virginica'),\n",
+    "(120,ARRAY[6.0,2.2,5.0,1.5],'Iris-virginica'),\n",
+    "(121,ARRAY[6.9,3.2,5.7,2.3],'Iris-virginica'),\n",
+    "(122,ARRAY[5.6,2.8,4.9,2.0],'Iris-virginica'),\n",
+    "(123,ARRAY[7.7,2.8,6.7,2.0],'Iris-virginica'),\n",
+    "(124,ARRAY[6.3,2.7,4.9,1.8],'Iris-virginica'),\n",
+    "(125,ARRAY[6.7,3.3,5.7,2.1],'Iris-virginica'),\n",
+    "(126,ARRAY[7.2,3.2,6.0,1.8],'Iris-virginica'),\n",
+    "(127,ARRAY[6.2,2.8,4.8,1.8],'Iris-virginica'),\n",
+    "(128,ARRAY[6.1,3.0,4.9,1.8],'Iris-virginica'),\n",
+    "(129,ARRAY[6.4,2.8,5.6,2.1],'Iris-virginica'),\n",
+    "(130,ARRAY[7.2,3.0,5.8,1.6],'Iris-virginica'),\n",
+    "(131,ARRAY[7.4,2.8,6.1,1.9],'Iris-virginica'),\n",
+    "(132,ARRAY[7.9,3.8,6.4,2.0],'Iris-virginica'),\n",
+    "(133,ARRAY[6.4,2.8,5.6,2.2],'Iris-virginica'),\n",
+    "(134,ARRAY[6.3,2.8,5.1,1.5],'Iris-virginica'),\n",
+    "(135,ARRAY[6.1,2.6,5.6,1.4],'Iris-virginica'),\n",
+    "(136,ARRAY[7.7,3.0,6.1,2.3],'Iris-virginica'),\n",
+    "(137,ARRAY[6.3,3.4,5.6,2.4],'Iris-virginica'),\n",
+    "(138,ARRAY[6.4,3.1,5.5,1.8],'Iris-virginica'),\n",
+    "(139,ARRAY[6.0,3.0,4.8,1.8],'Iris-virginica'),\n",
+    "(140,ARRAY[6.9,3.1,5.4,2.1],'Iris-virginica'),\n",
+    "(141,ARRAY[6.7,3.1,5.6,2.4],'Iris-virginica'),\n",
+    "(142,ARRAY[6.9,3.1,5.1,2.3],'Iris-virginica'),\n",
+    "(143,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(144,ARRAY[6.8,3.2,5.9,2.3],'Iris-virginica'),\n",
+    "(145,ARRAY[6.7,3.3,5.7,2.5],'Iris-virginica'),\n",
+    "(146,ARRAY[6.7,3.0,5.2,2.3],'Iris-virginica'),\n",
+    "(147,ARRAY[6.3,2.5,5.0,1.9],'Iris-virginica'),\n",
+    "(148,ARRAY[6.5,3.0,5.2,2.0],'Iris-virginica'),\n",
+    "(149,ARRAY[6.2,3.4,5.4,2.3],'Iris-virginica'),\n",
+    "(150,ARRAY[5.9,3.0,5.1,1.8],'Iris-virginica');\n",
+    "\n",
+    "SELECT * FROM iris_data ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Create a test/validation dataset from the training data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(120L,)]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train, iris_test;\n",
+    "\n",
+    "-- Set seed so results are reproducible\n",
+    "SELECT setseed(0);\n",
+    "\n",
+    "SELECT madlib.train_test_split('iris_data',     -- Source table\n",
+    "                               'iris',          -- Output table root name\n",
+    "                                0.8,            -- Train proportion\n",
+    "                                NULL,           -- Test proportion (0.2)\n",
+    "                                NULL,           -- Strata definition\n",
+    "                                NULL,           -- Output all columns\n",
+    "                                NULL,           -- Sample without replacement\n",
+    "                                TRUE            -- Separate output tables\n",
+    "                              );\n",
+    "\n",
+    "SELECT COUNT(*) FROM iris_train;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pp\"></a>\n",
+    "# 2. Call preprocessor for deep learning\n",
+    "Training dataset (uses training preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>attributes_shape</th>\n",
+       "        <th>class_text_shape</th>\n",
+       "        <th>buffer_id</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[60, 4]</td>\n",
+       "        <td>[60, 3]</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[60, 4]</td>\n",
+       "        <td>[60, 3]</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[([60, 4], [60, 3], 0), ([60, 4], [60, 3], 1)]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train_packed, iris_train_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('iris_train',         -- Source table\n",
+    "                                       'iris_train_packed',  -- Output table\n",
+    "                                       'class_text',        -- Dependent variable\n",
+    "                                       'attributes'         -- Independent variable\n",
+    "                                        ); \n",
+    "\n",
+    "SELECT attributes_shape, class_text_shape, buffer_id FROM iris_train_packed ORDER BY buffer_id;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train</td>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>60</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train', u'iris_train_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 60, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_train_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Validation dataset (uses validation preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>attributes_shape</th>\n",
+       "        <th>class_text_shape</th>\n",
+       "        <th>buffer_id</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 4]</td>\n",
+       "        <td>[15, 3]</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 4]</td>\n",
+       "        <td>[15, 3]</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[([15, 4], [15, 3], 0), ([15, 4], [15, 3], 1)]"
+      ]
+     },
+     "execution_count": 8,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_test_packed, iris_test_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('iris_test',          -- Source table\n",
+    "                                         'iris_test_packed',   -- Output table\n",
+    "                                         'class_text',         -- Dependent variable\n",
+    "                                         'attributes',         -- Independent variable\n",
+    "                                         'iris_train_packed'   -- From training preprocessor step\n",
+    "                                          ); \n",
+    "\n",
+    "SELECT attributes_shape, class_text_shape, buffer_id FROM iris_test_packed ORDER BY buffer_id;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_test</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>15</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_test', u'iris_test_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 15, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 9,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_test_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load\"></a>\n",
+    "# 3. Define and load model architecture\n",
+    "Import Keras libraries"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 1 hidden layer:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "WARNING:tensorflow:From /Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/tensorflow/python/ops/init_ops.py:1251: calling __init__ (from tensorflow.python.ops.init_ops) with dtype is deprecated and will be removed in a future version.\n",
+      "Instructions for updating:\n",
+      "Call initializer instance with the dtype argument instead of passing it to the constructor\n",
+      "Model: \"sequential\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense (Dense)                (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_1 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_2 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 193\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model1 = Sequential()\n",
+    "model1.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model1.add(Dense(10, activation='relu'))\n",
+    "model1.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model1.summary()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 12,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 12,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model1.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 2 hidden layers:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 13,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_1\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_3 (Dense)              (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_4 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_5 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_6 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 303\n",
+      "Trainable params: 303\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model2 = Sequential()\n",
+    "model2.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model2.summary()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 14,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_1\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 14,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model2.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 15,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>model_weights</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>__internal_madlib_id__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>MLP with 1 hidden layer</td>\n",
+       "        <td>__madlib_temp_4017958_1614991901_4240024__</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>MLP with 2 hidden layers</td>\n",
+       "        <td>__madlib_temp_28416680_1614991901_72274844__</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'MLP with 1 hidden layer', u'__madlib_temp_4017958_1614991901_4240024__'),\n",
+       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1835 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'MLP with 2 hidden layers', u'__madlib_temp_28416680_1614991901_72274844__')]"
+      ]
+     },
+     "execution_count": 15,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS model_arch_library;\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Sophie',              -- Name\n",
+    "                               'MLP with 1 hidden layer'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Maria',               -- Name\n",
+    "                               'MLP with 2 hidden layers'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT * FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"def_mst\"></a>\n",
+    "# 4.  Define and load model selection tuples"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Generate model configurations using grid search. The output table for grid search contains the unique combinations of model architectures, compile and fit parameters."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 16,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8')]"
+      ]
+     },
+     "execution_count": 16,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'],\n",
+    "                                             'optimizer_params_list': [ {'optimizer': ['Adam'], 'lr': [0.001, 0.01, 0.1]} ],\n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid\n",
+    "                                         $$\n",
+    "                                         { 'batch_size': [4, 8],\n",
+    "                                           'epochs': [1]\n",
+    "                                         }\n",
+    "                                         $$,                  -- fit_param_grid\n",
+    "                                         'grid'               -- search_type\n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "This is the name of the model architecture table that corresponds to the model selection table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 17,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>object_table</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'model_arch_library', None)]"
+      ]
+     },
+     "execution_count": 17,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM mst_table_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train\"></a>\n",
+    "# 5.  Train\n",
+    "Train multiple models:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 18,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit_multiple_model</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 18,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_multi_model, iris_multi_model_summary, iris_multi_model_info;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
+    "                                              'iris_multi_model',     -- model_output_table\n",
+    "                                              'mst_table',            -- model_selection_table\n",
+    "                                              10,                     -- num_iterations\n",
+    "                                              FALSE                   -- use gpus\n",
+    "                                             );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 19,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>warm_start</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>iris_multi_model</td>\n",
+       "        <td>iris_multi_model_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>mst_table</td>\n",
+       "        <td>None</td>\n",
+       "        <td>10</td>\n",
+       "        <td>10</td>\n",
+       "        <td>False</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2021-03-06 00:51:48.452654</td>\n",
+       "        <td>2021-03-06 00:53:20.221035</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[10]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', None, u'iris_multi_model', u'iris_multi_model_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'mst_table', None, 10, 10, False, None, None, datetime.datetime(2021, 3, 6, 0, 51, 48, 452654), datetime.datetime(2021, 3, 6, 0, 53, 20, 221035), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0, [10])]"
+      ]
+     },
+     "execution_count": 19,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View results for each model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 20,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[90.2427790164948]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.983333349228</td>\n",
+       "        <td>0.201789721847</td>\n",
+       "        <td>[0.983333349227905]</td>\n",
+       "        <td>[0.201789721846581]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[88.9964590072632]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.134730249643</td>\n",
+       "        <td>[0.933333337306976]</td>\n",
+       "        <td>[0.134730249643326]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[88.7690601348877]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.402144879103</td>\n",
+       "        <td>[0.933333337306976]</td>\n",
+       "        <td>[0.402144879102707]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[90.9196391105652]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.416792035103</td>\n",
+       "        <td>[0.933333337306976]</td>\n",
+       "        <td>[0.416792035102844]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[89.534707069397]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.908333361149</td>\n",
+       "        <td>0.19042557478</td>\n",
+       "        <td>[0.908333361148834]</td>\n",
+       "        <td>[0.19042557477951]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[89.273796081543]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.181902274489</td>\n",
+       "        <td>[0.899999976158142]</td>\n",
+       "        <td>[0.181902274489403]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[90.4800100326538]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.824999988079</td>\n",
+       "        <td>0.303107827902</td>\n",
+       "        <td>[0.824999988079071]</td>\n",
+       "        <td>[0.30310782790184]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[89.7936120033264]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.808333337307</td>\n",
+       "        <td>0.300039559603</td>\n",
+       "        <td>[0.808333337306976]</td>\n",
+       "        <td>[0.300039559602737]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[90.0158791542053]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.658333361149</td>\n",
+       "        <td>0.869387447834</td>\n",
+       "        <td>[0.658333361148834]</td>\n",
+       "        <td>[0.869387447834015]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[91.1929490566254]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.558333337307</td>\n",
+       "        <td>0.84612262249</td>\n",
+       "        <td>[0.558333337306976]</td>\n",
+       "        <td>[0.846122622489929]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[91.7660541534424]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.341666668653</td>\n",
+       "        <td>1.10138702393</td>\n",
+       "        <td>[0.341666668653488]</td>\n",
+       "        <td>[1.10138702392578]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[91.5026919841766]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.341666668653</td>\n",
+       "        <td>1.10163521767</td>\n",
+       "        <td>[0.341666668653488]</td>\n",
+       "        <td>[1.10163521766663]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [90.2427790164948], [u'accuracy'], u'categorical_crossentropy', 0.983333349227905, 0.201789721846581, [0.983333349227905], [0.201789721846581], None, None, None, None),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [88.9964590072632], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.134730249643326, [0.933333337306976], [0.134730249643326], None, None, None, None),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [88.7690601348877], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.402144879102707, [0.933333337306976], [0.402144879102707], None, None, None, None),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [90.9196391105652], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.416792035102844, [0.933333337306976], [0.416792035102844], None, None, None, None),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [89.534707069397], [u'accuracy'], u'categorical_crossentropy', 0.908333361148834, 0.19042557477951, [0.908333361148834], [0.19042557477951], None, None, None, None),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [89.273796081543], [u'accuracy'], u'categorical_crossentropy', 0.899999976158142, 0.181902274489403, [0.899999976158142], [0.181902274489403], None, None, None, None),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [90.4800100326538], [u'accuracy'], u'categorical_crossentropy', 0.824999988079071, 0.30310782790184, [0.824999988079071], [0.30310782790184], None, None, None, None),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [89.7936120033264], [u'accuracy'], u'categorical_crossentropy', 0.808333337306976, 0.300039559602737, [0.808333337306976], [0.300039559602737], None, None, None, None),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [90.0158791542053], [u'accuracy'], u'categorical_crossentropy', 0.658333361148834, 0.869387447834015, [0.658333361148834], [0.869387447834015], None, None, None, None),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [91.1929490566254], [u'accuracy'], u'categorical_crossentropy', 0.558333337306976, 0.846122622489929, [0.558333337306976], [0.846122622489929], None, None, None, None),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [91.7660541534424], [u'accuracy'], u'categorical_crossentropy', 0.341666668653488, 1.10138702392578, [0.341666668653488], [1.10138702392578], None, None, None, None),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [91.5026919841766], [u'accuracy'], u'categorical_crossentropy', 0.341666668653488, 1.10163521766663, [0.341666668653488], [1.10163521766663], None, None, None, None)]"
+      ]
+     },
+     "execution_count": 20,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_info ORDER BY training_metrics_final DESC, training_loss_final;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"eval\"></a>\n",
+    "# 6. Evaluate\n",
+    "\n",
+    "Now run evaluate using model we built above:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 21,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>loss</th>\n",
+       "        <th>metric</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>0.194916069508</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(0.194916069507599, 0.899999976158142, [u'accuracy'], u'categorical_crossentropy')]"
+      ]
+     },
+     "execution_count": 21,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_validate;\n",
+    "SELECT madlib.madlib_keras_evaluate('iris_multi_model',  -- model\n",
+    "                                    'iris_test_packed',  -- test table\n",
+    "                                    'iris_validate',     -- output table\n",
+    "                                     NULL,               -- use gpus\n",
+    "                                     9                   -- mst_key to use\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_validate;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred\"></a>\n",
+    "# 7. Predict\n",
+    "\n",
+    "Now predict using model we built.  We will use the validation data set for prediction as well, which is not usual but serves to show the syntax. The prediction is in the estimated_class_text column:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 22,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "30 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999999</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999999</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999999</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999999</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.99069124</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9864196</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9983382</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9991603</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9974559</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.60661113</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9940832</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9987955</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7598468</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8414144</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.715776</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9163472</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5081183</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.85080105</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9842195</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6804195</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.81555897</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.92707217</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7158722</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.55272627</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7662018</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (10, u'class_text', u'Iris-setosa', 0.9999999),\n",
+       " (12, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (14, u'class_text', u'Iris-setosa', 0.9999999),\n",
+       " (18, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (20, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (30, u'class_text', u'Iris-setosa', 0.9999999),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.9999999),\n",
+       " (49, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (55, u'class_text', u'Iris-versicolor', 0.99069124),\n",
+       " (64, u'class_text', u'Iris-versicolor', 0.9864196),\n",
+       " (70, u'class_text', u'Iris-versicolor', 0.9983382),\n",
+       " (76, u'class_text', u'Iris-versicolor', 0.9991603),\n",
+       " (82, u'class_text', u'Iris-versicolor', 0.9974559),\n",
+       " (84, u'class_text', u'Iris-versicolor', 0.60661113),\n",
+       " (92, u'class_text', u'Iris-versicolor', 0.9940832),\n",
+       " (98, u'class_text', u'Iris-versicolor', 0.9987955),\n",
+       " (99, u'class_text', u'Iris-versicolor', 0.7598468),\n",
+       " (102, u'class_text', u'Iris-virginica', 0.8414144),\n",
+       " (107, u'class_text', u'Iris-virginica', 0.715776),\n",
+       " (114, u'class_text', u'Iris-virginica', 0.9163472),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.5081183),\n",
+       " (121, u'class_text', u'Iris-virginica', 0.85080105),\n",
+       " (123, u'class_text', u'Iris-virginica', 0.9842195),\n",
+       " (125, u'class_text', u'Iris-virginica', 0.6804195),\n",
+       " (127, u'class_text', u'Iris-versicolor', 0.81555897),\n",
+       " (145, u'class_text', u'Iris-virginica', 0.92707217),\n",
+       " (147, u'class_text', u'Iris-virginica', 0.7158722),\n",
+       " (148, u'class_text', u'Iris-versicolor', 0.55272627),\n",
+       " (149, u'class_text', u'Iris-virginica', 0.7662018)]"
+      ]
+     },
+     "execution_count": 22,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_multi_model', -- model\n",
+    "                                   'iris_test',        -- test_table\n",
+    "                                   'id',               -- id column\n",
+    "                                   'attributes',       -- independent var\n",
+    "                                   'iris_predict',     -- output table\n",
+    "                                    'response',        -- prediction type\n",
+    "                                    FALSE,             -- use gpus\n",
+    "                                    9                  -- mst_key to use\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 23,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3L,)]"
+      ]
+     },
+     "execution_count": 23,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict JOIN iris_test USING (id) \n",
+    "WHERE iris_predict.class_value != iris_test.class_text;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 24,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90.00</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('90.00'),)]"
+      ]
+     },
+     "execution_count": 24,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict.class_value as estimated\n",
+    "     from iris_predict inner join iris_test\n",
+    "     on iris_test.id=iris_predict.id) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class2\"></a>\n",
+    "# Classification with Other Parameters\n",
+    "\n",
+    "<a id=\"val_dataset\"></a>\n",
+    "# 1.  Validation dataset\n",
+    "\n",
+    "Now use a validation dataset and compute metrics every 2nd iteration using the 'metrics_compute_frequency' parameter.  This can help reduce run time if you do not need metrics computed at every iteration."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 25,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit_multiple_model</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 25,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_multi_model, iris_multi_model_summary, iris_multi_model_info;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
+    "                                              'iris_multi_model',     -- model_output_table\n",
+    "                                              'mst_table',            -- model_selection_table\n",
+    "                                               10,                     -- num_iterations\n",
+    "                                               FALSE,                 -- use gpus\n",
+    "                                              'iris_test_packed',     -- validation dataset\n",
+    "                                               3,                     -- metrics compute frequency\n",
+    "                                               FALSE,                 -- warm start\n",
+    "                                              'Sophie L.',            -- name\n",
+    "                                              'Model selection for iris dataset'  -- description\n",
+    "                                             );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 26,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>warm_start</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>iris_multi_model</td>\n",
+       "        <td>iris_multi_model_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>mst_table</td>\n",
+       "        <td>None</td>\n",
+       "        <td>10</td>\n",
+       "        <td>3</td>\n",
+       "        <td>False</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Model selection for iris dataset</td>\n",
+       "        <td>2021-03-06 00:53:31.218406</td>\n",
+       "        <td>2021-03-06 00:55:25.621208</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3, 6, 9, 10]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_test_packed', u'iris_multi_model', u'iris_multi_model_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'mst_table', None, 10, 3, False, u'Sophie L.', u'Model selection for iris dataset', datetime.datetime(2021, 3, 6, 0, 53, 31, 218406), datetime.datetime(2021, 3, 6, 0, 55, 25, 621208), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0, [3, 6, 9, 10])]"
+      ]
+     },
+     "execution_count": 26,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View performance of each model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 27,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[31.5490398406982, 64.223620891571, 97.8899219036102, 113.156138896942]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.991666674614</td>\n",
+       "        <td>0.177691921592</td>\n",
+       "        <td>[0.824999988079071, 0.975000023841858, 0.933333337306976, 0.991666674613953]</td>\n",
+       "        <td>[0.508709609508514, 0.290052831172943, 0.217903628945351, 0.177691921591759]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.20564225316</td>\n",
+       "        <td>[0.833333313465118, 0.966666638851166, 0.933333337306976, 0.966666638851166]</td>\n",
+       "        <td>[0.516587793827057, 0.316147029399872, 0.228292018175125, 0.205642253160477]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[30.4000718593597, 62.9767029285431, 96.690801858902, 112.145288944244]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.908333361149</td>\n",
+       "        <td>0.203085869551</td>\n",
+       "        <td>[0.933333337306976, 0.808333337306976, 0.958333313465118, 0.908333361148834]</td>\n",
+       "        <td>[0.372362315654755, 0.304766088724136, 0.11820487678051, 0.203085869550705]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.190864190459</td>\n",
+       "        <td>[0.966666638851166, 0.833333313465118, 0.966666638851166, 0.933333337306976]</td>\n",
+       "        <td>[0.347199022769928, 0.290798246860504, 0.110275268554688, 0.190864190459251]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[30.875373840332, 63.4593389034271, 97.1958589553833, 112.702126979828]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.883333325386</td>\n",
+       "        <td>0.692279815674</td>\n",
+       "        <td>[0.533333361148834, 0.616666674613953, 0.875, 0.883333325386047]</td>\n",
+       "        <td>[1.08197057247162, 0.851473987102509, 0.729827761650085, 0.692279815673828]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.674779772758</td>\n",
+       "        <td>[0.600000023841858, 0.666666686534882, 0.899999976158142, 0.899999976158142]</td>\n",
+       "        <td>[1.05298256874084, 0.817528009414673, 0.710631787776947, 0.674779772758484]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[29.8903229236603, 62.4677069187164, 96.1764039993286, 111.539803981781]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.925000011921</td>\n",
+       "        <td>0.176520362496</td>\n",
+       "        <td>[0.833333313465118, 0.925000011920929, 0.774999976158142, 0.925000011920929]</td>\n",
+       "        <td>[0.324734181165695, 0.182637020945549, 0.468331128358841, 0.176520362496376]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.2585529387</td>\n",
+       "        <td>[0.866666674613953, 0.866666674613953, 0.866666674613953, 0.899999976158142]</td>\n",
+       "        <td>[0.341204434633255, 0.261798053979874, 0.45467621088028, 0.258552938699722]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[31.7836039066315, 64.4592599868774, 98.1328208446503, 113.377946853638]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.891666650772</td>\n",
+       "        <td>0.797108471394</td>\n",
+       "        <td>[0.341666668653488, 0.491666674613953, 0.916666686534882, 0.891666650772095]</td>\n",
+       "        <td>[1.09786474704742, 0.967048287391663, 0.838281869888306, 0.797108471393585]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.800795376301</td>\n",
+       "        <td>[0.300000011920929, 0.433333337306976, 0.933333337306976, 0.899999976158142]</td>\n",
+       "        <td>[1.07609903812408, 0.962578594684601, 0.834975183010101, 0.800795376300812]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[30.1456639766693, 62.722916841507, 96.4333670139313, 111.892151832581]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.816666662693</td>\n",
+       "        <td>0.734887838364</td>\n",
+       "        <td>[0.850000023841858, 0.958333313465118, 0.966666638851166, 0.816666662693024]</td>\n",
+       "        <td>[0.335647404193878, 0.0894104242324829, 0.0672163665294647, 0.734887838363647]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.665323019028</td>\n",
+       "        <td>[0.866666674613953, 0.966666638851166, 0.966666638851166, 0.866666674613953]</td>\n",
+       "        <td>[0.320426166057587, 0.154994085431099, 0.204012081027031, 0.66532301902771]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[32.0452349185944, 64.7241299152374, 98.4015560150146, 113.899842977524]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.791666686535</td>\n",
+       "        <td>0.772948563099</td>\n",
+       "        <td>[0.316666662693024, 0.349999994039536, 0.725000023841858, 0.791666686534882]</td>\n",
+       "        <td>[1.01266825199127, 0.905348658561707, 0.807280421257019, 0.772948563098907]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.740880072117</td>\n",
+       "        <td>[0.400000005960464, 0.466666668653488, 0.800000011920929, 0.866666674613953]</td>\n",
+       "        <td>[0.964996755123138, 0.868514597415924, 0.771895349025726, 0.740880072116852]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[30.6602540016174, 63.2428169250488, 96.9531948566437, 112.484740972519]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.691666662693</td>\n",
+       "        <td>0.501820206642</td>\n",
+       "        <td>[0.658333361148834, 0.658333361148834, 0.658333361148834, 0.691666662693024]</td>\n",
+       "        <td>[0.654709756374359, 0.581917643547058, 1.33844769001007, 0.501820206642151]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.457984447479</td>\n",
+       "        <td>[0.699999988079071, 0.699999988079071, 0.699999988079071, 0.766666650772095]</td>\n",
+       "        <td>[0.592061340808868, 0.525563180446625, 1.17788350582123, 0.457984447479248]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[31.0910878181458, 63.7646949291229, 97.4185988903046, 112.939773797989]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.666666686535</td>\n",
+       "        <td>0.50052946806</td>\n",
+       "        <td>[0.433333337306976, 0.641666650772095, 0.649999976158142, 0.666666686534882]</td>\n",
+       "        <td>[0.850135624408722, 0.611121952533722, 0.509139358997345, 0.50052946805954]</td>\n",
+       "        <td>0.733333349228</td>\n",
+       "        <td>0.459399551153</td>\n",
+       "        <td>[0.466666668653488, 0.699999988079071, 0.699999988079071, 0.733333349227905]</td>\n",
+       "        <td>[0.802468597888947, 0.571285247802734, 0.492577910423279, 0.459399551153183]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[29.6670269966125, 62.2440509796143, 95.9554150104523, 111.311369895935]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.733333349228</td>\n",
+       "        <td>0.821944594383</td>\n",
+       "        <td>[0.341666668653488, 0.341666668653488, 0.658333361148834, 0.733333349227905]</td>\n",
+       "        <td>[1.06431686878204, 0.996406197547913, 0.869706034660339, 0.82194459438324]</td>\n",
+       "        <td>0.699999988079</td>\n",
+       "        <td>0.852133929729</td>\n",
+       "        <td>[0.300000011920929, 0.300000011920929, 0.699999988079071, 0.699999988079071]</td>\n",
+       "        <td>[1.09268116950989, 1.01670277118683, 0.891825795173645, 0.852133929729462]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[32.5322558879852, 65.2217888832092, 98.9477097988129, 114.400418996811]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.683333337307</td>\n",
+       "        <td>0.455871999264</td>\n",
+       "        <td>[0.725000023841858, 0.683333337306976, 0.683333337306976, 0.683333337306976]</td>\n",
+       "        <td>[0.383917421102524, 0.457853585481644, 0.455943495035172, 0.455871999263763]</td>\n",
+       "        <td>0.600000023842</td>\n",
+       "        <td>0.488439053297</td>\n",
+       "        <td>[0.800000011920929, 0.600000023841858, 0.600000023841858, 0.600000023841858]</td>\n",
+       "        <td>[0.388951361179352, 0.50080794095993, 0.487448841333389, 0.488439053297043]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[32.2720308303833, 64.9502189159393, 98.6836059093475, 114.134181976318]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.675000011921</td>\n",
+       "        <td>0.452209770679</td>\n",
+       "        <td>[0.683333337306976, 0.675000011920929, 0.683333337306976, 0.675000011920929]</td>\n",
+       "        <td>[0.492754250764847, 0.469423890113831, 0.571796059608459, 0.452209770679474]</td>\n",
+       "        <td>0.600000023842</td>\n",
+       "        <td>0.464268505573</td>\n",
+       "        <td>[0.733333349227905, 0.766666650772095, 0.600000023841858, 0.600000023841858]</td>\n",
+       "        <td>[0.438488334417343, 0.390993624925613, 0.690678656101227, 0.464268505573273]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [31.5490398406982, 64.223620891571, 97.8899219036102, 113.156138896942], [u'accuracy'], u'categorical_crossentropy', 0.991666674613953, 0.177691921591759, [0.824999988079071, 0.975000023841858, 0.933333337306976, 0.991666674613953], [0.508709609508514, 0.290052831172943, 0.217903628945351, 0.177691921591759], 0.966666638851166, 0.205642253160477, [0.833333313465118, 0.966666638851166, 0.933333337306976, 0.966666638851166], [0.516587793827057, 0.316147029399872, 0.228292018175125, 0.205642253160477]),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [30.4000718593597, 62.9767029285431, 96.690801858902, 112.145288944244], [u'accuracy'], u'categorical_crossentropy', 0.908333361148834, 0.203085869550705, [0.933333337306976, 0.808333337306976, 0.958333313465118, 0.908333361148834], [0.372362315654755, 0.304766088724136, 0.11820487678051, 0.203085869550705], 0.933333337306976, 0.190864190459251, [0.966666638851166, 0.833333313465118, 0.966666638851166, 0.933333337306976], [0.347199022769928, 0.290798246860504, 0.110275268554688, 0.190864190459251]),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [30.875373840332, 63.4593389034271, 97.1958589553833, 112.702126979828], [u'accuracy'], u'categorical_crossentropy', 0.883333325386047, 0.692279815673828, [0.533333361148834, 0.616666674613953, 0.875, 0.883333325386047], [1.08197057247162, 0.851473987102509, 0.729827761650085, 0.692279815673828], 0.899999976158142, 0.674779772758484, [0.600000023841858, 0.666666686534882, 0.899999976158142, 0.899999976158142], [1.05298256874084, 0.817528009414673, 0.710631787776947, 0.674779772758484]),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [29.8903229236603, 62.4677069187164, 96.1764039993286, 111.539803981781], [u'accuracy'], u'categorical_crossentropy', 0.925000011920929, 0.176520362496376, [0.833333313465118, 0.925000011920929, 0.774999976158142, 0.925000011920929], [0.324734181165695, 0.182637020945549, 0.468331128358841, 0.176520362496376], 0.899999976158142, 0.258552938699722, [0.866666674613953, 0.866666674613953, 0.866666674613953, 0.899999976158142], [0.341204434633255, 0.261798053979874, 0.45467621088028, 0.258552938699722]),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [31.7836039066315, 64.4592599868774, 98.1328208446503, 113.377946853638], [u'accuracy'], u'categorical_crossentropy', 0.891666650772095, 0.797108471393585, [0.341666668653488, 0.491666674613953, 0.916666686534882, 0.891666650772095], [1.09786474704742, 0.967048287391663, 0.838281869888306, 0.797108471393585], 0.899999976158142, 0.800795376300812, [0.300000011920929, 0.433333337306976, 0.933333337306976, 0.899999976158142], [1.07609903812408, 0.962578594684601, 0.834975183010101, 0.800795376300812]),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [30.1456639766693, 62.722916841507, 96.4333670139313, 111.892151832581], [u'accuracy'], u'categorical_crossentropy', 0.816666662693024, 0.734887838363647, [0.850000023841858, 0.958333313465118, 0.966666638851166, 0.816666662693024], [0.335647404193878, 0.0894104242324829, 0.0672163665294647, 0.734887838363647], 0.866666674613953, 0.66532301902771, [0.866666674613953, 0.966666638851166, 0.966666638851166, 0.866666674613953], [0.320426166057587, 0.154994085431099, 0.204012081027031, 0.66532301902771]),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [32.0452349185944, 64.7241299152374, 98.4015560150146, 113.899842977524], [u'accuracy'], u'categorical_crossentropy', 0.791666686534882, 0.772948563098907, [0.316666662693024, 0.349999994039536, 0.725000023841858, 0.791666686534882], [1.01266825199127, 0.905348658561707, 0.807280421257019, 0.772948563098907], 0.866666674613953, 0.740880072116852, [0.400000005960464, 0.466666668653488, 0.800000011920929, 0.866666674613953], [0.964996755123138, 0.868514597415924, 0.771895349025726, 0.740880072116852]),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [30.6602540016174, 63.2428169250488, 96.9531948566437, 112.484740972519], [u'accuracy'], u'categorical_crossentropy', 0.691666662693024, 0.501820206642151, [0.658333361148834, 0.658333361148834, 0.658333361148834, 0.691666662693024], [0.654709756374359, 0.581917643547058, 1.33844769001007, 0.501820206642151], 0.766666650772095, 0.457984447479248, [0.699999988079071, 0.699999988079071, 0.699999988079071, 0.766666650772095], [0.592061340808868, 0.525563180446625, 1.17788350582123, 0.457984447479248]),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [31.0910878181458, 63.7646949291229, 97.4185988903046, 112.939773797989], [u'accuracy'], u'categorical_crossentropy', 0.666666686534882, 0.50052946805954, [0.433333337306976, 0.641666650772095, 0.649999976158142, 0.666666686534882], [0.850135624408722, 0.611121952533722, 0.509139358997345, 0.50052946805954], 0.733333349227905, 0.459399551153183, [0.466666668653488, 0.699999988079071, 0.699999988079071, 0.733333349227905], [0.802468597888947, 0.571285247802734, 0.492577910423279, 0.459399551153183]),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [29.6670269966125, 62.2440509796143, 95.9554150104523, 111.311369895935], [u'accuracy'], u'categorical_crossentropy', 0.733333349227905, 0.82194459438324, [0.341666668653488, 0.341666668653488, 0.658333361148834, 0.733333349227905], [1.06431686878204, 0.996406197547913, 0.869706034660339, 0.82194459438324], 0.699999988079071, 0.852133929729462, [0.300000011920929, 0.300000011920929, 0.699999988079071, 0.699999988079071], [1.09268116950989, 1.01670277118683, 0.891825795173645, 0.852133929729462]),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [32.5322558879852, 65.2217888832092, 98.9477097988129, 114.400418996811], [u'accuracy'], u'categorical_crossentropy', 0.683333337306976, 0.455871999263763, [0.725000023841858, 0.683333337306976, 0.683333337306976, 0.683333337306976], [0.383917421102524, 0.457853585481644, 0.455943495035172, 0.455871999263763], 0.600000023841858, 0.488439053297043, [0.800000011920929, 0.600000023841858, 0.600000023841858, 0.600000023841858], [0.388951361179352, 0.50080794095993, 0.487448841333389, 0.488439053297043]),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [32.2720308303833, 64.9502189159393, 98.6836059093475, 114.134181976318], [u'accuracy'], u'categorical_crossentropy', 0.675000011920929, 0.452209770679474, [0.683333337306976, 0.675000011920929, 0.683333337306976, 0.675000011920929], [0.492754250764847, 0.469423890113831, 0.571796059608459, 0.452209770679474], 0.600000023841858, 0.464268505573273, [0.733333349227905, 0.766666650772095, 0.600000023841858, 0.600000023841858], [0.438488334417343, 0.390993624925613, 0.690678656101227, 0.464268505573273])]"
+      ]
+     },
+     "execution_count": 27,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_info ORDER BY validation_metrics_final DESC;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Plot validation results"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 28,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%matplotlib notebook\n",
+    "import matplotlib.pyplot as plt\n",
+    "from matplotlib.ticker import MaxNLocator\n",
+    "from collections import defaultdict\n",
+    "import pandas as pd\n",
+    "import seaborn as sns\n",
+    "sns.set_palette(sns.color_palette(\"hls\", 20))\n",
+    "plt.rcParams.update({'font.size': 12})\n",
+    "pd.set_option('display.max_colwidth', -1)"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 29,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "7 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM iris_multi_model_info ORDER BY validation_loss ASC LIMIT 7;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "df_summary = %sql SELECT * FROM iris_multi_model_summary;\n",
+    "df_summary = df_summary.DataFrame()\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Iteration')\n",
+    "ax_metric.set_ylabel('Metric')\n",
+    "ax_metric.set_title('Validation metric curve')\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Iteration')\n",
+    "ax_loss.set_ylabel('Loss')\n",
+    "ax_loss.set_title('Validation loss curve')\n",
+    "\n",
+    "iters = df_summary['metrics_iters'][0]\n",
+    "\n",
+    "for mst_key in df_results['mst_key']:\n",
+    "    df_output_info = %sql SELECT validation_metrics,validation_loss FROM iris_multi_model_info WHERE mst_key = $mst_key\n",
+    "    df_output_info = df_output_info.DataFrame()\n",
+    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
+    "    validation_loss = df_output_info['validation_loss'][0]\n",
+    "    \n",
+    "    ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
+    "    ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred_prob\"></a>\n",
+    "# 2.  Predict probabilities\n",
+    "\n",
+    "Predict with probabilities for each class:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 30,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "90 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999932</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>6.7611923e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.2535056e-10</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999808</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>1.9209425e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>4.433645e-10</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99998367</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>1.6334934e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>4.3492965e-10</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999931</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>6.9504345e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.9190094e-10</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99999726</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>2.719827e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>2.4018267e-11</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999982</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>1.8036015e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.515534e-11</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99996376</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>3.623055e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.4014193e-09</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99995685</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>4.3105167e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.541236e-09</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99999833</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>1.6733742e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.0720992e-11</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.97456545</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.025385397</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>4.912654e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8837083</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.11627731</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.4444132e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9832433</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.016161945</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0005947249</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9934144</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.006202936</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00038262276</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9880006</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.01050145</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0014980072</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.743757</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.25624287</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.1804799e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9489498</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.050999135</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>5.1051586e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9882598</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.011410431</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00032975432</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7122672</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.2864844</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0012483773</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8344315</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.16556835</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>4.9313943e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7617606</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.23823881</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>6.2156596e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.85601324</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.1439867</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.4068247e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.76065344</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.23934652</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>4.0775706e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.65924823</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.34075174</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.7877243e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.968423</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.031577036</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.5606285e-11</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.72842705</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.2715729</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.7875385e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8053533</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.19464317</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.5179064e-06</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7297866</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.2702134</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>2.8784607e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5341273</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4658725</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>2.3799986e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.6266347</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3733647</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>5.7692125e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5517554</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4482443</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.1108453e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3, u'class_text', u'Iris-setosa', 0.9999932, 1),\n",
+       " (3, u'class_text', u'Iris-versicolor', 6.7611923e-06, 2),\n",
+       " (3, u'class_text', u'Iris-virginica', 1.2535056e-10, 3),\n",
+       " (10, u'class_text', u'Iris-setosa', 0.9999808, 1),\n",
+       " (10, u'class_text', u'Iris-versicolor', 1.9209425e-05, 2),\n",
+       " (10, u'class_text', u'Iris-virginica', 4.433645e-10, 3),\n",
+       " (12, u'class_text', u'Iris-setosa', 0.99998367, 1),\n",
+       " (12, u'class_text', u'Iris-versicolor', 1.6334934e-05, 2),\n",
+       " (12, u'class_text', u'Iris-virginica', 4.3492965e-10, 3),\n",
+       " (14, u'class_text', u'Iris-setosa', 0.9999931, 1),\n",
+       " (14, u'class_text', u'Iris-versicolor', 6.9504345e-06, 2),\n",
+       " (14, u'class_text', u'Iris-virginica', 1.9190094e-10, 3),\n",
+       " (18, u'class_text', u'Iris-setosa', 0.99999726, 1),\n",
+       " (18, u'class_text', u'Iris-versicolor', 2.719827e-06, 2),\n",
+       " (18, u'class_text', u'Iris-virginica', 2.4018267e-11, 3),\n",
+       " (20, u'class_text', u'Iris-setosa', 0.9999982, 1),\n",
+       " (20, u'class_text', u'Iris-versicolor', 1.8036015e-06, 2),\n",
+       " (20, u'class_text', u'Iris-virginica', 1.515534e-11, 3),\n",
+       " (30, u'class_text', u'Iris-setosa', 0.99996376, 1),\n",
+       " (30, u'class_text', u'Iris-versicolor', 3.623055e-05, 2),\n",
+       " (30, u'class_text', u'Iris-virginica', 1.4014193e-09, 3),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.99995685, 1),\n",
+       " (31, u'class_text', u'Iris-versicolor', 4.3105167e-05, 2),\n",
+       " (31, u'class_text', u'Iris-virginica', 1.541236e-09, 3),\n",
+       " (49, u'class_text', u'Iris-setosa', 0.99999833, 1),\n",
+       " (49, u'class_text', u'Iris-versicolor', 1.6733742e-06, 2),\n",
+       " (49, u'class_text', u'Iris-virginica', 1.0720992e-11, 3),\n",
+       " (55, u'class_text', u'Iris-versicolor', 0.97456545, 1),\n",
+       " (55, u'class_text', u'Iris-virginica', 0.025385397, 2),\n",
+       " (55, u'class_text', u'Iris-setosa', 4.912654e-05, 3),\n",
+       " (64, u'class_text', u'Iris-versicolor', 0.8837083, 1),\n",
+       " (64, u'class_text', u'Iris-virginica', 0.11627731, 2),\n",
+       " (64, u'class_text', u'Iris-setosa', 1.4444132e-05, 3),\n",
+       " (70, u'class_text', u'Iris-versicolor', 0.9832433, 1),\n",
+       " (70, u'class_text', u'Iris-virginica', 0.016161945, 2),\n",
+       " (70, u'class_text', u'Iris-setosa', 0.0005947249, 3),\n",
+       " (76, u'class_text', u'Iris-versicolor', 0.9934144, 1),\n",
+       " (76, u'class_text', u'Iris-virginica', 0.006202936, 2),\n",
+       " (76, u'class_text', u'Iris-setosa', 0.00038262276, 3),\n",
+       " (82, u'class_text', u'Iris-versicolor', 0.9880006, 1),\n",
+       " (82, u'class_text', u'Iris-virginica', 0.01050145, 2),\n",
+       " (82, u'class_text', u'Iris-setosa', 0.0014980072, 3),\n",
+       " (84, u'class_text', u'Iris-virginica', 0.743757, 1),\n",
+       " (84, u'class_text', u'Iris-versicolor', 0.25624287, 2),\n",
+       " (84, u'class_text', u'Iris-setosa', 1.1804799e-07, 3),\n",
+       " (92, u'class_text', u'Iris-versicolor', 0.9489498, 1),\n",
+       " (92, u'class_text', u'Iris-virginica', 0.050999135, 2),\n",
+       " (92, u'class_text', u'Iris-setosa', 5.1051586e-05, 3),\n",
+       " (98, u'class_text', u'Iris-versicolor', 0.9882598, 1),\n",
+       " (98, u'class_text', u'Iris-virginica', 0.011410431, 2),\n",
+       " (98, u'class_text', u'Iris-setosa', 0.00032975432, 3),\n",
+       " (99, u'class_text', u'Iris-versicolor', 0.7122672, 1),\n",
+       " (99, u'class_text', u'Iris-setosa', 0.2864844, 2),\n",
+       " (99, u'class_text', u'Iris-virginica', 0.0012483773, 3),\n",
+       " (102, u'class_text', u'Iris-virginica', 0.8344315, 1),\n",
+       " (102, u'class_text', u'Iris-versicolor', 0.16556835, 2),\n",
+       " (102, u'class_text', u'Iris-setosa', 4.9313943e-08, 3),\n",
+       " (107, u'class_text', u'Iris-virginica', 0.7617606, 1),\n",
+       " (107, u'class_text', u'Iris-versicolor', 0.23823881, 2),\n",
+       " (107, u'class_text', u'Iris-setosa', 6.2156596e-07, 3),\n",
+       " (114, u'class_text', u'Iris-virginica', 0.85601324, 1),\n",
+       " (114, u'class_text', u'Iris-versicolor', 0.1439867, 2),\n",
+       " (114, u'class_text', u'Iris-setosa', 3.4068247e-08, 3),\n",
+       " (117, u'class_text', u'Iris-virginica', 0.76065344, 1),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.23934652, 2),\n",
+       " (117, u'class_text', u'Iris-setosa', 4.0775706e-08, 3),\n",
+       " (121, u'class_text', u'Iris-virginica', 0.65924823, 1),\n",
+       " (121, u'class_text', u'Iris-versicolor', 0.34075174, 2),\n",
+       " (121, u'class_text', u'Iris-setosa', 3.7877243e-08, 3),\n",
+       " (123, u'class_text', u'Iris-virginica', 0.968423, 1),\n",
+       " (123, u'class_text', u'Iris-versicolor', 0.031577036, 2),\n",
+       " (123, u'class_text', u'Iris-setosa', 1.5606285e-11, 3),\n",
+       " (125, u'class_text', u'Iris-virginica', 0.72842705, 1),\n",
+       " (125, u'class_text', u'Iris-versicolor', 0.2715729, 2),\n",
+       " (125, u'class_text', u'Iris-setosa', 3.7875385e-08, 3),\n",
+       " (127, u'class_text', u'Iris-versicolor', 0.8053533, 1),\n",
+       " (127, u'class_text', u'Iris-virginica', 0.19464317, 2),\n",
+       " (127, u'class_text', u'Iris-setosa', 3.5179064e-06, 3),\n",
+       " (145, u'class_text', u'Iris-virginica', 0.7297866, 1),\n",
+       " (145, u'class_text', u'Iris-versicolor', 0.2702134, 2),\n",
+       " (145, u'class_text', u'Iris-setosa', 2.8784607e-08, 3),\n",
+       " (147, u'class_text', u'Iris-virginica', 0.5341273, 1),\n",
+       " (147, u'class_text', u'Iris-versicolor', 0.4658725, 2),\n",
+       " (147, u'class_text', u'Iris-setosa', 2.3799986e-07, 3),\n",
+       " (148, u'class_text', u'Iris-versicolor', 0.6266347, 1),\n",
+       " (148, u'class_text', u'Iris-virginica', 0.3733647, 2),\n",
+       " (148, u'class_text', u'Iris-setosa', 5.7692125e-07, 3),\n",
+       " (149, u'class_text', u'Iris-virginica', 0.5517554, 1),\n",
+       " (149, u'class_text', u'Iris-versicolor', 0.4482443, 2),\n",
+       " (149, u'class_text', u'Iris-setosa', 3.1108453e-07, 3)]"
+      ]
+     },
+     "execution_count": 30,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_multi_model', -- model\n",
+    "                                   'iris_test',        -- test_table\n",
+    "                                   'id',               -- id column\n",
+    "                                   'attributes',       -- independent var\n",
+    "                                   'iris_predict',     -- output table\n",
+    "                                    'prob',            -- prediction type\n",
+    "                                    FALSE,             -- use gpus\n",
+    "                                    3                  -- mst_key to use\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id, rank;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"warm_start\"></a>\n",
+    "# 3.  Warm start\n",
+    "\n",
+    "Next, use the warm_start parameter to continue learning, using the coefficients from the run above. Note that we don't drop the model table or model summary table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 31,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit_multiple_model</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 31,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
+    "                                              'iris_multi_model',     -- model_output_table\n",
+    "                                              'mst_table',            -- model_selection_table\n",
+    "                                               3,                     -- num_iterations\n",
+    "                                               FALSE,                 -- use gpus\n",
+    "                                              'iris_test_packed',     -- validation dataset\n",
+    "                                               1,                     -- metrics compute frequency\n",
+    "                                               TRUE,                  -- warm start\n",
+    "                                              'Sophie L.',            -- name\n",
+    "                                              'Simple MLP for iris dataset'  -- description\n",
+    "                                             );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 32,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>warm_start</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>iris_multi_model</td>\n",
+       "        <td>iris_multi_model_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>mst_table</td>\n",
+       "        <td>None</td>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>True</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Simple MLP for iris dataset</td>\n",
+       "        <td>2021-03-06 00:55:34.010762</td>\n",
+       "        <td>2021-03-06 00:56:20.576330</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[1, 2, 3]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_test_packed', u'iris_multi_model', u'iris_multi_model_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'mst_table', None, 3, 1, True, u'Sophie L.', u'Simple MLP for iris dataset', datetime.datetime(2021, 3, 6, 0, 55, 34, 10762), datetime.datetime(2021, 3, 6, 0, 56, 20, 576330), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0, [1, 2, 3])]"
+      ]
+     },
+     "execution_count": 32,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View performance of each model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 33,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[12.8246030807495, 28.3149819374084, 43.8511519432068]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.125932246447</td>\n",
+       "        <td>[0.983333349227905, 0.908333361148834, 0.949999988079071]</td>\n",
+       "        <td>[0.0759517326951027, 0.280529856681824, 0.125932246446609]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.262804627419</td>\n",
+       "        <td>[0.966666638851166, 0.933333337306976, 0.966666638851166]</td>\n",
+       "        <td>[0.115140154957771, 0.282798647880554, 0.262804627418518]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[12.3267669677734, 27.5790538787842, 43.3719210624695]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.958333313465</td>\n",
+       "        <td>0.646220803261</td>\n",
+       "        <td>[0.916666686534882, 0.774999976158142, 0.958333313465118]</td>\n",
+       "        <td>[0.760809063911438, 0.70676600933075, 0.646220803260803]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.676706075668</td>\n",
+       "        <td>[0.899999976158142, 0.699999988079071, 0.966666638851166]</td>\n",
+       "        <td>[0.789911270141602, 0.741125166416168, 0.676706075668335]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[13.8655989170074, 29.3921880722046, 45.186311006546]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.161019146442</td>\n",
+       "        <td>[0.608333349227905, 0.975000023841858, 0.966666638851166]</td>\n",
+       "        <td>[0.656926870346069, 0.154457986354828, 0.161019146442413]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.184286847711</td>\n",
+       "        <td>[0.666666686534882, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.60343611240387, 0.166501134634018, 0.184286847710609]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[12.5584180355072, 27.7957689762115, 43.5938129425049]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.925000011921</td>\n",
+       "        <td>0.125614732504</td>\n",
+       "        <td>[0.850000023841858, 0.908333361148834, 0.925000011920929]</td>\n",
+       "        <td>[0.311796188354492, 0.228279903531075, 0.125614732503891]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.205575048923</td>\n",
+       "        <td>[0.699999988079071, 0.899999976158142, 0.933333337306976]</td>\n",
+       "        <td>[0.434732705354691, 0.278642177581787, 0.205575048923492]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[14.3016650676727, 29.8289239406586, 45.6773319244385]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.916666686535</td>\n",
+       "        <td>0.680241525173</td>\n",
+       "        <td>[0.899999976158142, 0.899999976158142, 0.916666686534882]</td>\n",
+       "        <td>[0.75947380065918, 0.717410624027252, 0.680241525173187]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.685820519924</td>\n",
+       "        <td>[0.933333337306976, 0.933333337306976, 0.933333337306976]</td>\n",
+       "        <td>[0.764581918716431, 0.718774557113647, 0.685820519924164]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[13.6457929611206, 29.1624140739441, 44.9534199237823]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.891666650772</td>\n",
+       "        <td>0.590237081051</td>\n",
+       "        <td>[0.824999988079071, 0.783333361148834, 0.891666650772095]</td>\n",
+       "        <td>[0.666068911552429, 0.633061707019806, 0.590237081050873]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.576045572758</td>\n",
+       "        <td>[0.866666674613953, 0.866666674613953, 0.899999976158142]</td>\n",
+       "        <td>[0.645683944225311, 0.608498632907867, 0.576045572757721]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[14.0837008953094, 29.6097829341888, 45.4142129421234]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.916666686535</td>\n",
+       "        <td>0.174454689026</td>\n",
+       "        <td>[0.949999988079071, 0.958333313465118, 0.916666686534882]</td>\n",
+       "        <td>[0.166735425591469, 0.141851797699928, 0.174454689025879]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.219132959843</td>\n",
+       "        <td>[0.966666638851166, 0.933333337306976, 0.899999976158142]</td>\n",
+       "        <td>[0.186790466308594, 0.176578417420387, 0.219132959842682]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[13.1594960689545, 28.5860660076141, 44.1881170272827]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.285291582346</td>\n",
+       "        <td>[0.774999976158142, 0.949999988079071, 0.866666674613953]</td>\n",
+       "        <td>[0.441815197467804, 0.140827313065529, 0.285291582345963]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.246576815844</td>\n",
+       "        <td>[0.766666650772095, 0.966666638851166, 0.866666674613953]</td>\n",
+       "        <td>[0.4128278195858, 0.146319955587387, 0.246576815843582]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[14.5546190738678, 30.0798380374908, 45.94082903862]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.850000023842</td>\n",
+       "        <td>0.675731360912</td>\n",
+       "        <td>[0.791666686534882, 0.841666638851166, 0.850000023841858]</td>\n",
+       "        <td>[0.746130049228668, 0.706377267837524, 0.675731360912323]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.650432705879</td>\n",
+       "        <td>[0.866666674613953, 0.866666674613953, 0.866666674613953]</td>\n",
+       "        <td>[0.712817847728729, 0.677974581718445, 0.650432705879211]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[15.3575170040131, 30.5435180664062, 46.5635209083557]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.658333361149</td>\n",
+       "        <td>0.45723798871</td>\n",
+       "        <td>[0.658333361148834, 0.683333337306976, 0.658333361148834]</td>\n",
+       "        <td>[0.457635939121246, 0.455960959196091, 0.457237988710403]</td>\n",
+       "        <td>0.699999988079</td>\n",
+       "        <td>0.48275628686</td>\n",
+       "        <td>[0.699999988079071, 0.600000023841858, 0.699999988079071]</td>\n",
+       "        <td>[0.48207613825798, 0.491984754800797, 0.482756286859512]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[14.8466219902039, 30.2953569889069, 46.1656670570374]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.683333337307</td>\n",
+       "        <td>0.456283688545</td>\n",
+       "        <td>[0.925000011920929, 0.899999976158142, 0.683333337306976]</td>\n",
+       "        <td>[0.224153310060501, 0.295417010784149, 0.456283688545227]</td>\n",
+       "        <td>0.600000023842</td>\n",
+       "        <td>0.494575560093</td>\n",
+       "        <td>[0.966666638851166, 0.899999976158142, 0.600000023841858]</td>\n",
+       "        <td>[0.227903217077255, 0.345975488424301, 0.494575560092926]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[13.4095330238342, 28.938658952713, 44.7153990268707]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.691666662693</td>\n",
+       "        <td>0.528191685677</td>\n",
+       "        <td>[0.708333313465118, 0.966666638851166, 0.691666662693024]</td>\n",
+       "        <td>[0.395545929670334, 0.100506067276001, 0.528191685676575]</td>\n",
+       "        <td>0.566666662693</td>\n",
+       "        <td>0.720313131809</td>\n",
+       "        <td>[0.633333325386047, 0.966666638851166, 0.566666662693024]</td>\n",
+       "        <td>[0.508394777774811, 0.130626574158669, 0.720313131809235]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [12.8246030807495, 28.3149819374084, 43.8511519432068], [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.125932246446609, [0.983333349227905, 0.908333361148834, 0.949999988079071], [0.0759517326951027, 0.280529856681824, 0.125932246446609], 0.966666638851166, 0.262804627418518, [0.966666638851166, 0.933333337306976, 0.966666638851166], [0.115140154957771, 0.282798647880554, 0.262804627418518]),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [12.3267669677734, 27.5790538787842, 43.3719210624695], [u'accuracy'], u'categorical_crossentropy', 0.958333313465118, 0.646220803260803, [0.916666686534882, 0.774999976158142, 0.958333313465118], [0.760809063911438, 0.70676600933075, 0.646220803260803], 0.966666638851166, 0.676706075668335, [0.899999976158142, 0.699999988079071, 0.966666638851166], [0.789911270141602, 0.741125166416168, 0.676706075668335]),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [13.8655989170074, 29.3921880722046, 45.186311006546], [u'accuracy'], u'categorical_crossentropy', 0.966666638851166, 0.161019146442413, [0.608333349227905, 0.975000023841858, 0.966666638851166], [0.656926870346069, 0.154457986354828, 0.161019146442413], 0.966666638851166, 0.184286847710609, [0.666666686534882, 0.966666638851166, 0.966666638851166], [0.60343611240387, 0.166501134634018, 0.184286847710609]),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [12.5584180355072, 27.7957689762115, 43.5938129425049], [u'accuracy'], u'categorical_crossentropy', 0.925000011920929, 0.125614732503891, [0.850000023841858, 0.908333361148834, 0.925000011920929], [0.311796188354492, 0.228279903531075, 0.125614732503891], 0.933333337306976, 0.205575048923492, [0.699999988079071, 0.899999976158142, 0.933333337306976], [0.434732705354691, 0.278642177581787, 0.205575048923492]),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [14.3016650676727, 29.8289239406586, 45.6773319244385], [u'accuracy'], u'categorical_crossentropy', 0.916666686534882, 0.680241525173187, [0.899999976158142, 0.899999976158142, 0.916666686534882], [0.75947380065918, 0.717410624027252, 0.680241525173187], 0.933333337306976, 0.685820519924164, [0.933333337306976, 0.933333337306976, 0.933333337306976], [0.764581918716431, 0.718774557113647, 0.685820519924164]),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [13.6457929611206, 29.1624140739441, 44.9534199237823], [u'accuracy'], u'categorical_crossentropy', 0.891666650772095, 0.590237081050873, [0.824999988079071, 0.783333361148834, 0.891666650772095], [0.666068911552429, 0.633061707019806, 0.590237081050873], 0.899999976158142, 0.576045572757721, [0.866666674613953, 0.866666674613953, 0.899999976158142], [0.645683944225311, 0.608498632907867, 0.576045572757721]),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [14.0837008953094, 29.6097829341888, 45.4142129421234], [u'accuracy'], u'categorical_crossentropy', 0.916666686534882, 0.174454689025879, [0.949999988079071, 0.958333313465118, 0.916666686534882], [0.166735425591469, 0.141851797699928, 0.174454689025879], 0.899999976158142, 0.219132959842682, [0.966666638851166, 0.933333337306976, 0.899999976158142], [0.186790466308594, 0.176578417420387, 0.219132959842682]),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [13.1594960689545, 28.5860660076141, 44.1881170272827], [u'accuracy'], u'categorical_crossentropy', 0.866666674613953, 0.285291582345963, [0.774999976158142, 0.949999988079071, 0.866666674613953], [0.441815197467804, 0.140827313065529, 0.285291582345963], 0.866666674613953, 0.246576815843582, [0.766666650772095, 0.966666638851166, 0.866666674613953], [0.4128278195858, 0.146319955587387, 0.246576815843582]),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [14.5546190738678, 30.0798380374908, 45.94082903862], [u'accuracy'], u'categorical_crossentropy', 0.850000023841858, 0.675731360912323, [0.791666686534882, 0.841666638851166, 0.850000023841858], [0.746130049228668, 0.706377267837524, 0.675731360912323], 0.866666674613953, 0.650432705879211, [0.866666674613953, 0.866666674613953, 0.866666674613953], [0.712817847728729, 0.677974581718445, 0.650432705879211]),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [15.3575170040131, 30.5435180664062, 46.5635209083557], [u'accuracy'], u'categorical_crossentropy', 0.658333361148834, 0.457237988710403, [0.658333361148834, 0.683333337306976, 0.658333361148834], [0.457635939121246, 0.455960959196091, 0.457237988710403], 0.699999988079071, 0.482756286859512, [0.699999988079071, 0.600000023841858, 0.699999988079071], [0.48207613825798, 0.491984754800797, 0.482756286859512]),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [14.8466219902039, 30.2953569889069, 46.1656670570374], [u'accuracy'], u'categorical_crossentropy', 0.683333337306976, 0.456283688545227, [0.925000011920929, 0.899999976158142, 0.683333337306976], [0.224153310060501, 0.295417010784149, 0.456283688545227], 0.600000023841858, 0.494575560092926, [0.966666638851166, 0.899999976158142, 0.600000023841858], [0.227903217077255, 0.345975488424301, 0.494575560092926]),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [13.4095330238342, 28.938658952713, 44.7153990268707], [u'accuracy'], u'categorical_crossentropy', 0.691666662693024, 0.528191685676575, [0.708333313465118, 0.966666638851166, 0.691666662693024], [0.395545929670334, 0.100506067276001, 0.528191685676575], 0.566666662693024, 0.720313131809235, [0.633333325386047, 0.966666638851166, 0.566666662693024], [0.508394777774811, 0.130626574158669, 0.720313131809235])]"
+      ]
+     },
+     "execution_count": 33,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_info ORDER BY validation_metrics_final DESC;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Plot validation results:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 34,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "7 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM iris_multi_model_info ORDER BY validation_loss ASC LIMIT 7;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "df_summary = %sql SELECT * FROM iris_multi_model_summary;\n",
+    "df_summary = df_summary.DataFrame()\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Iteration')\n",
+    "ax_metric.set_ylabel('Metric')\n",
+    "ax_metric.set_title('Validation metric curve')\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Iteration')\n",
+    "ax_loss.set_ylabel('Loss')\n",
+    "ax_loss.set_title('Validation loss curve')\n",
+    "\n",
+    "iters = df_summary['metrics_iters'][0]\n",
+    "\n",
+    "for mst_key in df_results['mst_key']:\n",
+    "    df_output_info = %sql SELECT validation_metrics,validation_loss FROM iris_multi_model_info WHERE mst_key = $mst_key\n",
+    "    df_output_info = df_output_info.DataFrame()\n",
+    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
+    "    validation_loss = df_output_info['validation_loss'][0]\n",
+    "    \n",
+    "    ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
+    "    ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Train-multiple-models/AutoML-MLP-v1.ipynb b/community-artifacts/Deep-learning/Train-multiple-models/AutoML-MLP-v1.ipynb
new file mode 100755
index 0000000..c679b3c
--- /dev/null
+++ b/community-artifacts/Deep-learning/Train-multiple-models/AutoML-MLP-v1.ipynb
@@ -0,0 +1,6937 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# AutoML for Multilayer Perceptron\n",
+    "\n",
+    "E2E classification example using autoML methods for optimizing hyperparameters and model architectures.\n",
+    "\n",
+    "Deep learning works best on very large datasets, but that is not convenient for a quick introduction to the syntax.  So in this workbook we use the well known iris data set from https://archive.ics.uci.edu/ml/datasets/iris to help get you started.  It is similar to the example in user docs http://madlib.apache.org/docs/latest/index.html\n",
+    "\n",
+    "For more realistic examples please refer to the deep learning notebooks at\n",
+    "https://github.com/apache/madlib-site/tree/asf-site/community-artifacts\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#create_input_data\">1. Create input data</a>\n",
+    "\n",
+    "<a href=\"#pp\">2. Call preprocessor for deep learning</a>\n",
+    "\n",
+    "<a href=\"#load\">3. Define and load model architecture</a>\n",
+    "\n",
+    "<a href=\"#hyperband\">4. Hyperband</a>\n",
+    "\n",
+    "<a href=\"#hyperopt\">5. Hyperopt</a>\n",
+    "\n",
+    "<a href=\"#pred\">6. Predict</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP (PM demo machine) - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class\"></a>\n",
+    "# Classification\n",
+    "\n",
+    "<a id=\"create_input_data\"></a>\n",
+    "# 1.  Create input data\n",
+    "\n",
+    "Load iris data set."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "150 rows affected.\n",
+      "150 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>attributes</th>\n",
+       "        <th>class_text</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>[Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>21</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>22</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>25</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>35</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>36</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>37</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>39</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>40</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>42</td>\n",
+       "        <td>[Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>44</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>45</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>46</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>47</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>48</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>[Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>50</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>[Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>53</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>58</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>59</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>62</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>65</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>69</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>79</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>86</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>87</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>88</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>91</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>95</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>97</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>[Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>105</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>106</td>\n",
+       "        <td>[Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>[Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>109</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>110</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>118</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>124</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>129</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>130</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>[Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>[Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>133</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>139</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>140</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>141</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>142</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (2, [Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (3, [Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (4, [Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (5, [Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (6, [Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (7, [Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (8, [Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (9, [Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (10, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (11, [Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (12, [Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (13, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (14, [Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (15, [Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (16, [Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (17, [Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (18, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (19, [Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (20, [Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (21, [Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (22, [Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (23, [Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (24, [Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')], u'Iris-setosa'),\n",
+       " (25, [Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (26, [Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (27, [Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (28, [Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (29, [Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (30, [Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (31, [Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (32, [Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (33, [Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (34, [Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (35, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (36, [Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (37, [Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (38, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (39, [Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (40, [Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (41, [Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (42, [Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (43, [Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (44, [Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')], u'Iris-setosa'),\n",
+       " (45, [Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (46, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (47, [Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (48, [Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (49, [Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (50, [Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (51, [Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (52, [Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (53, [Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (54, [Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (55, [Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (56, [Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (57, [Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (58, [Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (59, [Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (60, [Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (61, [Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (62, [Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (63, [Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (64, [Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (65, [Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (66, [Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (67, [Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (68, [Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (69, [Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (70, [Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (71, [Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')], u'Iris-versicolor'),\n",
+       " (72, [Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (73, [Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (74, [Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (75, [Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (76, [Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (77, [Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (78, [Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')], u'Iris-versicolor'),\n",
+       " (79, [Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (80, [Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (81, [Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (82, [Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (83, [Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (84, [Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (85, [Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (86, [Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (87, [Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (88, [Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (89, [Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (90, [Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (91, [Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (92, [Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (93, [Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (94, [Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (95, [Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (96, [Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (97, [Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (98, [Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (99, [Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (100, [Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (101, [Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (102, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (103, [Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (104, [Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (105, [Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (106, [Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (107, [Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')], u'Iris-virginica'),\n",
+       " (108, [Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (109, [Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (110, [Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (111, [Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (112, [Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (113, [Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (114, [Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (115, [Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (116, [Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (117, [Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (118, [Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (119, [Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (120, [Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (121, [Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (122, [Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (123, [Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (124, [Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (125, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (126, [Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (127, [Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (128, [Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (129, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (130, [Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')], u'Iris-virginica'),\n",
+       " (131, [Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (132, [Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (133, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (134, [Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (135, [Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')], u'Iris-virginica'),\n",
+       " (136, [Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (137, [Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (138, [Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (139, [Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (140, [Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (141, [Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (142, [Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (143, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (144, [Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (145, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (146, [Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (147, [Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (148, [Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (149, [Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (150, [Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')], u'Iris-virginica')]"
+      ]
+     },
+     "execution_count": 4,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql \n",
+    "DROP TABLE IF EXISTS iris_data;\n",
+    "\n",
+    "CREATE TABLE iris_data(\n",
+    "    id serial,\n",
+    "    attributes numeric[],\n",
+    "    class_text varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO iris_data(id, attributes, class_text) VALUES\n",
+    "(1,ARRAY[5.1,3.5,1.4,0.2],'Iris-setosa'),\n",
+    "(2,ARRAY[4.9,3.0,1.4,0.2],'Iris-setosa'),\n",
+    "(3,ARRAY[4.7,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(4,ARRAY[4.6,3.1,1.5,0.2],'Iris-setosa'),\n",
+    "(5,ARRAY[5.0,3.6,1.4,0.2],'Iris-setosa'),\n",
+    "(6,ARRAY[5.4,3.9,1.7,0.4],'Iris-setosa'),\n",
+    "(7,ARRAY[4.6,3.4,1.4,0.3],'Iris-setosa'),\n",
+    "(8,ARRAY[5.0,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(9,ARRAY[4.4,2.9,1.4,0.2],'Iris-setosa'),\n",
+    "(10,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(11,ARRAY[5.4,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(12,ARRAY[4.8,3.4,1.6,0.2],'Iris-setosa'),\n",
+    "(13,ARRAY[4.8,3.0,1.4,0.1],'Iris-setosa'),\n",
+    "(14,ARRAY[4.3,3.0,1.1,0.1],'Iris-setosa'),\n",
+    "(15,ARRAY[5.8,4.0,1.2,0.2],'Iris-setosa'),\n",
+    "(16,ARRAY[5.7,4.4,1.5,0.4],'Iris-setosa'),\n",
+    "(17,ARRAY[5.4,3.9,1.3,0.4],'Iris-setosa'),\n",
+    "(18,ARRAY[5.1,3.5,1.4,0.3],'Iris-setosa'),\n",
+    "(19,ARRAY[5.7,3.8,1.7,0.3],'Iris-setosa'),\n",
+    "(20,ARRAY[5.1,3.8,1.5,0.3],'Iris-setosa'),\n",
+    "(21,ARRAY[5.4,3.4,1.7,0.2],'Iris-setosa'),\n",
+    "(22,ARRAY[5.1,3.7,1.5,0.4],'Iris-setosa'),\n",
+    "(23,ARRAY[4.6,3.6,1.0,0.2],'Iris-setosa'),\n",
+    "(24,ARRAY[5.1,3.3,1.7,0.5],'Iris-setosa'),\n",
+    "(25,ARRAY[4.8,3.4,1.9,0.2],'Iris-setosa'),\n",
+    "(26,ARRAY[5.0,3.0,1.6,0.2],'Iris-setosa'),\n",
+    "(27,ARRAY[5.0,3.4,1.6,0.4],'Iris-setosa'),\n",
+    "(28,ARRAY[5.2,3.5,1.5,0.2],'Iris-setosa'),\n",
+    "(29,ARRAY[5.2,3.4,1.4,0.2],'Iris-setosa'),\n",
+    "(30,ARRAY[4.7,3.2,1.6,0.2],'Iris-setosa'),\n",
+    "(31,ARRAY[4.8,3.1,1.6,0.2],'Iris-setosa'),\n",
+    "(32,ARRAY[5.4,3.4,1.5,0.4],'Iris-setosa'),\n",
+    "(33,ARRAY[5.2,4.1,1.5,0.1],'Iris-setosa'),\n",
+    "(34,ARRAY[5.5,4.2,1.4,0.2],'Iris-setosa'),\n",
+    "(35,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(36,ARRAY[5.0,3.2,1.2,0.2],'Iris-setosa'),\n",
+    "(37,ARRAY[5.5,3.5,1.3,0.2],'Iris-setosa'),\n",
+    "(38,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(39,ARRAY[4.4,3.0,1.3,0.2],'Iris-setosa'),\n",
+    "(40,ARRAY[5.1,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(41,ARRAY[5.0,3.5,1.3,0.3],'Iris-setosa'),\n",
+    "(42,ARRAY[4.5,2.3,1.3,0.3],'Iris-setosa'),\n",
+    "(43,ARRAY[4.4,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(44,ARRAY[5.0,3.5,1.6,0.6],'Iris-setosa'),\n",
+    "(45,ARRAY[5.1,3.8,1.9,0.4],'Iris-setosa'),\n",
+    "(46,ARRAY[4.8,3.0,1.4,0.3],'Iris-setosa'),\n",
+    "(47,ARRAY[5.1,3.8,1.6,0.2],'Iris-setosa'),\n",
+    "(48,ARRAY[4.6,3.2,1.4,0.2],'Iris-setosa'),\n",
+    "(49,ARRAY[5.3,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(50,ARRAY[5.0,3.3,1.4,0.2],'Iris-setosa'),\n",
+    "(51,ARRAY[7.0,3.2,4.7,1.4],'Iris-versicolor'),\n",
+    "(52,ARRAY[6.4,3.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(53,ARRAY[6.9,3.1,4.9,1.5],'Iris-versicolor'),\n",
+    "(54,ARRAY[5.5,2.3,4.0,1.3],'Iris-versicolor'),\n",
+    "(55,ARRAY[6.5,2.8,4.6,1.5],'Iris-versicolor'),\n",
+    "(56,ARRAY[5.7,2.8,4.5,1.3],'Iris-versicolor'),\n",
+    "(57,ARRAY[6.3,3.3,4.7,1.6],'Iris-versicolor'),\n",
+    "(58,ARRAY[4.9,2.4,3.3,1.0],'Iris-versicolor'),\n",
+    "(59,ARRAY[6.6,2.9,4.6,1.3],'Iris-versicolor'),\n",
+    "(60,ARRAY[5.2,2.7,3.9,1.4],'Iris-versicolor'),\n",
+    "(61,ARRAY[5.0,2.0,3.5,1.0],'Iris-versicolor'),\n",
+    "(62,ARRAY[5.9,3.0,4.2,1.5],'Iris-versicolor'),\n",
+    "(63,ARRAY[6.0,2.2,4.0,1.0],'Iris-versicolor'),\n",
+    "(64,ARRAY[6.1,2.9,4.7,1.4],'Iris-versicolor'),\n",
+    "(65,ARRAY[5.6,2.9,3.6,1.3],'Iris-versicolor'),\n",
+    "(66,ARRAY[6.7,3.1,4.4,1.4],'Iris-versicolor'),\n",
+    "(67,ARRAY[5.6,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(68,ARRAY[5.8,2.7,4.1,1.0],'Iris-versicolor'),\n",
+    "(69,ARRAY[6.2,2.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(70,ARRAY[5.6,2.5,3.9,1.1],'Iris-versicolor'),\n",
+    "(71,ARRAY[5.9,3.2,4.8,1.8],'Iris-versicolor'),\n",
+    "(72,ARRAY[6.1,2.8,4.0,1.3],'Iris-versicolor'),\n",
+    "(73,ARRAY[6.3,2.5,4.9,1.5],'Iris-versicolor'),\n",
+    "(74,ARRAY[6.1,2.8,4.7,1.2],'Iris-versicolor'),\n",
+    "(75,ARRAY[6.4,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(76,ARRAY[6.6,3.0,4.4,1.4],'Iris-versicolor'),\n",
+    "(77,ARRAY[6.8,2.8,4.8,1.4],'Iris-versicolor'),\n",
+    "(78,ARRAY[6.7,3.0,5.0,1.7],'Iris-versicolor'),\n",
+    "(79,ARRAY[6.0,2.9,4.5,1.5],'Iris-versicolor'),\n",
+    "(80,ARRAY[5.7,2.6,3.5,1.0],'Iris-versicolor'),\n",
+    "(81,ARRAY[5.5,2.4,3.8,1.1],'Iris-versicolor'),\n",
+    "(82,ARRAY[5.5,2.4,3.7,1.0],'Iris-versicolor'),\n",
+    "(83,ARRAY[5.8,2.7,3.9,1.2],'Iris-versicolor'),\n",
+    "(84,ARRAY[6.0,2.7,5.1,1.6],'Iris-versicolor'),\n",
+    "(85,ARRAY[5.4,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(86,ARRAY[6.0,3.4,4.5,1.6],'Iris-versicolor'),\n",
+    "(87,ARRAY[6.7,3.1,4.7,1.5],'Iris-versicolor'),\n",
+    "(88,ARRAY[6.3,2.3,4.4,1.3],'Iris-versicolor'),\n",
+    "(89,ARRAY[5.6,3.0,4.1,1.3],'Iris-versicolor'),\n",
+    "(90,ARRAY[5.5,2.5,4.0,1.3],'Iris-versicolor'),\n",
+    "(91,ARRAY[5.5,2.6,4.4,1.2],'Iris-versicolor'),\n",
+    "(92,ARRAY[6.1,3.0,4.6,1.4],'Iris-versicolor'),\n",
+    "(93,ARRAY[5.8,2.6,4.0,1.2],'Iris-versicolor'),\n",
+    "(94,ARRAY[5.0,2.3,3.3,1.0],'Iris-versicolor'),\n",
+    "(95,ARRAY[5.6,2.7,4.2,1.3],'Iris-versicolor'),\n",
+    "(96,ARRAY[5.7,3.0,4.2,1.2],'Iris-versicolor'),\n",
+    "(97,ARRAY[5.7,2.9,4.2,1.3],'Iris-versicolor'),\n",
+    "(98,ARRAY[6.2,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(99,ARRAY[5.1,2.5,3.0,1.1],'Iris-versicolor'),\n",
+    "(100,ARRAY[5.7,2.8,4.1,1.3],'Iris-versicolor'),\n",
+    "(101,ARRAY[6.3,3.3,6.0,2.5],'Iris-virginica'),\n",
+    "(102,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(103,ARRAY[7.1,3.0,5.9,2.1],'Iris-virginica'),\n",
+    "(104,ARRAY[6.3,2.9,5.6,1.8],'Iris-virginica'),\n",
+    "(105,ARRAY[6.5,3.0,5.8,2.2],'Iris-virginica'),\n",
+    "(106,ARRAY[7.6,3.0,6.6,2.1],'Iris-virginica'),\n",
+    "(107,ARRAY[4.9,2.5,4.5,1.7],'Iris-virginica'),\n",
+    "(108,ARRAY[7.3,2.9,6.3,1.8],'Iris-virginica'),\n",
+    "(109,ARRAY[6.7,2.5,5.8,1.8],'Iris-virginica'),\n",
+    "(110,ARRAY[7.2,3.6,6.1,2.5],'Iris-virginica'),\n",
+    "(111,ARRAY[6.5,3.2,5.1,2.0],'Iris-virginica'),\n",
+    "(112,ARRAY[6.4,2.7,5.3,1.9],'Iris-virginica'),\n",
+    "(113,ARRAY[6.8,3.0,5.5,2.1],'Iris-virginica'),\n",
+    "(114,ARRAY[5.7,2.5,5.0,2.0],'Iris-virginica'),\n",
+    "(115,ARRAY[5.8,2.8,5.1,2.4],'Iris-virginica'),\n",
+    "(116,ARRAY[6.4,3.2,5.3,2.3],'Iris-virginica'),\n",
+    "(117,ARRAY[6.5,3.0,5.5,1.8],'Iris-virginica'),\n",
+    "(118,ARRAY[7.7,3.8,6.7,2.2],'Iris-virginica'),\n",
+    "(119,ARRAY[7.7,2.6,6.9,2.3],'Iris-virginica'),\n",
+    "(120,ARRAY[6.0,2.2,5.0,1.5],'Iris-virginica'),\n",
+    "(121,ARRAY[6.9,3.2,5.7,2.3],'Iris-virginica'),\n",
+    "(122,ARRAY[5.6,2.8,4.9,2.0],'Iris-virginica'),\n",
+    "(123,ARRAY[7.7,2.8,6.7,2.0],'Iris-virginica'),\n",
+    "(124,ARRAY[6.3,2.7,4.9,1.8],'Iris-virginica'),\n",
+    "(125,ARRAY[6.7,3.3,5.7,2.1],'Iris-virginica'),\n",
+    "(126,ARRAY[7.2,3.2,6.0,1.8],'Iris-virginica'),\n",
+    "(127,ARRAY[6.2,2.8,4.8,1.8],'Iris-virginica'),\n",
+    "(128,ARRAY[6.1,3.0,4.9,1.8],'Iris-virginica'),\n",
+    "(129,ARRAY[6.4,2.8,5.6,2.1],'Iris-virginica'),\n",
+    "(130,ARRAY[7.2,3.0,5.8,1.6],'Iris-virginica'),\n",
+    "(131,ARRAY[7.4,2.8,6.1,1.9],'Iris-virginica'),\n",
+    "(132,ARRAY[7.9,3.8,6.4,2.0],'Iris-virginica'),\n",
+    "(133,ARRAY[6.4,2.8,5.6,2.2],'Iris-virginica'),\n",
+    "(134,ARRAY[6.3,2.8,5.1,1.5],'Iris-virginica'),\n",
+    "(135,ARRAY[6.1,2.6,5.6,1.4],'Iris-virginica'),\n",
+    "(136,ARRAY[7.7,3.0,6.1,2.3],'Iris-virginica'),\n",
+    "(137,ARRAY[6.3,3.4,5.6,2.4],'Iris-virginica'),\n",
+    "(138,ARRAY[6.4,3.1,5.5,1.8],'Iris-virginica'),\n",
+    "(139,ARRAY[6.0,3.0,4.8,1.8],'Iris-virginica'),\n",
+    "(140,ARRAY[6.9,3.1,5.4,2.1],'Iris-virginica'),\n",
+    "(141,ARRAY[6.7,3.1,5.6,2.4],'Iris-virginica'),\n",
+    "(142,ARRAY[6.9,3.1,5.1,2.3],'Iris-virginica'),\n",
+    "(143,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(144,ARRAY[6.8,3.2,5.9,2.3],'Iris-virginica'),\n",
+    "(145,ARRAY[6.7,3.3,5.7,2.5],'Iris-virginica'),\n",
+    "(146,ARRAY[6.7,3.0,5.2,2.3],'Iris-virginica'),\n",
+    "(147,ARRAY[6.3,2.5,5.0,1.9],'Iris-virginica'),\n",
+    "(148,ARRAY[6.5,3.0,5.2,2.0],'Iris-virginica'),\n",
+    "(149,ARRAY[6.2,3.4,5.4,2.3],'Iris-virginica'),\n",
+    "(150,ARRAY[5.9,3.0,5.1,1.8],'Iris-virginica');\n",
+    "\n",
+    "SELECT * FROM iris_data ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Create a test/validation dataset from the training data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(120L,)]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train, iris_test;\n",
+    "\n",
+    "-- Set seed so results are reproducible\n",
+    "SELECT setseed(0);\n",
+    "\n",
+    "SELECT madlib.train_test_split('iris_data',     -- Source table\n",
+    "                               'iris',          -- Output table root name\n",
+    "                                0.8,            -- Train proportion\n",
+    "                                NULL,           -- Test proportion (0.2)\n",
+    "                                NULL,           -- Strata definition\n",
+    "                                NULL,           -- Output all columns\n",
+    "                                NULL,           -- Sample without replacement\n",
+    "                                TRUE            -- Separate output tables\n",
+    "                              );\n",
+    "\n",
+    "SELECT COUNT(*) FROM iris_train;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pp\"></a>\n",
+    "# 2. Call preprocessor for deep learning\n",
+    "Training dataset (uses training preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>attributes_shape</th>\n",
+       "        <th>class_text_shape</th>\n",
+       "        <th>buffer_id</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[60, 4]</td>\n",
+       "        <td>[60, 3]</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[60, 4]</td>\n",
+       "        <td>[60, 3]</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[([60, 4], [60, 3], 0), ([60, 4], [60, 3], 1)]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train_packed, iris_train_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('iris_train',         -- Source table\n",
+    "                                       'iris_train_packed',  -- Output table\n",
+    "                                       'class_text',        -- Dependent variable\n",
+    "                                       'attributes'         -- Independent variable\n",
+    "                                        ); \n",
+    "\n",
+    "SELECT attributes_shape, class_text_shape, buffer_id FROM iris_train_packed ORDER BY buffer_id;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train</td>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>60</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train', u'iris_train_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 60, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_train_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Validation dataset (uses validation preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>attributes_shape</th>\n",
+       "        <th>class_text_shape</th>\n",
+       "        <th>buffer_id</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 4]</td>\n",
+       "        <td>[15, 3]</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 4]</td>\n",
+       "        <td>[15, 3]</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[([15, 4], [15, 3], 0), ([15, 4], [15, 3], 1)]"
+      ]
+     },
+     "execution_count": 8,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_test_packed, iris_test_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('iris_test',          -- Source table\n",
+    "                                         'iris_test_packed',   -- Output table\n",
+    "                                         'class_text',         -- Dependent variable\n",
+    "                                         'attributes',         -- Independent variable\n",
+    "                                         'iris_train_packed'   -- From training preprocessor step\n",
+    "                                          ); \n",
+    "\n",
+    "SELECT attributes_shape, class_text_shape, buffer_id FROM iris_test_packed ORDER BY buffer_id;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_test</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>15</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_test', u'iris_test_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 15, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 9,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_test_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load\"></a>\n",
+    "# 3. Define and load model architecture\n",
+    "Import Keras libraries"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 1 hidden layer:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "WARNING:tensorflow:From /Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/tensorflow/python/ops/init_ops.py:1251: calling __init__ (from tensorflow.python.ops.init_ops) with dtype is deprecated and will be removed in a future version.\n",
+      "Instructions for updating:\n",
+      "Call initializer instance with the dtype argument instead of passing it to the constructor\n",
+      "Model: \"sequential\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense (Dense)                (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_1 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_2 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 193\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model1 = Sequential()\n",
+    "model1.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model1.add(Dense(10, activation='relu'))\n",
+    "model1.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model1.summary();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 12,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 12,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model1.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 2 hidden layers:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 13,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_1\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_3 (Dense)              (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_4 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_5 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_6 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 303\n",
+      "Trainable params: 303\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model2 = Sequential()\n",
+    "model2.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model2.summary();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 14,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_1\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 14,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model2.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 15,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>model_weights</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>__internal_madlib_id__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>MLP with 1 hidden layer</td>\n",
+       "        <td>__madlib_temp_71395301_1614988659_10232289__</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>MLP with 2 hidden layers</td>\n",
+       "        <td>__madlib_temp_60560187_1614988660_9612153__</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'MLP with 1 hidden layer', u'__madlib_temp_71395301_1614988659_10232289__'),\n",
+       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1835 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'MLP with 2 hidden layers', u'__madlib_temp_60560187_1614988660_9612153__')]"
+      ]
+     },
+     "execution_count": 15,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS model_arch_library;\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Sophie',              -- Name\n",
+    "                               'MLP with 1 hidden layer'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Maria',               -- Name\n",
+    "                               'MLP with 2 hidden layers'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT * FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"hyperband\"></a>\n",
+    "# 4.  Hyperband"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Print schedule for run:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 16,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "6 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>s</th>\n",
+       "        <th>i</th>\n",
+       "        <th>n_i</th>\n",
+       "        <th>r_i</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>0</td>\n",
+       "        <td>9</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>3</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>9</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>0</td>\n",
+       "        <td>3</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>9</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>0</td>\n",
+       "        <td>0</td>\n",
+       "        <td>3</td>\n",
+       "        <td>9</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2, 0, 9, 1),\n",
+       " (2, 1, 3, 3),\n",
+       " (2, 2, 1, 9),\n",
+       " (1, 0, 3, 3),\n",
+       " (1, 1, 1, 9),\n",
+       " (0, 0, 3, 9)]"
+      ]
+     },
+     "execution_count": 16,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS hb_schedule;\n",
+    "SELECT madlib.hyperband_schedule ('hb_schedule', \n",
+    "                                   9,\n",
+    "                                   3,\n",
+    "                                   0);\n",
+    "SELECT * FROM hb_schedule ORDER BY s DESC, i;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 17,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_automl</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 17,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS automl_output, automl_output_info, automl_output_summary, automl_mst_table, automl_mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_automl('iris_train_packed',                -- source table\n",
+    "                                  'automl_output',                    -- model output table\n",
+    "                                  'model_arch_library',               -- model architecture table\n",
+    "                                  'automl_mst_table',                 -- model selection output table\n",
+    "                                  ARRAY[1,2],                         -- model IDs\n",
+    "                                  $${\n",
+    "                                      'loss': ['categorical_crossentropy'], \n",
+    "                                      'optimizer_params_list': [ \n",
+    "                                          {'optimizer': ['Adam'],'lr': [0.001, 0.1, 'log']},\n",
+    "                                          {'optimizer': ['RMSprop'],'lr': [0.001, 0.1, 'log']}\n",
+    "                                      ],\n",
+    "                                      'metrics': ['accuracy']\n",
+    "                                  } $$,                               -- compile param grid\n",
+    "                                  $${'batch_size': [4, 8], 'epochs': [1]}$$,  -- fit params grid\n",
+    "                                  'hyperband',                        -- autoML method\n",
+    "                                  'R=9, eta=3, skip_last=0',          -- autoML params\n",
+    "                                  NULL,                               -- random state\n",
+    "                                  NULL,                               -- object table\n",
+    "                                  FALSE,                              -- use GPUs\n",
+    "                                  'iris_test_packed',                 -- validation table\n",
+    "                                  1,                                  -- metrics compute freq\n",
+    "                                  NULL,                               -- name\n",
+    "                                  NULL);                              -- descr"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 18,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>automl_method</th>\n",
+       "        <th>automl_params</th>\n",
+       "        <th>random_state</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>use_gpus</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>automl_output</td>\n",
+       "        <td>automl_output_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>automl_mst_table</td>\n",
+       "        <td>hyperband</td>\n",
+       "        <td>R=9, eta=3, skip_last=0</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>False</td>\n",
+       "        <td>1</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2021-03-05 23:57:44</td>\n",
+       "        <td>2021-03-05 23:59:24</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_test_packed', u'automl_output', u'automl_output_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'automl_mst_table', u'hyperband', u'R=9, eta=3, skip_last=0', None, None, False, 1, None, None, datetime.datetime(2021, 3, 5, 23, 57, 44), datetime.datetime(2021, 3, 5, 23, 59, 24), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0)]"
+      ]
+     },
+     "execution_count": 18,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM automl_output_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View results for each model"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 19,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "15 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>s</th>\n",
+       "        <th>i</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.04232194170481019)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[21.911346912384, 29.2674539089203, 36.8268938064575, 44.9022789001465, 51.1760609149933, 57.6593999862671, 64.184476852417, 70.5566418170929, 77.0253269672394, 83.4826798439026, 90.1138219833374, 96.4566838741302]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.975000023842</td>\n",
+       "        <td>0.0775080993772</td>\n",
+       "        <td>[0.791666686534882, 0.608333349227905, 0.966666638851166, 0.975000023841858, 0.966666638851166, 0.800000011920929, 0.975000023841858, 0.683333337306976, 0.733333349227905, 0.949999988079071, 0.949999988079071, 0.975000023841858]</td>\n",
+       "        <td>[0.374035209417343, 0.732228577136993, 0.170820266008377, 0.112313792109489, 0.172022193670273, 0.384404003620148, 0.115418829023838, 0.450868725776672, 0.457187473773956, 0.140348106622696, 0.15950845181942, 0.0775080993771553]</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>0.0383280552924</td>\n",
+       "        <td>[0.899999976158142, 0.566666662693024, 0.966666638851166, 1.0, 1.0, 0.800000011920929, 0.966666638851166, 0.833333313465118, 0.899999976158142, 1.0, 0.899999976158142, 1.0]</td>\n",
+       "        <td>[0.273769021034241, 0.709117114543915, 0.154145583510399, 0.093109056353569, 0.130981177091599, 0.318724304437637, 0.102762393653393, 0.268609821796417, 0.253254026174545, 0.103913448750973, 0.194639429450035, 0.0383280552923679]</td>\n",
+       "        <td>[2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13]</td>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.009905852828976726)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[44.6836378574371, 50.92365193367, 57.3649659156799, 63.7576060295105, 70.1174209117889, 76.788703918457, 83.2217078208923, 89.8764188289642, 96.2273638248444]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.975000023842</td>\n",
+       "        <td>0.0799522325397</td>\n",
+       "        <td>[0.949999988079071, 0.791666686534882, 0.800000011920929, 0.850000023841858, 0.958333313465118, 0.975000023841858, 0.899999976158142, 0.975000023841858, 0.975000023841858]</td>\n",
+       "        <td>[0.447714686393738, 0.36309215426445, 0.324623554944992, 0.301780551671982, 0.142947062849998, 0.120139442384243, 0.255296260118484, 0.0816238224506378, 0.0799522325396538]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.0760450512171</td>\n",
+       "        <td>[0.966666638851166, 0.899999976158142, 0.800000011920929, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.899999976158142, 1.0, 0.966666638851166]</td>\n",
+       "        <td>[0.370463252067566, 0.25237438082695, 0.317549884319305, 0.187985330820084, 0.104904659092426, 0.112288065254688, 0.160248279571533, 0.0687147378921509, 0.0760450512170792]</td>\n",
+       "        <td>[5, 6, 7, 8, 9, 10, 11, 12, 13]</td>\n",
+       "        <td>0</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01678679876224294)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[11.5813798904419, 21.0226759910583, 28.4713099002838, 35.9315679073334, 44.4656569957733, 50.7044010162354, 57.1448848247528, 63.4595718383789, 69.8967549800873, 76.3639938831329, 82.7779839038849, 89.6248579025269, 95.9936518669128]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.958333313465</td>\n",
+       "        <td>0.113370150328</td>\n",
+       "        <td>[0.641666650772095, 0.908333361148834, 0.891666650772095, 0.891666650772095, 0.866666674613953, 0.941666662693024, 0.941666662693024, 0.933333337306976, 0.933333337306976, 0.858333349227905, 0.966666638851166, 0.958333313465118, 0.958333313465118]</td>\n",
+       "        <td>[0.656313836574554, 0.41341444849968, 0.324400961399078, 0.304112106561661, 0.336616456508636, 0.160554125905037, 0.135852053761482, 0.159805878996849, 0.174078181385994, 0.316538035869598, 0.104411341249943, 0.105065681040287, 0.113370150327682]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.121701933444</td>\n",
+       "        <td>[0.766666650772095, 0.933333337306976, 0.866666674613953, 0.866666674613953, 0.899999976158142, 0.966666638851166, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.563848853111267, 0.330921590328217, 0.292684972286224, 0.273869335651398, 0.317834258079529, 0.144475534558296, 0.147552534937859, 0.153202146291733, 0.158350095152855, 0.22741986811161, 0.114596471190453, 0.117612592875957, 0.121701933443546]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13]</td>\n",
+       "        <td>2</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.01930169481426345)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[44.2033138275146, 50.4514999389648, 56.8880548477173, 63.2074518203735, 69.5435798168182, 76.1080069541931, 82.519660949707, 89.1418299674988, 95.518424987793]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.941666662693</td>\n",
+       "        <td>0.206491559744</td>\n",
+       "        <td>[0.566666662693024, 0.916666686534882, 0.883333325386047, 0.958333313465118, 0.841666638851166, 0.875, 0.783333361148834, 0.766666650772095, 0.941666662693024]</td>\n",
+       "        <td>[0.774700284004211, 0.651901543140411, 0.496851295232773, 0.405008375644684, 0.356276631355286, 0.340960919857025, 0.381286114454269, 0.388935476541519, 0.206491559743881]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.170937761664</td>\n",
+       "        <td>[0.733333349227905, 0.933333337306976, 0.866666674613953, 0.933333337306976, 0.866666674613953, 0.866666674613953, 0.833333313465118, 0.833333313465118, 0.966666638851166]</td>\n",
+       "        <td>[0.76544976234436, 0.657529413700104, 0.4853755235672, 0.377188384532928, 0.356318116188049, 0.332274377346039, 0.372768431901932, 0.397462010383606, 0.170937761664391]</td>\n",
+       "        <td>[5, 6, 7, 8, 9, 10, 11, 12, 13]</td>\n",
+       "        <td>0</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.011578246765795313)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[12.2524900436401, 22.163911819458, 29.4894979000092, 37.043762922287]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.791666686535</td>\n",
+       "        <td>0.595036566257</td>\n",
+       "        <td>[0.0333333350718021, 0.633333325386047, 0.816666662693024, 0.791666686534882]</td>\n",
+       "        <td>[0.947992205619812, 0.782966256141663, 0.679944217205048, 0.595036566257477]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.56917822361</td>\n",
+       "        <td>[0.100000001490116, 0.766666650772095, 0.933333337306976, 0.899999976158142]</td>\n",
+       "        <td>[0.972650408744812, 0.776390075683594, 0.681758761405945, 0.569178223609924]</td>\n",
+       "        <td>[1, 2, 3, 4]</td>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.0699102360375282)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[22.5178759098053, 29.7163498401642, 37.2609059810638]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.699999988079</td>\n",
+       "        <td>0.389858365059</td>\n",
+       "        <td>[0.641666650772095, 0.641666650772095, 0.699999988079071]</td>\n",
+       "        <td>[0.79219126701355, 0.460052192211151, 0.389858365058899]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.250768661499</td>\n",
+       "        <td>[0.766666650772095, 0.766666650772095, 0.866666674613953]</td>\n",
+       "        <td>[0.765598654747009, 0.334016799926758, 0.250768661499023]</td>\n",
+       "        <td>[2, 3, 4]</td>\n",
+       "        <td>1</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.024714880320122704)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[12.0343589782715, 21.4979238510132, 29.0434989929199, 36.4899458885193]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.649999976158</td>\n",
+       "        <td>0.673553228378</td>\n",
+       "        <td>[0.691666662693024, 0.841666638851166, 0.983333349227905, 0.649999976158142]</td>\n",
+       "        <td>[0.388701051473618, 0.424284487962723, 0.180928915739059, 0.673553228378296]</td>\n",
+       "        <td>0.800000011921</td>\n",
+       "        <td>0.384207844734</td>\n",
+       "        <td>[0.833333313465118, 0.800000011920929, 0.966666638851166, 0.800000011920929]</td>\n",
+       "        <td>[0.255310624837875, 0.357394397258759, 0.147510275244713, 0.384207844734192]</td>\n",
+       "        <td>[1, 2, 3, 4]</td>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.05573574908119242)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[45.168872833252, 51.7013738155365, 58.1221590042114, 64.4642739295959, 70.8308379650116, 77.295382976532, 83.7621510028839, 90.3811860084534, 96.7183079719543]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.816666662693</td>\n",
+       "        <td>0.457783430815</td>\n",
+       "        <td>[0.358333319425583, 0.925000011920929, 0.675000011920929, 0.733333349227905, 0.949999988079071, 0.666666686534882, 0.741666674613953, 0.908333361148834, 0.816666662693024]</td>\n",
+       "        <td>[1.03486049175262, 0.43449866771698, 0.842896223068237, 0.392013370990753, 0.195524752140045, 0.572380185127258, 0.43743160367012, 0.278554767370224, 0.457783430814743]</td>\n",
+       "        <td>0.733333349228</td>\n",
+       "        <td>0.670406579971</td>\n",
+       "        <td>[0.233333334326744, 0.966666638851166, 0.866666674613953, 0.866666674613953, 0.966666638851166, 0.666666686534882, 0.899999976158142, 0.933333337306976, 0.733333349227905]</td>\n",
+       "        <td>[1.05548679828644, 0.372740298509598, 0.427788466215134, 0.282503575086594, 0.135918349027634, 0.589654743671417, 0.253296822309494, 0.159830048680305, 0.670406579971313]</td>\n",
+       "        <td>[5, 6, 7, 8, 9, 10, 11, 12, 13]</td>\n",
+       "        <td>0</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.09641245863612281)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[12.7177708148956]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.658333361149</td>\n",
+       "        <td>1.25986480713</td>\n",
+       "        <td>[0.658333361148834]</td>\n",
+       "        <td>[1.25986480712891]</td>\n",
+       "        <td>0.633333325386</td>\n",
+       "        <td>1.26717245579</td>\n",
+       "        <td>[0.633333325386047]</td>\n",
+       "        <td>[1.26717245578766]</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>2</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.003730347382813742)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[11.1050899028778]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.600000023842</td>\n",
+       "        <td>1.23435640335</td>\n",
+       "        <td>[0.600000023841858]</td>\n",
+       "        <td>[1.23435640335083]</td>\n",
+       "        <td>0.5</td>\n",
+       "        <td>1.37250542641</td>\n",
+       "        <td>[0.5]</td>\n",
+       "        <td>[1.37250542640686]</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>2</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.0018352035707327032)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[11.3283720016479]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.466666668653</td>\n",
+       "        <td>1.01645076275</td>\n",
+       "        <td>[0.466666668653488]</td>\n",
+       "        <td>[1.01645076274872]</td>\n",
+       "        <td>0.433333337307</td>\n",
+       "        <td>1.01912522316</td>\n",
+       "        <td>[0.433333337306976]</td>\n",
+       "        <td>[1.01912522315979]</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>2</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.03837714620063437)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[12.5016968250275]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.308333337307</td>\n",
+       "        <td>1.0995465517</td>\n",
+       "        <td>[0.308333337306976]</td>\n",
+       "        <td>[1.09954655170441]</td>\n",
+       "        <td>0.433333337307</td>\n",
+       "        <td>1.0980553627</td>\n",
+       "        <td>[0.433333337306976]</td>\n",
+       "        <td>[1.09805536270142]</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>2</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.0017052377620857802)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[10.8097839355469]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.341666668653</td>\n",
+       "        <td>1.28075575829</td>\n",
+       "        <td>[0.341666668653488]</td>\n",
+       "        <td>[1.28075575828552]</td>\n",
+       "        <td>0.366666674614</td>\n",
+       "        <td>1.43494951725</td>\n",
+       "        <td>[0.366666674613953]</td>\n",
+       "        <td>[1.43494951725006]</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>2</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.0015217424326594508)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[21.2741727828979, 28.7427089214325, 36.1846778392792]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.333333343267</td>\n",
+       "        <td>1.07403242588</td>\n",
+       "        <td>[0.474999994039536, 0.358333319425583, 0.333333343267441]</td>\n",
+       "        <td>[1.08657968044281, 1.07721281051636, 1.07403242588043]</td>\n",
+       "        <td>0.333333343267</td>\n",
+       "        <td>1.09314000607</td>\n",
+       "        <td>[0.433333337306976, 0.300000011920929, 0.333333343267441]</td>\n",
+       "        <td>[1.11294913291931, 1.10347521305084, 1.09314000606537]</td>\n",
+       "        <td>[2, 3, 4]</td>\n",
+       "        <td>1</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.051964270528848694)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[11.8142108917236]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.333333343267</td>\n",
+       "        <td>1.09948420525</td>\n",
+       "        <td>[0.333333343267441]</td>\n",
+       "        <td>[1.09948420524597]</td>\n",
+       "        <td>0.333333343267</td>\n",
+       "        <td>1.09620642662</td>\n",
+       "        <td>[0.333333343267441]</td>\n",
+       "        <td>[1.09620642662048]</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>2</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(10, 1, u\"optimizer='Adam(lr=0.04232194170481019)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [21.911346912384, 29.2674539089203, 36.8268938064575, 44.9022789001465, 51.1760609149933, 57.6593999862671, 64.184476852417, 70.5566418170929, 77.0253269672394, 83.4826798439026, 90.1138219833374, 96.4566838741302], [u'accuracy'], u'categorical_crossentropy', 0.975000023841858, 0.0775080993771553, [0.791666686534882, 0.608333349227905, 0.966666638851166, 0.975000023841858, 0.966666638851166, 0.800000011920929, 0.975000023841858, 0.683333337306976, 0.733333349227905, 0.949999988079071, 0.949999988079071, 0.975000023841858], [0.374035209417343, 0.732228577136993, 0.170820266008377, 0.112313792109489, 0.172022193670273, 0.384404003620148, 0.115418829023838, 0.450868725776672, 0.457187473773956, 0.140348106622696, 0.15950845181942, 0.0775080993771553], 1.0, 0.0383280552923679, [0.899999976158142, 0.566666662693024, 0.966666638851166, 1.0, 1.0, 0.800000011920929, 0.966666638851166, 0.833333313465118, 0.899999976158142, 1.0, 0.899999976158142, 1.0], [0.273769021034241, 0.709117114543915, 0.154145583510399, 0.093109056353569, 0.130981177091599, 0.318724304437637, 0.102762393653393, 0.268609821796417, 0.253254026174545, 0.103913448750973, 0.194639429450035, 0.0383280552923679], [2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13], 1, 1),\n",
+       " (13, 1, u\"optimizer='Adam(lr=0.009905852828976726)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [44.6836378574371, 50.92365193367, 57.3649659156799, 63.7576060295105, 70.1174209117889, 76.788703918457, 83.2217078208923, 89.8764188289642, 96.2273638248444], [u'accuracy'], u'categorical_crossentropy', 0.975000023841858, 0.0799522325396538, [0.949999988079071, 0.791666686534882, 0.800000011920929, 0.850000023841858, 0.958333313465118, 0.975000023841858, 0.899999976158142, 0.975000023841858, 0.975000023841858], [0.447714686393738, 0.36309215426445, 0.324623554944992, 0.301780551671982, 0.142947062849998, 0.120139442384243, 0.255296260118484, 0.0816238224506378, 0.0799522325396538], 0.966666638851166, 0.0760450512170792, [0.966666638851166, 0.899999976158142, 0.800000011920929, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.899999976158142, 1.0, 0.966666638851166], [0.370463252067566, 0.25237438082695, 0.317549884319305, 0.187985330820084, 0.104904659092426, 0.112288065254688, 0.160248279571533, 0.0687147378921509, 0.0760450512170792], [5, 6, 7, 8, 9, 10, 11, 12, 13], 0, 0),\n",
+       " (5, 2, u\"optimizer='Adam(lr=0.01678679876224294)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [11.5813798904419, 21.0226759910583, 28.4713099002838, 35.9315679073334, 44.4656569957733, 50.7044010162354, 57.1448848247528, 63.4595718383789, 69.8967549800873, 76.3639938831329, 82.7779839038849, 89.6248579025269, 95.9936518669128], [u'accuracy'], u'categorical_crossentropy', 0.958333313465118, 0.113370150327682, [0.641666650772095, 0.908333361148834, 0.891666650772095, 0.891666650772095, 0.866666674613953, 0.941666662693024, 0.941666662693024, 0.933333337306976, 0.933333337306976, 0.858333349227905, 0.966666638851166, 0.958333313465118, 0.958333313465118], [0.656313836574554, 0.41341444849968, 0.324400961399078, 0.304112106561661, 0.336616456508636, 0.160554125905037, 0.135852053761482, 0.159805878996849, 0.174078181385994, 0.316538035869598, 0.104411341249943, 0.105065681040287, 0.113370150327682], 0.966666638851166, 0.121701933443546, [0.766666650772095, 0.933333337306976, 0.866666674613953, 0.866666674613953, 0.899999976158142, 0.966666638851166, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.966666638851166], [0.563848853111267, 0.330921590328217, 0.292684972286224, 0.273869335651398, 0.317834258079529, 0.144475534558296, 0.147552534937859, 0.153202146291733, 0.158350095152855, 0.22741986811161, 0.114596471190453, 0.117612592875957, 0.121701933443546], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13], 2, 2),\n",
+       " (14, 2, u\"optimizer='RMSprop(lr=0.01930169481426345)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [44.2033138275146, 50.4514999389648, 56.8880548477173, 63.2074518203735, 69.5435798168182, 76.1080069541931, 82.519660949707, 89.1418299674988, 95.518424987793], [u'accuracy'], u'categorical_crossentropy', 0.941666662693024, 0.206491559743881, [0.566666662693024, 0.916666686534882, 0.883333325386047, 0.958333313465118, 0.841666638851166, 0.875, 0.783333361148834, 0.766666650772095, 0.941666662693024], [0.774700284004211, 0.651901543140411, 0.496851295232773, 0.405008375644684, 0.356276631355286, 0.340960919857025, 0.381286114454269, 0.388935476541519, 0.206491559743881], 0.966666638851166, 0.170937761664391, [0.733333349227905, 0.933333337306976, 0.866666674613953, 0.933333337306976, 0.866666674613953, 0.866666674613953, 0.833333313465118, 0.833333313465118, 0.966666638851166], [0.76544976234436, 0.657529413700104, 0.4853755235672, 0.377188384532928, 0.356318116188049, 0.332274377346039, 0.372768431901932, 0.397462010383606, 0.170937761664391], [5, 6, 7, 8, 9, 10, 11, 12, 13], 0, 0),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.011578246765795313)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [12.2524900436401, 22.163911819458, 29.4894979000092, 37.043762922287], [u'accuracy'], u'categorical_crossentropy', 0.791666686534882, 0.595036566257477, [0.0333333350718021, 0.633333325386047, 0.816666662693024, 0.791666686534882], [0.947992205619812, 0.782966256141663, 0.679944217205048, 0.595036566257477], 0.899999976158142, 0.569178223609924, [0.100000001490116, 0.766666650772095, 0.933333337306976, 0.899999976158142], [0.972650408744812, 0.776390075683594, 0.681758761405945, 0.569178223609924], [1, 2, 3, 4], 2, 1),\n",
+       " (12, 1, u\"optimizer='Adam(lr=0.0699102360375282)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [22.5178759098053, 29.7163498401642, 37.2609059810638], [u'accuracy'], u'categorical_crossentropy', 0.699999988079071, 0.389858365058899, [0.641666650772095, 0.641666650772095, 0.699999988079071], [0.79219126701355, 0.460052192211151, 0.389858365058899], 0.866666674613953, 0.250768661499023, [0.766666650772095, 0.766666650772095, 0.866666674613953], [0.765598654747009, 0.334016799926758, 0.250768661499023], [2, 3, 4], 1, 0),\n",
+       " (1, 1, u\"optimizer='RMSprop(lr=0.024714880320122704)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [12.0343589782715, 21.4979238510132, 29.0434989929199, 36.4899458885193], [u'accuracy'], u'categorical_crossentropy', 0.649999976158142, 0.673553228378296, [0.691666662693024, 0.841666638851166, 0.983333349227905, 0.649999976158142], [0.388701051473618, 0.424284487962723, 0.180928915739059, 0.673553228378296], 0.800000011920929, 0.384207844734192, [0.833333313465118, 0.800000011920929, 0.966666638851166, 0.800000011920929], [0.255310624837875, 0.357394397258759, 0.147510275244713, 0.384207844734192], [1, 2, 3, 4], 2, 1),\n",
+       " (15, 2, u\"optimizer='Adam(lr=0.05573574908119242)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [45.168872833252, 51.7013738155365, 58.1221590042114, 64.4642739295959, 70.8308379650116, 77.295382976532, 83.7621510028839, 90.3811860084534, 96.7183079719543], [u'accuracy'], u'categorical_crossentropy', 0.816666662693024, 0.457783430814743, [0.358333319425583, 0.925000011920929, 0.675000011920929, 0.733333349227905, 0.949999988079071, 0.666666686534882, 0.741666674613953, 0.908333361148834, 0.816666662693024], [1.03486049175262, 0.43449866771698, 0.842896223068237, 0.392013370990753, 0.195524752140045, 0.572380185127258, 0.43743160367012, 0.278554767370224, 0.457783430814743], 0.733333349227905, 0.670406579971313, [0.233333334326744, 0.966666638851166, 0.866666674613953, 0.866666674613953, 0.966666638851166, 0.666666686534882, 0.899999976158142, 0.933333337306976, 0.733333349227905], [1.05548679828644, 0.372740298509598, 0.427788466215134, 0.282503575086594, 0.135918349027634, 0.589654743671417, 0.253296822309494, 0.159830048680305, 0.670406579971313], [5, 6, 7, 8, 9, 10, 11, 12, 13], 0, 0),\n",
+       " (8, 1, u\"optimizer='RMSprop(lr=0.09641245863612281)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [12.7177708148956], [u'accuracy'], u'categorical_crossentropy', 0.658333361148834, 1.25986480712891, [0.658333361148834], [1.25986480712891], 0.633333325386047, 1.26717245578766, [0.633333325386047], [1.26717245578766], [1], 2, 0),\n",
+       " (2, 1, u\"optimizer='RMSprop(lr=0.003730347382813742)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [11.1050899028778], [u'accuracy'], u'categorical_crossentropy', 0.600000023841858, 1.23435640335083, [0.600000023841858], [1.23435640335083], 0.5, 1.37250542640686, [0.5], [1.37250542640686], [1], 2, 0),\n",
+       " (7, 1, u\"optimizer='Adam(lr=0.0018352035707327032)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [11.3283720016479], [u'accuracy'], u'categorical_crossentropy', 0.466666668653488, 1.01645076274872, [0.466666668653488], [1.01645076274872], 0.433333337306976, 1.01912522315979, [0.433333337306976], [1.01912522315979], [1], 2, 0),\n",
+       " (4, 2, u\"optimizer='Adam(lr=0.03837714620063437)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [12.5016968250275], [u'accuracy'], u'categorical_crossentropy', 0.308333337306976, 1.09954655170441, [0.308333337306976], [1.09954655170441], 0.433333337306976, 1.09805536270142, [0.433333337306976], [1.09805536270142], [1], 2, 0),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.0017052377620857802)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [10.8097839355469], [u'accuracy'], u'categorical_crossentropy', 0.341666668653488, 1.28075575828552, [0.341666668653488], [1.28075575828552], 0.366666674613953, 1.43494951725006, [0.366666674613953], [1.43494951725006], [1], 2, 0),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.0015217424326594508)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [21.2741727828979, 28.7427089214325, 36.1846778392792], [u'accuracy'], u'categorical_crossentropy', 0.333333343267441, 1.07403242588043, [0.474999994039536, 0.358333319425583, 0.333333343267441], [1.08657968044281, 1.07721281051636, 1.07403242588043], 0.333333343267441, 1.09314000606537, [0.433333337306976, 0.300000011920929, 0.333333343267441], [1.11294913291931, 1.10347521305084, 1.09314000606537], [2, 3, 4], 1, 0),\n",
+       " (3, 1, u\"optimizer='RMSprop(lr=0.051964270528848694)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [11.8142108917236], [u'accuracy'], u'categorical_crossentropy', 0.333333343267441, 1.09948420524597, [0.333333343267441], [1.09948420524597], 0.333333343267441, 1.09620642662048, [0.333333343267441], [1.09620642662048], [1], 2, 0)]"
+      ]
+     },
+     "execution_count": 19,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM automl_output_info ORDER BY validation_metrics_final DESC, validation_loss_final;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Plot results"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 20,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%matplotlib notebook\n",
+    "import matplotlib.pyplot as plt\n",
+    "from matplotlib.ticker import MaxNLocator\n",
+    "from collections import defaultdict\n",
+    "import pandas as pd\n",
+    "import seaborn as sns\n",
+    "sns.set_palette(sns.color_palette(\"hls\", 20))\n",
+    "plt.rcParams.update({'font.size': 12})\n",
+    "pd.set_option('display.max_colwidth', -1)"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 21,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "15 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM automl_output_info;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "df_summary = %sql SELECT * FROM automl_output_summary;\n",
+    "df_summary = df_summary.DataFrame()\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Iteration')\n",
+    "ax_metric.set_ylabel('Metric')\n",
+    "ax_metric.set_title('Validation metric curve')\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Iteration')\n",
+    "ax_loss.set_ylabel('Loss')\n",
+    "ax_loss.set_title('Validation loss curve')\n",
+    "\n",
+    "for mst_key in df_results['mst_key']:\n",
+    "    df_output_info = %sql SELECT validation_metrics,validation_loss,metrics_iters FROM automl_output_info WHERE mst_key = $mst_key;\n",
+    "    df_output_info = df_output_info.DataFrame()\n",
+    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
+    "    validation_loss = df_output_info['validation_loss'][0]\n",
+    "    iters = df_output_info['metrics_iters'][0]\n",
+    "    \n",
+    "    #ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
+    "    ax_metric.plot(iters, validation_metrics, marker='o')\n",
+    "    #ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
+    "    ax_loss.plot(iters, validation_loss, marker='o')\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"hyperopt\"></a>\n",
+    "# 5.  Hyperopt"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 22,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_automl</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 22,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS automl_output, automl_output_info, automl_output_summary, automl_mst_table, automl_mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_automl('iris_train_packed',                -- source table\n",
+    "                                  'automl_output',                    -- model output table\n",
+    "                                  'model_arch_library',               -- model architecture table\n",
+    "                                  'automl_mst_table',                 -- model selection output table\n",
+    "                                  ARRAY[1,2],                         -- model IDs\n",
+    "                                  $${\n",
+    "                                      'loss': ['categorical_crossentropy'], \n",
+    "                                      'optimizer_params_list': [ \n",
+    "                                          {'optimizer': ['Adam'],'lr': [0.001, 0.1, 'log']},\n",
+    "                                          {'optimizer': ['RMSprop'],'lr': [0.001, 0.1, 'log']}\n",
+    "                                      ],\n",
+    "                                      'metrics': ['accuracy']\n",
+    "                                  } $$,                               -- compile param grid\n",
+    "                                  $${'batch_size': [4, 8], 'epochs': [1]}$$,  -- fit params grid\n",
+    "                                  'hyperopt',                         -- autoML method\n",
+    "                                  'num_configs=20, num_iterations=10, algorithm=tpe',  -- autoML params\n",
+    "                                  NULL,                               -- random state\n",
+    "                                  NULL,                               -- object table\n",
+    "                                  FALSE,                              -- use GPUs\n",
+    "                                  'iris_test_packed',                 -- validation table\n",
+    "                                  1,                                  -- metrics compute freq\n",
+    "                                  NULL,                               -- name\n",
+    "                                  NULL);                              -- descr"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 23,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>automl_method</th>\n",
+       "        <th>automl_params</th>\n",
+       "        <th>random_state</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>use_gpus</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>automl_output</td>\n",
+       "        <td>automl_output_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>automl_mst_table</td>\n",
+       "        <td>hyperopt</td>\n",
+       "        <td>num_configs=20, num_iterations=10, algorithm=tpe</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>False</td>\n",
+       "        <td>1</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2021-03-05 23:59:31</td>\n",
+       "        <td>2021-03-06 00:03:57</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_test_packed', u'automl_output', u'automl_output_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'automl_mst_table', u'hyperopt', u'num_configs=20, num_iterations=10, algorithm=tpe', None, None, False, 1, None, None, datetime.datetime(2021, 3, 5, 23, 59, 31), datetime.datetime(2021, 3, 6, 0, 3, 57), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0)]"
+      ]
+     },
+     "execution_count": 23,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM automl_output_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the results for each model"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 24,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "20 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.0084793872639979)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[56.9403030872345, 59.805566072464, 62.2339789867401, 64.8922078609467, 67.5616340637207, 70.2253429889679, 72.8736228942871, 75.5874469280243, 78.2902030944824, 80.9871909618378]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.975000023842</td>\n",
+       "        <td>0.0910520926118</td>\n",
+       "        <td>[0.649999976158142, 0.891666650772095, 0.883333325386047, 0.949999988079071, 0.975000023841858, 0.883333325386047, 0.850000023841858, 0.949999988079071, 0.975000023841858, 0.975000023841858]</td>\n",
+       "        <td>[0.559232711791992, 0.335382640361786, 0.259929001331329, 0.158979862928391, 0.114544428884983, 0.269487291574478, 0.293675005435944, 0.0902178362011909, 0.0766977593302727, 0.0910520926117897]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.0768957436085</td>\n",
+       "        <td>[0.833333313465118, 0.933333337306976, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.899999976158142, 0.899999976158142, 0.933333337306976, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.485892802476883, 0.249617904424667, 0.258282363414764, 0.11016520857811, 0.0912857726216316, 0.280073672533035, 0.178015038371086, 0.087411992251873, 0.062506839632988, 0.0768957436084747]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.03366551083145706)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[161.690346240997, 164.041937112808, 166.454420089722, 168.906048059464, 171.067217111588, 173.555004119873, 175.944698095322, 178.445127248764, 180.502294063568, 183.037788152695]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.144752591848</td>\n",
+       "        <td>[0.316666662693024, 0.666666686534882, 0.716666638851166, 0.641666650772095, 0.675000011920929, 0.975000023841858, 0.791666686534882, 0.975000023841858, 0.966666638851166, 0.949999988079071]</td>\n",
+       "        <td>[1.57745933532715, 0.405172228813171, 0.471270889043808, 1.00022745132446, 0.840015530586243, 0.128021001815796, 0.473532497882843, 0.091586634516716, 0.112696528434753, 0.144752591848373]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.100186347961</td>\n",
+       "        <td>[0.433333337306976, 0.666666686534882, 0.899999976158142, 0.766666650772095, 0.866666674613953, 0.966666638851166, 0.899999976158142, 0.933333337306976, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[1.36917245388031, 0.372486144304276, 0.266377687454224, 0.571163833141327, 0.457086622714996, 0.103041857481003, 0.276452839374542, 0.0908508822321892, 0.0997116342186928, 0.100186347961426]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.009794369846837002)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[186.611872196198, 188.95641207695, 191.037184238434, 193.397297143936, 195.740861177444, 197.805513143539, 200.180992126465, 202.689172029495, 205.040098190308, 207.208242177963]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.116746708751</td>\n",
+       "        <td>[0.75, 0.975000023841858, 0.850000023841858, 0.941666662693024, 0.933333337306976, 0.949999988079071, 0.949999988079071, 0.983333349227905, 0.958333313465118, 0.949999988079071]</td>\n",
+       "        <td>[0.5159512758255, 0.353324204683304, 0.333910763263702, 0.245715036988258, 0.188893154263496, 0.161517903208733, 0.137443989515305, 0.122971840202808, 0.14612153172493, 0.116746708750725]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.106192082167</td>\n",
+       "        <td>[0.899999976158142, 0.966666638851166, 0.899999976158142, 0.966666638851166, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.423073083162308, 0.298538327217102, 0.234973803162575, 0.176778241991997, 0.170526877045631, 0.145023569464684, 0.119270212948322, 0.103897586464882, 0.104170136153698, 0.106192082166672]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.007581048101981366)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[57.225380897522, 60.0725290775299, 62.731920003891, 65.1544499397278, 67.8143260478973, 70.4762139320374, 73.1227269172668, 75.8475530147552, 78.555095911026, 81.2564718723297]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.958333313465</td>\n",
+       "        <td>0.133664950728</td>\n",
+       "        <td>[0.649999976158142, 0.883333325386047, 0.941666662693024, 0.899999976158142, 0.958333313465118, 0.925000011920929, 0.941666662693024, 0.933333337306976, 0.983333349227905, 0.958333313465118]</td>\n",
+       "        <td>[1.03808128833771, 0.883756637573242, 0.686505734920502, 0.517532765865326, 0.401096671819687, 0.259793311357498, 0.177235946059227, 0.168946355581284, 0.128713861107826, 0.133664950728416]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.122641228139</td>\n",
+       "        <td>[0.633333325386047, 0.899999976158142, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.933333337306976, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[1.02865540981293, 0.850838720798492, 0.612184524536133, 0.452387690544128, 0.33221709728241, 0.23906472325325, 0.165990635752678, 0.164969280362129, 0.12097629904747, 0.1226412281394]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.012596538573477555)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[137.034833192825, 139.500956058502, 141.807043075562, 143.897747039795, 146.264532089233, 148.888093233109, 150.934833049774, 153.475459098816, 155.848874092102, 158.239592075348]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.916666686535</td>\n",
+       "        <td>0.184266731143</td>\n",
+       "        <td>[0.566666662693024, 0.933333337306976, 0.649999976158142, 0.666666686534882, 0.808333337306976, 0.891666650772095, 0.891666650772095, 0.975000023841858, 0.933333337306976, 0.916666686534882]</td>\n",
+       "        <td>[0.829304933547974, 0.631127297878265, 0.597909092903137, 0.552545011043549, 0.428654760122299, 0.233174994587898, 0.236562281847, 0.119615346193314, 0.191903278231621, 0.184266731142998]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.124655880034</td>\n",
+       "        <td>[0.699999988079071, 0.899999976158142, 0.800000011920929, 0.833333313465118, 0.899999976158142, 0.933333337306976, 0.866666674613953, 1.0, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.74066150188446, 0.532437741756439, 0.480882078409195, 0.435436576604843, 0.310187846422195, 0.234515085816383, 0.247250944375992, 0.107902131974697, 0.136675015091896, 0.12465588003397]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.005441362966347114)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[109.630754947662, 112.341638088226, 114.763943910599, 117.389002084732, 119.991095066071, 122.593973875046, 125.238317966461, 127.8488509655, 130.375853061676, 133.020073890686]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.916666686535</td>\n",
+       "        <td>0.216380029917</td>\n",
+       "        <td>[0.641666650772095, 0.733333349227905, 0.649999976158142, 0.858333349227905, 0.958333313465118, 0.699999988079071, 0.916666686534882, 0.858333349227905, 0.966666638851166, 0.916666686534882]</td>\n",
+       "        <td>[0.73615950345993, 0.489604264497757, 0.45263683795929, 0.37542662024498, 0.334106951951981, 0.419453173875809, 0.284875482320786, 0.27151495218277, 0.185230866074562, 0.216380029916763]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.127150848508</td>\n",
+       "        <td>[0.766666650772095, 0.699999988079071, 0.800000011920929, 0.933333337306976, 0.966666638851166, 0.666666686534882, 0.899999976158142, 0.933333337306976, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.631976008415222, 0.448555260896683, 0.323729306459427, 0.28750941157341, 0.281407296657562, 0.421543717384338, 0.259464651346207, 0.158164814114571, 0.135125860571861, 0.127150848507881]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.04966544234738768)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[2.26167011260986, 4.64152717590332, 7.38891315460205, 10.1575191020966, 12.6948411464691, 15.5193021297455, 17.8266370296478, 20.364767074585, 23.1241211891174, 25.6702241897583]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.183234125376</td>\n",
+       "        <td>[0.641666650772095, 0.666666686534882, 0.966666638851166, 0.958333313465118, 0.716666638851166, 0.983333349227905, 0.966666638851166, 0.983333349227905, 0.975000023841858, 0.949999988079071]</td>\n",
+       "        <td>[0.829066038131714, 0.490932732820511, 0.341925740242004, 0.215810611844063, 0.400910943746567, 0.107548490166664, 0.0985226780176163, 0.0732712596654892, 0.070111908018589, 0.183234125375748]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.137178555131</td>\n",
+       "        <td>[0.866666674613953, 0.666666686534882, 0.966666638851166, 0.966666638851166, 0.699999988079071, 0.966666638851166, 0.966666638851166, 0.933333337306976, 1.0, 0.966666638851166]</td>\n",
+       "        <td>[0.843752980232239, 0.435137718915939, 0.26888644695282, 0.157842606306076, 0.406648069620132, 0.0976309478282928, 0.0788726136088371, 0.0751720294356346, 0.0686705932021141, 0.137178555130959]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.0023223781742022285)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[29.5305609703064, 31.8714768886566, 34.5841488838196, 37.194139957428, 40.0518889427185, 42.6473689079285, 44.9830038547516, 47.734827041626, 50.3327059745789, 52.9546790122986]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.247659549117</td>\n",
+       "        <td>[0.641666650772095, 0.691666662693024, 0.641666650772095, 0.966666638851166, 0.941666662693024, 0.958333313465118, 0.933333337306976, 0.925000011920929, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.949797213077545, 0.797480285167694, 0.672827661037445, 0.523928940296173, 0.444453626871109, 0.386174380779266, 0.347499698400497, 0.321201831102371, 0.278330504894257, 0.247659549117088]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.198830261827</td>\n",
+       "        <td>[0.766666650772095, 0.833333313465118, 0.766666650772095, 0.966666638851166, 0.966666638851166, 0.933333337306976, 0.966666638851166, 0.899999976158142, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.931245148181915, 0.763193428516388, 0.600658059120178, 0.456866592168808, 0.373299777507782, 0.326453566551208, 0.275230079889297, 0.284671515226364, 0.221188083291054, 0.198830261826515]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.0014467801648012073)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[84.6684620380402, 86.9988820552826, 89.4947271347046, 91.8522582054138, 93.9668672084808, 96.3584721088409, 98.7448561191559, 101.170181035995, 103.282526016235, 105.678196191788]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.395356565714</td>\n",
+       "        <td>[0.625, 0.641666650772095, 0.741666674613953, 0.883333325386047, 0.833333313465118, 0.941666662693024, 0.949999988079071, 0.983333349227905, 0.875, 0.966666638851166]</td>\n",
+       "        <td>[0.941141128540039, 0.820547699928284, 0.723011374473572, 0.646571576595306, 0.57060444355011, 0.514499425888062, 0.46852970123291, 0.439025938510895, 0.416335105895996, 0.395356565713882]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.326709568501</td>\n",
+       "        <td>[0.766666650772095, 0.766666650772095, 0.899999976158142, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.933333337306976, 0.966666638851166]</td>\n",
+       "        <td>[0.941896796226501, 0.816571235656738, 0.707062959671021, 0.61734527349472, 0.521940350532532, 0.45942959189415, 0.405136495828629, 0.374987095594406, 0.33730074763298, 0.326709568500519]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.0011757973913283008)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[237.369596242905, 240.072783231735, 242.490661382675, 245.210073232651, 247.916568279266, 250.679228305817, 253.465747356415, 256.328309297562, 259.131007194519, 262.085569381714]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.916666686535</td>\n",
+       "        <td>0.567601382732</td>\n",
+       "        <td>[0.333333343267441, 0.441666662693024, 0.433333337306976, 0.349999994039536, 0.349999994039536, 0.675000011920929, 0.683333337306976, 0.800000011920929, 0.774999976158142, 0.916666686534882]</td>\n",
+       "        <td>[1.14459049701691, 1.08465170860291, 1.02045607566833, 0.952999651432037, 0.883513271808624, 0.809052526950836, 0.748401284217834, 0.682490110397339, 0.620046377182007, 0.567601382732391]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.512901842594</td>\n",
+       "        <td>[0.333333343267441, 0.400000005960464, 0.400000005960464, 0.366666674613953, 0.366666674613953, 0.800000011920929, 0.800000011920929, 0.866666674613953, 0.866666674613953, 0.966666638851166]</td>\n",
+       "        <td>[1.21284127235413, 1.13662087917328, 1.04775261878967, 0.957895994186401, 0.871163666248322, 0.780500650405884, 0.705705106258392, 0.636253237724304, 0.558390736579895, 0.512901842594147]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.003695186053629043)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[211.166339159012, 213.628123044968, 216.070516109467, 218.28012919426, 220.77138209343, 223.166202068329, 225.899930000305, 228.167545080185, 230.690491199493, 233.073115110397]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.114436306059</td>\n",
+       "        <td>[0.641666650772095, 0.875, 0.824999988079071, 0.824999988079071, 0.958333313465118, 0.866666674613953, 0.966666638851166, 0.941666662693024, 0.958333313465118, 0.966666638851166]</td>\n",
+       "        <td>[0.718437075614929, 0.535359025001526, 0.403026401996613, 0.348048120737076, 0.244051590561867, 0.264052510261536, 0.1720110476017, 0.164029255509377, 0.154526039958, 0.114436306059361]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.107093170285</td>\n",
+       "        <td>[0.766666650772095, 0.933333337306976, 0.899999976158142, 0.833333313465118, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.933333337306976, 0.966666638851166, 0.933333337306976]</td>\n",
+       "        <td>[0.647899329662323, 0.474290877580643, 0.308415770530701, 0.318869024515152, 0.206334576010704, 0.250639617443085, 0.129751890897751, 0.156522572040558, 0.112601205706596, 0.107093170285225]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.03271173767396424)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[136.810528039932, 139.271977186203, 141.348733186722, 143.678931236267, 146.042705059052, 148.659409046173, 150.713799238205, 153.240673065186, 155.618861198425, 158.003229141235]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.200596183538</td>\n",
+       "        <td>[0.341666668653488, 0.925000011920929, 0.958333313465118, 0.675000011920929, 0.966666638851166, 0.766666650772095, 0.975000023841858, 0.75, 0.975000023841858, 0.899999976158142]</td>\n",
+       "        <td>[1.04381895065308, 0.384325951337814, 0.263480663299561, 0.593676149845123, 0.141404688358307, 0.362050473690033, 0.0923048332333565, 0.351189643144608, 0.0946881100535393, 0.200596183538437]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.257636517286</td>\n",
+       "        <td>[0.533333361148834, 0.866666674613953, 0.966666638851166, 0.800000011920929, 0.966666638851166, 0.833333313465118, 0.966666638851166, 0.866666674613953, 0.966666638851166, 0.899999976158142]</td>\n",
+       "        <td>[0.91362202167511, 0.340268641710281, 0.21289549767971, 0.362329840660095, 0.136535987257957, 0.327440768480301, 0.111000411212444, 0.227803841233253, 0.111130490899086, 0.257636517286301]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.0027814197503322115)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[161.464279174805, 163.823823213577, 166.230634212494, 168.657960176468, 170.846117019653, 173.335704088211, 175.715650081635, 178.106207132339, 180.278338193893, 182.815184116364]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.351116120815</td>\n",
+       "        <td>[0.600000023841858, 0.558333337306976, 0.541666686534882, 0.600000023841858, 0.899999976158142, 0.916666686534882, 0.850000023841858, 0.925000011920929, 0.933333337306976, 0.933333337306976]</td>\n",
+       "        <td>[0.9764444231987, 0.860457479953766, 0.76110851764679, 0.688599288463593, 0.623845517635345, 0.558117687702179, 0.516568541526794, 0.438872784376144, 0.389935582876205, 0.351116120815277]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.326276868582</td>\n",
+       "        <td>[0.433333337306976, 0.433333337306976, 0.5, 0.433333337306976, 0.899999976158142, 0.866666674613953, 0.933333337306976, 0.866666674613953, 0.899999976158142, 0.899999976158142]</td>\n",
+       "        <td>[1.03803777694702, 0.913994610309601, 0.794906616210938, 0.723303020000458, 0.652373254299164, 0.566653609275818, 0.481746405363083, 0.454111516475677, 0.379933565855026, 0.326276868581772]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.035340389425615855)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[2.53016018867493, 5.03768014907837, 7.64597201347351, 10.4127900600433, 13.0584251880646, 15.7928349971771, 18.0860531330109, 20.625785112381, 23.3826050758362, 25.9297461509705]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.875</td>\n",
+       "        <td>0.28657540679</td>\n",
+       "        <td>[0.850000023841858, 0.491666674613953, 0.941666662693024, 0.783333361148834, 0.925000011920929, 0.850000023841858, 0.966666638851166, 0.958333313465118, 0.908333361148834, 0.875]</td>\n",
+       "        <td>[0.313798636198044, 0.746647894382477, 0.232485517859459, 0.384049296379089, 0.201492115855217, 0.276773244142532, 0.144450753927231, 0.116710871458054, 0.210491970181465, 0.28657540678978]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.334158778191</td>\n",
+       "        <td>[0.833333313465118, 0.533333361148834, 0.833333313465118, 0.899999976158142, 0.899999976158142, 0.833333313465118, 0.966666638851166, 1.0, 0.899999976158142, 0.899999976158142]</td>\n",
+       "        <td>[0.281513780355453, 0.781840145587921, 0.252955704927444, 0.219736546278, 0.268592208623886, 0.332309901714325, 0.131899908185005, 0.0595534667372704, 0.255705177783966, 0.334158778190613]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.08208550087461897)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[84.8872451782227, 87.2208690643311, 89.7248260974884, 92.1014380455017, 94.1999170780182, 96.5836410522461, 98.9723200798035, 101.409075021744, 103.511981010437, 105.902093172073]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.591654956341</td>\n",
+       "        <td>[0.333333343267441, 0.641666650772095, 0.566666662693024, 0.591666638851166, 0.758333325386047, 0.666666686534882, 0.966666638851166, 0.916666686534882, 0.949999988079071, 0.766666650772095]</td>\n",
+       "        <td>[1.02616000175476, 0.486202239990234, 0.636112213134766, 0.692184090614319, 0.505898773670197, 0.467963546514511, 0.268672525882721, 0.176122322678566, 0.122547559440136, 0.59165495634079]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.33915963769</td>\n",
+       "        <td>[0.333333343267441, 0.766666650772095, 0.733333349227905, 0.766666650772095, 0.733333349227905, 0.666666686534882, 0.966666638851166, 0.899999976158142, 0.899999976158142, 0.899999976158142]</td>\n",
+       "        <td>[1.12080752849579, 0.381140530109406, 0.5174320936203, 0.473030716180801, 0.607473373413086, 0.409501492977142, 0.212725415825844, 0.296080023050308, 0.18353745341301, 0.33915963768959]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.0025753473010720596)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[186.835414171219, 189.195631027222, 191.254861116409, 193.619318246841, 195.966614246368, 198.311106204987, 200.407158136368, 202.908673048019, 205.313441038132, 207.438939094543]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.891666650772</td>\n",
+       "        <td>0.950204193592</td>\n",
+       "        <td>[0.358333319425583, 0.358333319425583, 0.358333319425583, 0.449999988079071, 0.400000005960464, 0.516666650772095, 0.583333313465118, 0.608333349227905, 0.899999976158142, 0.891666650772095]</td>\n",
+       "        <td>[1.09640550613403, 1.08471190929413, 1.0751405954361, 1.06720495223999, 1.06165635585785, 1.0469172000885, 1.0301650762558, 1.00463593006134, 0.979537725448608, 0.950204193592072]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.931918859482</td>\n",
+       "        <td>[0.233333334326744, 0.233333334326744, 0.233333334326744, 0.366666674613953, 0.266666680574417, 0.400000005960464, 0.400000005960464, 0.433333337306976, 0.833333313465118, 0.866666674613953]</td>\n",
+       "        <td>[1.10118973255157, 1.08938491344452, 1.07863283157349, 1.06669425964355, 1.06701147556305, 1.04324889183044, 1.02750730514526, 0.996739327907562, 0.966157376766205, 0.931918859481812]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.056702169442788934)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[237.640507221222, 240.350925207138, 243.027331352234, 245.468372344971, 248.176068305969, 250.940598249435, 253.720994234085, 256.680767297745, 259.394066333771, 262.358667373657]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.649999976158</td>\n",
+       "        <td>0.545458972454</td>\n",
+       "        <td>[0.658333361148834, 0.649999976158142, 0.641666650772095, 0.641666650772095, 0.641666650772095, 0.666666686534882, 0.666666686534882, 0.666666686534882, 0.666666686534882, 0.649999976158142]</td>\n",
+       "        <td>[0.616556286811829, 0.493021905422211, 0.488242834806442, 0.486079841852188, 0.479775160551071, 0.482189744710922, 0.496939599514008, 0.479279518127441, 0.543927192687988, 0.545458972454071]</td>\n",
+       "        <td>0.800000011921</td>\n",
+       "        <td>0.362693428993</td>\n",
+       "        <td>[0.633333325386047, 0.800000011920929, 0.766666650772095, 0.766666650772095, 0.766666650772095, 0.666666686534882, 0.666666686534882, 0.666666686534882, 0.666666686534882, 0.800000011920929]</td>\n",
+       "        <td>[0.53449285030365, 0.394388288259506, 0.390281409025192, 0.385460764169693, 0.392662823200226, 0.410547375679016, 0.439140349626541, 0.395850986242294, 0.503270268440247, 0.362693428993225]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.06312207575548352)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[29.8041570186615, 32.351331949234, 34.8423700332642, 37.4489560127258, 40.3089909553528, 42.915864944458, 45.6521019935608, 47.9889349937439, 50.5978739261627, 53.2138829231262]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.641666650772</td>\n",
+       "        <td>0.471347987652</td>\n",
+       "        <td>[0.666666686534882, 0.641666650772095, 0.641666650772095, 0.366666674613953, 0.666666686534882, 0.966666638851166, 0.483333319425583, 0.641666650772095, 0.941666662693024, 0.641666650772095]</td>\n",
+       "        <td>[0.520724713802338, 0.578253924846649, 0.518827021121979, 1.67398142814636, 0.512235522270203, 0.131752595305443, 1.25291848182678, 0.453146934509277, 0.185879185795784, 0.471347987651825]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.337222576141</td>\n",
+       "        <td>[0.666666686534882, 0.766666650772095, 0.766666650772095, 0.333333343267441, 0.666666686534882, 0.933333337306976, 0.566666662693024, 0.766666650772095, 0.899999976158142, 0.766666650772095]</td>\n",
+       "        <td>[0.462848216295242, 0.38900101184845, 0.356701970100403, 1.92939758300781, 0.458910882472992, 0.14183434844017, 0.83171159029007, 0.332331091165543, 0.311882764101028, 0.337222576141357]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.0658839037116738)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[211.389490127563, 213.85792016983, 216.39094209671, 218.503707170486, 220.992606163025, 223.471295118332, 226.145341157913, 228.394971132278, 230.912840127945, 233.303599119186]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.641666650772</td>\n",
+       "        <td>0.482037603855</td>\n",
+       "        <td>[0.800000011920929, 0.666666686534882, 0.666666686534882, 0.808333337306976, 0.683333337306976, 0.666666686534882, 0.875, 0.666666686534882, 0.666666686534882, 0.641666650772095]</td>\n",
+       "        <td>[0.508525788784027, 0.431755125522614, 0.435203284025192, 0.344938695430756, 0.478766769170761, 0.330143094062805, 0.273075610399246, 0.479535788297653, 0.48390719294548, 0.482037603855133]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.39202862978</td>\n",
+       "        <td>[0.833333313465118, 0.666666686534882, 0.666666686534882, 0.899999976158142, 0.666666686534882, 0.666666686534882, 0.933333337306976, 0.666666686534882, 0.666666686534882, 0.766666650772095]</td>\n",
+       "        <td>[0.443316102027893, 0.401963800191879, 0.399077832698822, 0.240811541676521, 0.410095393657684, 0.290450870990753, 0.254536032676697, 0.396871030330658, 0.413692444562912, 0.392028629779816]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='RMSprop(lr=0.0020197253642543623)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[109.946217060089, 112.64745092392, 115.244435071945, 117.609509944916, 120.211313962936, 122.821636915207, 125.485604047775, 128.088088035583, 130.593991041183, 133.237344026566]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.725000023842</td>\n",
+       "        <td>0.553534567356</td>\n",
+       "        <td>[0.358333319425583, 0.358333319425583, 0.508333325386047, 0.916666686534882, 0.658333361148834, 0.633333325386047, 0.633333325386047, 0.641666650772095, 0.649999976158142, 0.725000023841858]</td>\n",
+       "        <td>[1.44853103160858, 1.05627000331879, 0.960374653339386, 0.903020858764648, 0.811912178993225, 0.744573473930359, 0.693612813949585, 0.683376550674438, 0.593345999717712, 0.55353456735611]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.458936661482</td>\n",
+       "        <td>[0.233333334326744, 0.233333334326744, 0.433333337306976, 0.933333337306976, 0.733333349227905, 0.733333349227905, 0.733333349227905, 0.766666650772095, 0.766666650772095, 0.766666650772095]</td>\n",
+       "        <td>[1.5010712146759, 1.05707538127899, 0.939342617988586, 0.863560140132904, 0.760088205337524, 0.681271374225616, 0.617161631584167, 0.568128407001495, 0.501981496810913, 0.458936661481857]</td>\n",
+       "        <td>[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(5, 2, u\"optimizer='RMSprop(lr=0.0084793872639979)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [56.9403030872345, 59.805566072464, 62.2339789867401, 64.8922078609467, 67.5616340637207, 70.2253429889679, 72.8736228942871, 75.5874469280243, 78.2902030944824, 80.9871909618378], [u'accuracy'], u'categorical_crossentropy', 0.975000023841858, 0.0910520926117897, [0.649999976158142, 0.891666650772095, 0.883333325386047, 0.949999988079071, 0.975000023841858, 0.883333325386047, 0.850000023841858, 0.949999988079071, 0.975000023841858, 0.975000023841858], [0.559232711791992, 0.335382640361786, 0.259929001331329, 0.158979862928391, 0.114544428884983, 0.269487291574478, 0.293675005435944, 0.0902178362011909, 0.0766977593302727, 0.0910520926117897], 0.966666638851166, 0.0768957436084747, [0.833333313465118, 0.933333337306976, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.899999976158142, 0.899999976158142, 0.933333337306976, 0.966666638851166, 0.966666638851166], [0.485892802476883, 0.249617904424667, 0.258282363414764, 0.11016520857811, 0.0912857726216316, 0.280073672533035, 0.178015038371086, 0.087411992251873, 0.062506839632988, 0.0768957436084747], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (14, 1, u\"optimizer='RMSprop(lr=0.03366551083145706)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [161.690346240997, 164.041937112808, 166.454420089722, 168.906048059464, 171.067217111588, 173.555004119873, 175.944698095322, 178.445127248764, 180.502294063568, 183.037788152695], [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.144752591848373, [0.316666662693024, 0.666666686534882, 0.716666638851166, 0.641666650772095, 0.675000011920929, 0.975000023841858, 0.791666686534882, 0.975000023841858, 0.966666638851166, 0.949999988079071], [1.57745933532715, 0.405172228813171, 0.471270889043808, 1.00022745132446, 0.840015530586243, 0.128021001815796, 0.473532497882843, 0.091586634516716, 0.112696528434753, 0.144752591848373], 0.966666638851166, 0.100186347961426, [0.433333337306976, 0.666666686534882, 0.899999976158142, 0.766666650772095, 0.866666674613953, 0.966666638851166, 0.899999976158142, 0.933333337306976, 0.966666638851166, 0.966666638851166], [1.36917245388031, 0.372486144304276, 0.266377687454224, 0.571163833141327, 0.457086622714996, 0.103041857481003, 0.276452839374542, 0.0908508822321892, 0.0997116342186928, 0.100186347961426], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (15, 1, u\"optimizer='Adam(lr=0.009794369846837002)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [186.611872196198, 188.95641207695, 191.037184238434, 193.397297143936, 195.740861177444, 197.805513143539, 200.180992126465, 202.689172029495, 205.040098190308, 207.208242177963], [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.116746708750725, [0.75, 0.975000023841858, 0.850000023841858, 0.941666662693024, 0.933333337306976, 0.949999988079071, 0.949999988079071, 0.983333349227905, 0.958333313465118, 0.949999988079071], [0.5159512758255, 0.353324204683304, 0.333910763263702, 0.245715036988258, 0.188893154263496, 0.161517903208733, 0.137443989515305, 0.122971840202808, 0.14612153172493, 0.116746708750725], 0.966666638851166, 0.106192082166672, [0.899999976158142, 0.966666638851166, 0.899999976158142, 0.966666638851166, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166], [0.423073083162308, 0.298538327217102, 0.234973803162575, 0.176778241991997, 0.170526877045631, 0.145023569464684, 0.119270212948322, 0.103897586464882, 0.104170136153698, 0.106192082166672], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (6, 2, u\"optimizer='Adam(lr=0.007581048101981366)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [57.225380897522, 60.0725290775299, 62.731920003891, 65.1544499397278, 67.8143260478973, 70.4762139320374, 73.1227269172668, 75.8475530147552, 78.555095911026, 81.2564718723297], [u'accuracy'], u'categorical_crossentropy', 0.958333313465118, 0.133664950728416, [0.649999976158142, 0.883333325386047, 0.941666662693024, 0.899999976158142, 0.958333313465118, 0.925000011920929, 0.941666662693024, 0.933333337306976, 0.983333349227905, 0.958333313465118], [1.03808128833771, 0.883756637573242, 0.686505734920502, 0.517532765865326, 0.401096671819687, 0.259793311357498, 0.177235946059227, 0.168946355581284, 0.128713861107826, 0.133664950728416], 0.966666638851166, 0.1226412281394, [0.633333325386047, 0.899999976158142, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.933333337306976, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.966666638851166], [1.02865540981293, 0.850838720798492, 0.612184524536133, 0.452387690544128, 0.33221709728241, 0.23906472325325, 0.165990635752678, 0.164969280362129, 0.12097629904747, 0.1226412281394], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (12, 1, u\"optimizer='RMSprop(lr=0.012596538573477555)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [137.034833192825, 139.500956058502, 141.807043075562, 143.897747039795, 146.264532089233, 148.888093233109, 150.934833049774, 153.475459098816, 155.848874092102, 158.239592075348], [u'accuracy'], u'categorical_crossentropy', 0.916666686534882, 0.184266731142998, [0.566666662693024, 0.933333337306976, 0.649999976158142, 0.666666686534882, 0.808333337306976, 0.891666650772095, 0.891666650772095, 0.975000023841858, 0.933333337306976, 0.916666686534882], [0.829304933547974, 0.631127297878265, 0.597909092903137, 0.552545011043549, 0.428654760122299, 0.233174994587898, 0.236562281847, 0.119615346193314, 0.191903278231621, 0.184266731142998], 0.966666638851166, 0.12465588003397, [0.699999988079071, 0.899999976158142, 0.800000011920929, 0.833333313465118, 0.899999976158142, 0.933333337306976, 0.866666674613953, 1.0, 0.966666638851166, 0.966666638851166], [0.74066150188446, 0.532437741756439, 0.480882078409195, 0.435436576604843, 0.310187846422195, 0.234515085816383, 0.247250944375992, 0.107902131974697, 0.136675015091896, 0.12465588003397], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (9, 2, u\"optimizer='RMSprop(lr=0.005441362966347114)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [109.630754947662, 112.341638088226, 114.763943910599, 117.389002084732, 119.991095066071, 122.593973875046, 125.238317966461, 127.8488509655, 130.375853061676, 133.020073890686], [u'accuracy'], u'categorical_crossentropy', 0.916666686534882, 0.216380029916763, [0.641666650772095, 0.733333349227905, 0.649999976158142, 0.858333349227905, 0.958333313465118, 0.699999988079071, 0.916666686534882, 0.858333349227905, 0.966666638851166, 0.916666686534882], [0.73615950345993, 0.489604264497757, 0.45263683795929, 0.37542662024498, 0.334106951951981, 0.419453173875809, 0.284875482320786, 0.27151495218277, 0.185230866074562, 0.216380029916763], 0.966666638851166, 0.127150848507881, [0.766666650772095, 0.699999988079071, 0.800000011920929, 0.933333337306976, 0.966666638851166, 0.666666686534882, 0.899999976158142, 0.933333337306976, 0.966666638851166, 0.966666638851166], [0.631976008415222, 0.448555260896683, 0.323729306459427, 0.28750941157341, 0.281407296657562, 0.421543717384338, 0.259464651346207, 0.158164814114571, 0.135125860571861, 0.127150848507881], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.04966544234738768)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [2.26167011260986, 4.64152717590332, 7.38891315460205, 10.1575191020966, 12.6948411464691, 15.5193021297455, 17.8266370296478, 20.364767074585, 23.1241211891174, 25.6702241897583], [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.183234125375748, [0.641666650772095, 0.666666686534882, 0.966666638851166, 0.958333313465118, 0.716666638851166, 0.983333349227905, 0.966666638851166, 0.983333349227905, 0.975000023841858, 0.949999988079071], [0.829066038131714, 0.490932732820511, 0.341925740242004, 0.215810611844063, 0.400910943746567, 0.107548490166664, 0.0985226780176163, 0.0732712596654892, 0.070111908018589, 0.183234125375748], 0.966666638851166, 0.137178555130959, [0.866666674613953, 0.666666686534882, 0.966666638851166, 0.966666638851166, 0.699999988079071, 0.966666638851166, 0.966666638851166, 0.933333337306976, 1.0, 0.966666638851166], [0.843752980232239, 0.435137718915939, 0.26888644695282, 0.157842606306076, 0.406648069620132, 0.0976309478282928, 0.0788726136088371, 0.0751720294356346, 0.0686705932021141, 0.137178555130959], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.0023223781742022285)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [29.5305609703064, 31.8714768886566, 34.5841488838196, 37.194139957428, 40.0518889427185, 42.6473689079285, 44.9830038547516, 47.734827041626, 50.3327059745789, 52.9546790122986], [u'accuracy'], u'categorical_crossentropy', 0.966666638851166, 0.247659549117088, [0.641666650772095, 0.691666662693024, 0.641666650772095, 0.966666638851166, 0.941666662693024, 0.958333313465118, 0.933333337306976, 0.925000011920929, 0.966666638851166, 0.966666638851166], [0.949797213077545, 0.797480285167694, 0.672827661037445, 0.523928940296173, 0.444453626871109, 0.386174380779266, 0.347499698400497, 0.321201831102371, 0.278330504894257, 0.247659549117088], 0.966666638851166, 0.198830261826515, [0.766666650772095, 0.833333313465118, 0.766666650772095, 0.966666638851166, 0.966666638851166, 0.933333337306976, 0.966666638851166, 0.899999976158142, 0.966666638851166, 0.966666638851166], [0.931245148181915, 0.763193428516388, 0.600658059120178, 0.456866592168808, 0.373299777507782, 0.326453566551208, 0.275230079889297, 0.284671515226364, 0.221188083291054, 0.198830261826515], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (7, 1, u\"optimizer='RMSprop(lr=0.0014467801648012073)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [84.6684620380402, 86.9988820552826, 89.4947271347046, 91.8522582054138, 93.9668672084808, 96.3584721088409, 98.7448561191559, 101.170181035995, 103.282526016235, 105.678196191788], [u'accuracy'], u'categorical_crossentropy', 0.966666638851166, 0.395356565713882, [0.625, 0.641666650772095, 0.741666674613953, 0.883333325386047, 0.833333313465118, 0.941666662693024, 0.949999988079071, 0.983333349227905, 0.875, 0.966666638851166], [0.941141128540039, 0.820547699928284, 0.723011374473572, 0.646571576595306, 0.57060444355011, 0.514499425888062, 0.46852970123291, 0.439025938510895, 0.416335105895996, 0.395356565713882], 0.966666638851166, 0.326709568500519, [0.766666650772095, 0.766666650772095, 0.899999976158142, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.933333337306976, 0.966666638851166], [0.941896796226501, 0.816571235656738, 0.707062959671021, 0.61734527349472, 0.521940350532532, 0.45942959189415, 0.405136495828629, 0.374987095594406, 0.33730074763298, 0.326709568500519], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (19, 2, u\"optimizer='Adam(lr=0.0011757973913283008)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [237.369596242905, 240.072783231735, 242.490661382675, 245.210073232651, 247.916568279266, 250.679228305817, 253.465747356415, 256.328309297562, 259.131007194519, 262.085569381714], [u'accuracy'], u'categorical_crossentropy', 0.916666686534882, 0.567601382732391, [0.333333343267441, 0.441666662693024, 0.433333337306976, 0.349999994039536, 0.349999994039536, 0.675000011920929, 0.683333337306976, 0.800000011920929, 0.774999976158142, 0.916666686534882], [1.14459049701691, 1.08465170860291, 1.02045607566833, 0.952999651432037, 0.883513271808624, 0.809052526950836, 0.748401284217834, 0.682490110397339, 0.620046377182007, 0.567601382732391], 0.966666638851166, 0.512901842594147, [0.333333343267441, 0.400000005960464, 0.400000005960464, 0.366666674613953, 0.366666674613953, 0.800000011920929, 0.800000011920929, 0.866666674613953, 0.866666674613953, 0.966666638851166], [1.21284127235413, 1.13662087917328, 1.04775261878967, 0.957895994186401, 0.871163666248322, 0.780500650405884, 0.705705106258392, 0.636253237724304, 0.558390736579895, 0.512901842594147], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (17, 1, u\"optimizer='RMSprop(lr=0.003695186053629043)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [211.166339159012, 213.628123044968, 216.070516109467, 218.28012919426, 220.77138209343, 223.166202068329, 225.899930000305, 228.167545080185, 230.690491199493, 233.073115110397], [u'accuracy'], u'categorical_crossentropy', 0.966666638851166, 0.114436306059361, [0.641666650772095, 0.875, 0.824999988079071, 0.824999988079071, 0.958333313465118, 0.866666674613953, 0.966666638851166, 0.941666662693024, 0.958333313465118, 0.966666638851166], [0.718437075614929, 0.535359025001526, 0.403026401996613, 0.348048120737076, 0.244051590561867, 0.264052510261536, 0.1720110476017, 0.164029255509377, 0.154526039958, 0.114436306059361], 0.933333337306976, 0.107093170285225, [0.766666650772095, 0.933333337306976, 0.899999976158142, 0.833333313465118, 0.933333337306976, 0.933333337306976, 0.966666638851166, 0.933333337306976, 0.966666638851166, 0.933333337306976], [0.647899329662323, 0.474290877580643, 0.308415770530701, 0.318869024515152, 0.206334576010704, 0.250639617443085, 0.129751890897751, 0.156522572040558, 0.112601205706596, 0.107093170285225], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (11, 1, u\"optimizer='Adam(lr=0.03271173767396424)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [136.810528039932, 139.271977186203, 141.348733186722, 143.678931236267, 146.042705059052, 148.659409046173, 150.713799238205, 153.240673065186, 155.618861198425, 158.003229141235], [u'accuracy'], u'categorical_crossentropy', 0.899999976158142, 0.200596183538437, [0.341666668653488, 0.925000011920929, 0.958333313465118, 0.675000011920929, 0.966666638851166, 0.766666650772095, 0.975000023841858, 0.75, 0.975000023841858, 0.899999976158142], [1.04381895065308, 0.384325951337814, 0.263480663299561, 0.593676149845123, 0.141404688358307, 0.362050473690033, 0.0923048332333565, 0.351189643144608, 0.0946881100535393, 0.200596183538437], 0.899999976158142, 0.257636517286301, [0.533333361148834, 0.866666674613953, 0.966666638851166, 0.800000011920929, 0.966666638851166, 0.833333313465118, 0.966666638851166, 0.866666674613953, 0.966666638851166, 0.899999976158142], [0.91362202167511, 0.340268641710281, 0.21289549767971, 0.362329840660095, 0.136535987257957, 0.327440768480301, 0.111000411212444, 0.227803841233253, 0.111130490899086, 0.257636517286301], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (13, 1, u\"optimizer='RMSprop(lr=0.0027814197503322115)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [161.464279174805, 163.823823213577, 166.230634212494, 168.657960176468, 170.846117019653, 173.335704088211, 175.715650081635, 178.106207132339, 180.278338193893, 182.815184116364], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.351116120815277, [0.600000023841858, 0.558333337306976, 0.541666686534882, 0.600000023841858, 0.899999976158142, 0.916666686534882, 0.850000023841858, 0.925000011920929, 0.933333337306976, 0.933333337306976], [0.9764444231987, 0.860457479953766, 0.76110851764679, 0.688599288463593, 0.623845517635345, 0.558117687702179, 0.516568541526794, 0.438872784376144, 0.389935582876205, 0.351116120815277], 0.899999976158142, 0.326276868581772, [0.433333337306976, 0.433333337306976, 0.5, 0.433333337306976, 0.899999976158142, 0.866666674613953, 0.933333337306976, 0.866666674613953, 0.899999976158142, 0.899999976158142], [1.03803777694702, 0.913994610309601, 0.794906616210938, 0.723303020000458, 0.652373254299164, 0.566653609275818, 0.481746405363083, 0.454111516475677, 0.379933565855026, 0.326276868581772], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (2, 2, u\"optimizer='Adam(lr=0.035340389425615855)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [2.53016018867493, 5.03768014907837, 7.64597201347351, 10.4127900600433, 13.0584251880646, 15.7928349971771, 18.0860531330109, 20.625785112381, 23.3826050758362, 25.9297461509705], [u'accuracy'], u'categorical_crossentropy', 0.875, 0.28657540678978, [0.850000023841858, 0.491666674613953, 0.941666662693024, 0.783333361148834, 0.925000011920929, 0.850000023841858, 0.966666638851166, 0.958333313465118, 0.908333361148834, 0.875], [0.313798636198044, 0.746647894382477, 0.232485517859459, 0.384049296379089, 0.201492115855217, 0.276773244142532, 0.144450753927231, 0.116710871458054, 0.210491970181465, 0.28657540678978], 0.899999976158142, 0.334158778190613, [0.833333313465118, 0.533333361148834, 0.833333313465118, 0.899999976158142, 0.899999976158142, 0.833333313465118, 0.966666638851166, 1.0, 0.899999976158142, 0.899999976158142], [0.281513780355453, 0.781840145587921, 0.252955704927444, 0.219736546278, 0.268592208623886, 0.332309901714325, 0.131899908185005, 0.0595534667372704, 0.255705177783966, 0.334158778190613], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (8, 1, u\"optimizer='Adam(lr=0.08208550087461897)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [84.8872451782227, 87.2208690643311, 89.7248260974884, 92.1014380455017, 94.1999170780182, 96.5836410522461, 98.9723200798035, 101.409075021744, 103.511981010437, 105.902093172073], [u'accuracy'], u'categorical_crossentropy', 0.766666650772095, 0.59165495634079, [0.333333343267441, 0.641666650772095, 0.566666662693024, 0.591666638851166, 0.758333325386047, 0.666666686534882, 0.966666638851166, 0.916666686534882, 0.949999988079071, 0.766666650772095], [1.02616000175476, 0.486202239990234, 0.636112213134766, 0.692184090614319, 0.505898773670197, 0.467963546514511, 0.268672525882721, 0.176122322678566, 0.122547559440136, 0.59165495634079], 0.899999976158142, 0.33915963768959, [0.333333343267441, 0.766666650772095, 0.733333349227905, 0.766666650772095, 0.733333349227905, 0.666666686534882, 0.966666638851166, 0.899999976158142, 0.899999976158142, 0.899999976158142], [1.12080752849579, 0.381140530109406, 0.5174320936203, 0.473030716180801, 0.607473373413086, 0.409501492977142, 0.212725415825844, 0.296080023050308, 0.18353745341301, 0.33915963768959], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (16, 1, u\"optimizer='Adam(lr=0.0025753473010720596)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [186.835414171219, 189.195631027222, 191.254861116409, 193.619318246841, 195.966614246368, 198.311106204987, 200.407158136368, 202.908673048019, 205.313441038132, 207.438939094543], [u'accuracy'], u'categorical_crossentropy', 0.891666650772095, 0.950204193592072, [0.358333319425583, 0.358333319425583, 0.358333319425583, 0.449999988079071, 0.400000005960464, 0.516666650772095, 0.583333313465118, 0.608333349227905, 0.899999976158142, 0.891666650772095], [1.09640550613403, 1.08471190929413, 1.0751405954361, 1.06720495223999, 1.06165635585785, 1.0469172000885, 1.0301650762558, 1.00463593006134, 0.979537725448608, 0.950204193592072], 0.866666674613953, 0.931918859481812, [0.233333334326744, 0.233333334326744, 0.233333334326744, 0.366666674613953, 0.266666680574417, 0.400000005960464, 0.400000005960464, 0.433333337306976, 0.833333313465118, 0.866666674613953], [1.10118973255157, 1.08938491344452, 1.07863283157349, 1.06669425964355, 1.06701147556305, 1.04324889183044, 1.02750730514526, 0.996739327907562, 0.966157376766205, 0.931918859481812], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (20, 2, u\"optimizer='RMSprop(lr=0.056702169442788934)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [237.640507221222, 240.350925207138, 243.027331352234, 245.468372344971, 248.176068305969, 250.940598249435, 253.720994234085, 256.680767297745, 259.394066333771, 262.358667373657], [u'accuracy'], u'categorical_crossentropy', 0.649999976158142, 0.545458972454071, [0.658333361148834, 0.649999976158142, 0.641666650772095, 0.641666650772095, 0.641666650772095, 0.666666686534882, 0.666666686534882, 0.666666686534882, 0.666666686534882, 0.649999976158142], [0.616556286811829, 0.493021905422211, 0.488242834806442, 0.486079841852188, 0.479775160551071, 0.482189744710922, 0.496939599514008, 0.479279518127441, 0.543927192687988, 0.545458972454071], 0.800000011920929, 0.362693428993225, [0.633333325386047, 0.800000011920929, 0.766666650772095, 0.766666650772095, 0.766666650772095, 0.666666686534882, 0.666666686534882, 0.666666686534882, 0.666666686534882, 0.800000011920929], [0.53449285030365, 0.394388288259506, 0.390281409025192, 0.385460764169693, 0.392662823200226, 0.410547375679016, 0.439140349626541, 0.395850986242294, 0.503270268440247, 0.362693428993225], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (4, 2, u\"optimizer='Adam(lr=0.06312207575548352)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [29.8041570186615, 32.351331949234, 34.8423700332642, 37.4489560127258, 40.3089909553528, 42.915864944458, 45.6521019935608, 47.9889349937439, 50.5978739261627, 53.2138829231262], [u'accuracy'], u'categorical_crossentropy', 0.641666650772095, 0.471347987651825, [0.666666686534882, 0.641666650772095, 0.641666650772095, 0.366666674613953, 0.666666686534882, 0.966666638851166, 0.483333319425583, 0.641666650772095, 0.941666662693024, 0.641666650772095], [0.520724713802338, 0.578253924846649, 0.518827021121979, 1.67398142814636, 0.512235522270203, 0.131752595305443, 1.25291848182678, 0.453146934509277, 0.185879185795784, 0.471347987651825], 0.766666650772095, 0.337222576141357, [0.666666686534882, 0.766666650772095, 0.766666650772095, 0.333333343267441, 0.666666686534882, 0.933333337306976, 0.566666662693024, 0.766666650772095, 0.899999976158142, 0.766666650772095], [0.462848216295242, 0.38900101184845, 0.356701970100403, 1.92939758300781, 0.458910882472992, 0.14183434844017, 0.83171159029007, 0.332331091165543, 0.311882764101028, 0.337222576141357], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (18, 1, u\"optimizer='RMSprop(lr=0.0658839037116738)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [211.389490127563, 213.85792016983, 216.39094209671, 218.503707170486, 220.992606163025, 223.471295118332, 226.145341157913, 228.394971132278, 230.912840127945, 233.303599119186], [u'accuracy'], u'categorical_crossentropy', 0.641666650772095, 0.482037603855133, [0.800000011920929, 0.666666686534882, 0.666666686534882, 0.808333337306976, 0.683333337306976, 0.666666686534882, 0.875, 0.666666686534882, 0.666666686534882, 0.641666650772095], [0.508525788784027, 0.431755125522614, 0.435203284025192, 0.344938695430756, 0.478766769170761, 0.330143094062805, 0.273075610399246, 0.479535788297653, 0.48390719294548, 0.482037603855133], 0.766666650772095, 0.392028629779816, [0.833333313465118, 0.666666686534882, 0.666666686534882, 0.899999976158142, 0.666666686534882, 0.666666686534882, 0.933333337306976, 0.666666686534882, 0.666666686534882, 0.766666650772095], [0.443316102027893, 0.401963800191879, 0.399077832698822, 0.240811541676521, 0.410095393657684, 0.290450870990753, 0.254536032676697, 0.396871030330658, 0.413692444562912, 0.392028629779816], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),\n",
+       " (10, 1, u\"optimizer='RMSprop(lr=0.0020197253642543623)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [109.946217060089, 112.64745092392, 115.244435071945, 117.609509944916, 120.211313962936, 122.821636915207, 125.485604047775, 128.088088035583, 130.593991041183, 133.237344026566], [u'accuracy'], u'categorical_crossentropy', 0.725000023841858, 0.55353456735611, [0.358333319425583, 0.358333319425583, 0.508333325386047, 0.916666686534882, 0.658333361148834, 0.633333325386047, 0.633333325386047, 0.641666650772095, 0.649999976158142, 0.725000023841858], [1.44853103160858, 1.05627000331879, 0.960374653339386, 0.903020858764648, 0.811912178993225, 0.744573473930359, 0.693612813949585, 0.683376550674438, 0.593345999717712, 0.55353456735611], 0.766666650772095, 0.458936661481857, [0.233333334326744, 0.233333334326744, 0.433333337306976, 0.933333337306976, 0.733333349227905, 0.733333349227905, 0.733333349227905, 0.766666650772095, 0.766666650772095, 0.766666650772095], [1.5010712146759, 1.05707538127899, 0.939342617988586, 0.863560140132904, 0.760088205337524, 0.681271374225616, 0.617161631584167, 0.568128407001495, 0.501981496810913, 0.458936661481857], [1, 2, 3, 4, 5, 6, 7, 8, 9, 10])]"
+      ]
+     },
+     "execution_count": 24,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM automl_output_info ORDER BY validation_metrics_final DESC, validation_loss_final;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Plot results"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 25,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "20 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM automl_output_info;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "df_summary = %sql SELECT * FROM automl_output_summary;\n",
+    "df_summary = df_summary.DataFrame()\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Iteration')\n",
+    "ax_metric.set_ylabel('Metric')\n",
+    "ax_metric.set_title('Validation metric curve')\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Iteration')\n",
+    "ax_loss.set_ylabel('Loss')\n",
+    "ax_loss.set_title('Validation loss curve')\n",
+    "\n",
+    "for mst_key in df_results['mst_key']:\n",
+    "    df_output_info = %sql SELECT validation_metrics,validation_loss,metrics_iters FROM automl_output_info WHERE mst_key = $mst_key;\n",
+    "    df_output_info = df_output_info.DataFrame()\n",
+    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
+    "    validation_loss = df_output_info['validation_loss'][0]\n",
+    "    iters = df_output_info['metrics_iters'][0]\n",
+    "    \n",
+    "    #ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
+    "    ax_metric.plot(iters, validation_metrics)\n",
+    "    #ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
+    "    ax_loss.plot(iters, validation_loss)\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Show each trial"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 26,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "20 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM automl_output_info;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Trial')\n",
+    "#ax_metric.set_ylabel('Accuracy')\n",
+    "ax_metric.set_title('Validation Accuracy')\n",
+    "#ax_metric.lines.remove(ax_metric.lines)\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Trial')\n",
+    "#ax_loss.set_ylabel('Cross Entropy Loss')\n",
+    "ax_loss.set_title('Validation Loss (Cross Entropy)')\n",
+    "\n",
+    "validation_metrics_final = df_results['validation_metrics_final']\n",
+    "validation_loss_final = df_results['validation_loss_final']\n",
+    "iters = df_results['mst_key']\n",
+    "#iters = [x - (iters[0]-1) for x in iters]\n",
+    "\n",
+    "#ax_metric.plot(iters, training_metrics_final, label=mst_key, marker='o')\n",
+    "ax_metric.plot(iters, validation_metrics_final, marker='o', linestyle='None', markersize=4)\n",
+    "#ax_metric.plot(iters, training_metrics)\n",
+    "    \n",
+    "#ax_loss.plot(iters, training_loss_final, label=mst_key, marker='o')\n",
+    "ax_loss.plot(iters, validation_loss_final, marker='o', linestyle='None', markersize=4)\n",
+    "#ax_loss.plot(iters, training_loss)\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Show best by trial"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 27,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "20 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n",
+      "3 rows affected.\n",
+      "4 rows affected.\n",
+      "5 rows affected.\n",
+      "6 rows affected.\n",
+      "7 rows affected.\n",
+      "8 rows affected.\n",
+      "9 rows affected.\n",
+      "10 rows affected.\n",
+      "11 rows affected.\n",
+      "12 rows affected.\n",
+      "13 rows affected.\n",
+      "14 rows affected.\n",
+      "15 rows affected.\n",
+      "16 rows affected.\n",
+      "17 rows affected.\n",
+      "18 rows affected.\n",
+      "19 rows affected.\n",
+      "20 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM automl_output_info ORDER BY mst_key;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "best_so_far_acc = []\n",
+    "best_so_far_loss = []\n",
+    "\n",
+    "for mst_key in df_results['mst_key']:\n",
+    "    df_output_info = %sql SELECT mst_key, validation_metrics_final, validation_loss_final FROM automl_output_info WHERE mst_key <= $mst_key; \n",
+    "    df_output_info = df_output_info.DataFrame()\n",
+    "    best_so_far_acc.append([mst_key, df_output_info['validation_metrics_final'].max()])\n",
+    "    best_so_far_loss.append([mst_key, df_output_info['validation_loss_final'].min()])\n",
+    "\n",
+    "df1 = pd.DataFrame(best_so_far_acc,columns=['Trial','Validation Accuracy'])\n",
+    "df2 = pd.DataFrame(best_so_far_loss,columns=['Trial','Validation Loss'])\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Trial')\n",
+    "#ax_metric.set_ylabel('Accuracy')\n",
+    "ax_metric.set_title('Best Validation Accuracy')\n",
+    "#ax_metric.lines.remove(ax_metric.lines)\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Trial')\n",
+    "#ax_loss.set_ylabel('Cross Entropy Loss')\n",
+    "ax_loss.set_title('Best Validation Loss (Cross Entropy)')\n",
+    "\n",
+    "validation_metrics_final = df1['Validation Accuracy']\n",
+    "validation_loss_final = df2['Validation Loss']\n",
+    "iters1 = df1['Trial']\n",
+    "iters2 = df2['Trial']\n",
+    "\n",
+    "#ax_metric.plot(iters1, training_metrics_final, label=mst_key, marker='o')\n",
+    "#ax_metric.plot(iters1, validation_metrics_final, marker='o', linestyle='None', markersize=4)\n",
+    "#ax_metric.plot(iters1, validation_metrics_final, marker='o', markersize=0.5)\n",
+    "ax_metric.plot(iters1, validation_metrics_final)\n",
+    "    \n",
+    "#ax_loss.plot(iters2, training_loss_final, label=mst_key, marker='o')\n",
+    "#ax_loss.plot(iters2, validation_loss_final, marker='o', linestyle='None', markersize=4)\n",
+    "ax_loss.plot(iters2, validation_loss_final)\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred\"></a>\n",
+    "# 6. Predict\n",
+    "\n",
+    "Now predict using model we built.  We will use the validation data set for prediction as well, which is not usual but serves to show the syntax:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 28,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "30 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.77083427</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.76736474</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.7637215</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.76102996</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.7710857</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.7592268</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.7686342</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.76880336</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.7689748</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.77831817</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.7722787</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>47</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.76963973</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>50</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.7690843</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5698719</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>65</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.71937233</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>69</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7692964</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8146459</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>79</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5106894</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7508131</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.77062976</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>105</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9310901</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9202143</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>110</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9461502</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.53245807</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>130</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5506391</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.58871216</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>139</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.76892227</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.94731</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7525016</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5657851</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'class_text', u'Iris-setosa', 0.77083427),\n",
+       " (2, u'class_text', u'Iris-setosa', 0.76736474),\n",
+       " (3, u'class_text', u'Iris-setosa', 0.7637215),\n",
+       " (4, u'class_text', u'Iris-setosa', 0.76102996),\n",
+       " (6, u'class_text', u'Iris-setosa', 0.7710857),\n",
+       " (7, u'class_text', u'Iris-setosa', 0.7592268),\n",
+       " (8, u'class_text', u'Iris-setosa', 0.7686342),\n",
+       " (10, u'class_text', u'Iris-setosa', 0.76880336),\n",
+       " (18, u'class_text', u'Iris-setosa', 0.7689748),\n",
+       " (19, u'class_text', u'Iris-setosa', 0.77831817),\n",
+       " (28, u'class_text', u'Iris-setosa', 0.7722787),\n",
+       " (47, u'class_text', u'Iris-setosa', 0.76963973),\n",
+       " (50, u'class_text', u'Iris-setosa', 0.7690843),\n",
+       " (60, u'class_text', u'Iris-virginica', 0.5698719),\n",
+       " (65, u'class_text', u'Iris-versicolor', 0.71937233),\n",
+       " (69, u'class_text', u'Iris-versicolor', 0.7692964),\n",
+       " (75, u'class_text', u'Iris-versicolor', 0.8146459),\n",
+       " (79, u'class_text', u'Iris-versicolor', 0.5106894),\n",
+       " (80, u'class_text', u'Iris-versicolor', 0.7508131),\n",
+       " (82, u'class_text', u'Iris-versicolor', 0.77062976),\n",
+       " (105, u'class_text', u'Iris-virginica', 0.9310901),\n",
+       " (107, u'class_text', u'Iris-virginica', 0.9202143),\n",
+       " (110, u'class_text', u'Iris-virginica', 0.9461502),\n",
+       " (120, u'class_text', u'Iris-versicolor', 0.53245807),\n",
+       " (130, u'class_text', u'Iris-versicolor', 0.5506391),\n",
+       " (136, u'class_text', u'Iris-virginica', 0.58871216),\n",
+       " (139, u'class_text', u'Iris-virginica', 0.76892227),\n",
+       " (145, u'class_text', u'Iris-virginica', 0.94731),\n",
+       " (146, u'class_text', u'Iris-virginica', 0.7525016),\n",
+       " (147, u'class_text', u'Iris-virginica', 0.5657851)]"
+      ]
+     },
+     "execution_count": 28,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('automl_output',    -- model\n",
+    "                                   'iris_test',        -- test_table\n",
+    "                                   'id',               -- id column\n",
+    "                                   'attributes',       -- independent var\n",
+    "                                   'iris_predict',     -- output table\n",
+    "                                    'response',        -- prediction type\n",
+    "                                    FALSE,             -- use gpus\n",
+    "                                    13                 -- MST key\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 31,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3L,)]"
+      ]
+     },
+     "execution_count": 31,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict JOIN iris_test USING (id) \n",
+    "WHERE iris_predict.class_value != iris_test.class_text;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 32,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90.00</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('90.00'),)]"
+      ]
+     },
+     "execution_count": 32,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict.class_value as estimated\n",
+    "     from iris_predict inner join iris_test\n",
+    "     on iris_test.id=iris_predict.id) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Train-multiple-models/Define-model-configurations-v2.ipynb b/community-artifacts/Deep-learning/Train-multiple-models/Define-model-configurations-v2.ipynb
new file mode 100755
index 0000000..fbe5ee5
--- /dev/null
+++ b/community-artifacts/Deep-learning/Train-multiple-models/Define-model-configurations-v2.ipynb
@@ -0,0 +1,2025 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Define model configurations\n",
+    "This module generates model configurations using grid search or random search.\n",
+    "\n",
+    "Once the configurations are defined, they can be used by the fit function in Train Model Configurations. By model configurations we mean both hyperparameters and model architectures. The output table from this module defines the combinations of model architectures, compile and fit parameters to be trained in parallel.\n",
+    "\n",
+    "This utility was added in MADlib 1.17.0.  Improvements were made in MADlib 1.18.0 including support for custom loss functions and custom metrics.\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#define_model_arch\">1. Define model architecture table</a>\n",
+    "\n",
+    "<a href=\"#load_model_arch\">2. Load model architecture</a>\n",
+    "\n",
+    "<a href=\"#generate_configs\">3. Generate model configurations</a>\n",
+    "\n",
+    "  - <a href=\"#grid_search\">3a. Grid search</a>\n",
+    "  \n",
+    "  - <a href=\"#random_search\">3b. Random search</a>\n",
+    "  \n",
+    "  - <a href=\"#incremental_load\">3c. Incremental loading</a>\n",
+    "  \n",
+    "<a href=\"#load_model_selection_manual\">4. Create model selection table manually</a>\n",
+    "\n",
+    "<a href=\"#custom\">5. Custom loss functions and custom metrics NOT COMPLETE</a>\n",
+    "\n",
+    "<a href=\"#load_model_selection\">6. Load model selection table [deprecated]</a>\n"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 37,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "The sql extension is already loaded. To reload it, use:\n",
+      "  %reload_ext sql\n"
+     ]
+    }
+   ],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 38,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP (PM demo machine) - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 39,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 39,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"define_model_arch\"></a>\n",
+    "# 1. Define model architecture table\n",
+    "The model selection loader works in conjunction with the model architecture table, so we first create a model architecture table with two different models.  See http://madlib.apache.org/docs/latest/group__grp__keras__model__arch.html for more details on the model architecture table.\n",
+    "\n",
+    "Import Keras libraries"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 40,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 1 hidden layer:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 41,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_4\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_14 (Dense)             (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_15 (Dense)             (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_16 (Dense)             (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 193\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model1 = Sequential()\n",
+    "model1.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model1.add(Dense(10, activation='relu'))\n",
+    "model1.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model1.summary();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 42,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_14\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_15\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_16\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_4\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 42,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model1.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 2, \"batch_input_shape\": [null, 3], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"new_dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'\n",
+    "        "
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 2 hidden layers:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 43,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_5\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_17 (Dense)             (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_18 (Dense)             (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_19 (Dense)             (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_20 (Dense)             (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 303\n",
+      "Trainable params: 303\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model2 = Sequential()\n",
+    "model2.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model2.summary();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 44,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_17\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_18\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_19\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_20\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_5\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 44,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model2.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_model_arch\"></a>\n",
+    "# 2. Load model architecture\n",
+    "\n",
+    "Load both into model architecture table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 45,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>model_weights</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>__internal_madlib_id__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>MLP with 1 hidden layer</td>\n",
+       "        <td>__madlib_temp_61202069_1614901986_7314581__</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>MLP with 2 hidden layers</td>\n",
+       "        <td>__madlib_temp_12006647_1614901987_43673839__</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'MLP with 1 hidden layer', u'__madlib_temp_61202069_1614901986_7314581__'),\n",
+       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1835 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'MLP with 2 hidden layers', u'__madlib_temp_12006647_1614901987_43673839__')]"
+      ]
+     },
+     "execution_count": 45,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS model_arch_library;\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Sophie',              -- Name\n",
+    "                               'MLP with 1 hidden layer'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Maria',               -- Name\n",
+    "                               'MLP with 2 hidden layers'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT * FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"generate_configs\"></a>\n",
+    "# 3. Generate model configurations\n",
+    "\n",
+    "<a id=\"grid_search\"></a>\n",
+    "## 3a. Grid search\n",
+    "\n",
+    "The output table for grid search contains the unique combinations of model architectures, compile and fit parameters."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 46,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "16 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128')]"
+      ]
+     },
+     "execution_count": 46,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [ {'optimizer': ['Adam', 'SGD'], 'lr': [0.001, 0.01]} ], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'grid'               -- search_type \n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Note that above uses the same learning rate for the two optimizers.  If you wanted to use different learning rates and different parameters for different optimizers (common):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 47,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "20 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 2, u\"optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 2, u\"optimizer='SGD()',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='SGD(lr=0.0001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001,momentum=0.95)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 1, u\"optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 1, u\"optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 1, u\"optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 1, u\"optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (17, 2, u\"optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (18, 2, u\"optimizer='Adam(lr=0.01,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (19, 2, u\"optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (20, 2, u\"optimizer='Adam(lr=0.1,decay=0.0001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128')]"
+      ]
+     },
+     "execution_count": 47,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [\n",
+    "                                                 {'optimizer': ['SGD']}, \n",
+    "                                                 {'optimizer': ['SGD'], 'lr': [0.0001, 0.001], 'momentum': [0.95]}, \n",
+    "                                                 {'optimizer': ['Adam'], 'lr': [0.01, 0.1], 'decay': [1e-4]}], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'grid'               -- search_type \n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"random_search\"></a>\n",
+    "## 3b. Random search\n",
+    "\n",
+    "The output table for random search contains the specified number of model architectures, compile and fit parameters, sampled from the specified distributions."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 48,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "20 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.0347167931002948,decay=4.746966178774611e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01062006045632861,decay=1.1876016717166215e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0006995070125407458,momentum=0.9844790514730665)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.07439975848075757,decay=1.7976337634506005e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.09030450672567254,decay=1.340890767690431e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01357387578284614,decay=2.3014993523846666e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.00010336714004241796,momentum=0.9711372680116186)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.00011116485234161093,momentum=0.9664752194346332)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0003071392825766392,momentum=0.9697893478568044)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.03540256307419597,decay=2.7490870549984347e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00026429087119428287,momentum=0.9702132562449013)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.04882317737663686,decay=8.006807036282709e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0005040379351745158,momentum=0.9863934944304705)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00037668508410008814,momentum=0.978821521218891)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.016426175651771575,decay=1.6439282808391488e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00046988338854109496,momentum=0.988290883937812)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0005402557401986037,momentum=0.9795021324622476)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.012596752275640428,decay=1.2801865417619381e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01055415187064375,decay=7.646989120220466e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00014021314734214438,momentum=0.9663397507032889)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 2, u\"optimizer='Adam(lr=0.0347167931002948,decay=4.746966178774611e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.01062006045632861,decay=1.1876016717166215e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.0006995070125407458,momentum=0.9844790514730665)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.07439975848075757,decay=1.7976337634506005e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (5, 2, u\"optimizer='Adam(lr=0.09030450672567254,decay=1.340890767690431e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01357387578284614,decay=2.3014993523846666e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (7, 2, u\"optimizer='SGD(lr=0.00010336714004241796,momentum=0.9711372680116186)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 2, u\"optimizer='SGD(lr=0.00011116485234161093,momentum=0.9664752194346332)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='SGD(lr=0.0003071392825766392,momentum=0.9697893478568044)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 1, u\"optimizer='Adam(lr=0.03540256307419597,decay=2.7490870549984347e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (11, 1, u\"optimizer='SGD(lr=0.00026429087119428287,momentum=0.9702132562449013)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 1, u\"optimizer='Adam(lr=0.04882317737663686,decay=8.006807036282709e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (13, 2, u\"optimizer='SGD(lr=0.0005040379351745158,momentum=0.9863934944304705)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (14, 1, u\"optimizer='SGD(lr=0.00037668508410008814,momentum=0.978821521218891)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (15, 1, u\"optimizer='Adam(lr=0.016426175651771575,decay=1.6439282808391488e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (16, 1, u\"optimizer='SGD(lr=0.00046988338854109496,momentum=0.988290883937812)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (17, 1, u\"optimizer='SGD(lr=0.0005402557401986037,momentum=0.9795021324622476)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (18, 2, u\"optimizer='Adam(lr=0.012596752275640428,decay=1.2801865417619381e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (19, 2, u\"optimizer='Adam(lr=0.01055415187064375,decay=7.646989120220466e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (20, 1, u\"optimizer='SGD(lr=0.00014021314734214438,momentum=0.9663397507032889)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64')]"
+      ]
+     },
+     "execution_count": 48,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [ \n",
+    "                                                 {'optimizer': ['SGD'], 'lr': [0.0001, 0.001, 'log'], 'momentum': [0.95, 0.99, 'log_near_one']}, \n",
+    "                                                 {'optimizer': ['Adam'], 'lr': [0.01, 0.1, 'log'], 'decay': [1e-6, 1e-4, 'log']}], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'random',            -- search_type\n",
+    "                                         20                   -- num_configs\n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"incremental_load\"></a>\n",
+    "# 3c.  Incremental loading for more complex combinations\n",
+    "\n",
+    "If it is easier to generate the model configurations incrementally rather than all at once, you can do that by not dropping the model selection table and associated summary table, in which case the new model configurations will be appended to the existing table.  Here we combine 2 of the previous examples in to a single output table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 49,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "16 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128')]"
+      ]
+     },
+     "execution_count": 49,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql \n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [ {'optimizer': ['Adam', 'SGD'], 'lr': [0.001, 0.01]} ], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'grid'               -- search_type \n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Now add to the existing table and note that mst_key continues where it left off:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 50,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "36 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00020031615564004395,momentum=0.9724038009180801)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.07529364006470769,decay=1.463102386655202e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.017537612203171578,decay=9.268965340542783e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.02436723652830891,decay=2.7036693659868636e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>21</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0009162225178908051,momentum=0.9636373679078051)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>22</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00020973934011486018,momentum=0.9810505351311615)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00011669504881554843,momentum=0.9563917160422619)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.019887949889421844,decay=1.3512689688436213e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>25</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.06844958467546351,decay=1.0949453143707621e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.0279719469411538,decay=3.116565475127251e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0005340915863494089,momentum=0.9846555995292319)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.00037236518835129966,momentum=0.9750593509631483)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.0001703149580002491,momentum=0.9516827304557754)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0003205092574897573,momentum=0.9745610627224451)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.000638802198775629,momentum=0.9896674744988915)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.0007145264848827797,momentum=0.9859303213231139)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.053811310244363884,decay=5.1052295876998844e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.0617217468673046,decay=2.0871014466512653e-06)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>35</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.013012045649000626,decay=5.7173240691732966e-05)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>36</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.000575290475361327,momentum=0.9883738353302843)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 1, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 2, u\"optimizer='SGD(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (17, 1, u\"optimizer='SGD(lr=0.00020031615564004395,momentum=0.9724038009180801)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (18, 2, u\"optimizer='Adam(lr=0.07529364006470769,decay=1.463102386655202e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (19, 1, u\"optimizer='Adam(lr=0.017537612203171578,decay=9.268965340542783e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (20, 1, u\"optimizer='Adam(lr=0.02436723652830891,decay=2.7036693659868636e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (21, 2, u\"optimizer='SGD(lr=0.0009162225178908051,momentum=0.9636373679078051)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (22, 1, u\"optimizer='SGD(lr=0.00020973934011486018,momentum=0.9810505351311615)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (23, 1, u\"optimizer='SGD(lr=0.00011669504881554843,momentum=0.9563917160422619)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (24, 1, u\"optimizer='Adam(lr=0.019887949889421844,decay=1.3512689688436213e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (25, 2, u\"optimizer='Adam(lr=0.06844958467546351,decay=1.0949453143707621e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (26, 1, u\"optimizer='Adam(lr=0.0279719469411538,decay=3.116565475127251e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (27, 1, u\"optimizer='SGD(lr=0.0005340915863494089,momentum=0.9846555995292319)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (28, 1, u\"optimizer='SGD(lr=0.00037236518835129966,momentum=0.9750593509631483)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (29, 1, u\"optimizer='SGD(lr=0.0001703149580002491,momentum=0.9516827304557754)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (30, 2, u\"optimizer='SGD(lr=0.0003205092574897573,momentum=0.9745610627224451)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (31, 2, u\"optimizer='SGD(lr=0.000638802198775629,momentum=0.9896674744988915)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (32, 2, u\"optimizer='SGD(lr=0.0007145264848827797,momentum=0.9859303213231139)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (33, 2, u\"optimizer='Adam(lr=0.053811310244363884,decay=5.1052295876998844e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (34, 1, u\"optimizer='Adam(lr=0.0617217468673046,decay=2.0871014466512653e-06)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64'),\n",
+       " (35, 1, u\"optimizer='Adam(lr=0.013012045649000626,decay=5.7173240691732966e-05)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=128'),\n",
+       " (36, 1, u\"optimizer='SGD(lr=0.000575290475361327,momentum=0.9883738353302843)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=10,batch_size=64')]"
+      ]
+     },
+     "execution_count": 50,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'], \n",
+    "                                             'optimizer_params_list': [ \n",
+    "                                                 {'optimizer': ['SGD'], 'lr': [0.0001, 0.001, 'log'], 'momentum': [0.95, 0.99, 'log_near_one']}, \n",
+    "                                                 {'optimizer': ['Adam'], 'lr': [0.01, 0.1, 'log'], 'decay': [1e-6, 1e-4, 'log']}], \n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid    \n",
+    "                                         $$ \n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10] \n",
+    "                                         } \n",
+    "                                         $$,                  -- fit_param_grid                                          \n",
+    "                                         'random',            -- search_type\n",
+    "                                         20                   -- num_configs\n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_model_selection_manual\"></a>\n",
+    "# 4.  Create model selection table manually\n",
+    "\n",
+    "If you want more control over the content of the model selection table, you could use grid or random search to generate a large number of combinations, then SELECT a subset of rows for training.\n",
+    "\n",
+    "Alternatively, you could manually create the model selection table and the associated summary table.  Both must be created since they are needed by the multiple model fit module.\n",
+    "\n",
+    "For example, let's say we don't want all combinations but only want batch_size=4 for model_id=1 and batch_size=8 for model_id=2:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 51,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "6 rows affected.\n",
+      "6 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_arch_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (3, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (4, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (5, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (6, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1')]"
+      ]
+     },
+     "execution_count": 51,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table_manual;\n",
+    "\n",
+    "CREATE TABLE mst_table_manual(\n",
+    "    mst_key serial,\n",
+    "    model_arch_id integer,\n",
+    "    compile_params varchar,\n",
+    "    fit_params varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO mst_table_manual(model_arch_id, compile_params, fit_params) VALUES\n",
+    "(1, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$, 'batch_size=4,epochs=1'),\n",
+    "(1, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']$$, 'batch_size=4,epochs=1'),\n",
+    "(1, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$, 'batch_size=4,epochs=1'),\n",
+    "(2, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$, 'batch_size=8,epochs=1'),\n",
+    "(2, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.01)',metrics=['accuracy']$$, 'batch_size=8,epochs=1'),\n",
+    "(2, $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$, 'batch_size=8,epochs=1');\n",
+    "\n",
+    "SELECT * FROM mst_table_manual ORDER BY mst_key; "
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Create the summary table which must be named with the model selection output table appended by \"_summary\":"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 52,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_arch_table</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>model_arch_library</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'model_arch_library',)]"
+      ]
+     },
+     "execution_count": 52,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table_manual_summary;\n",
+    "\n",
+    "CREATE TABLE mst_table_manual_summary (\n",
+    "    model_arch_table varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO mst_table_manual_summary(model_arch_table) VALUES\n",
+    "('model_arch_library');\n",
+    "\n",
+    "SELECT * FROM mst_table_manual_summary; "
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"custom\"></a>\n",
+    "# 5. Custom loss functions and custom metrics\n",
+    "\n",
+    "Define custom functions using the utility \"Define Custom Functions\". Psycopg is a PostgreSQL database adapter for the Python programming language. Note need to use the psycopg2.Binary() method to pass as bytes."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 53,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# import database connector psycopg2 and create connection cursor\n",
+    "import psycopg2 as p2\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
+    "cur = conn.cursor()\n",
+    "\n",
+    "# import Dill and define functions\n",
+    "import dill\n",
+    "\n",
+    "# custom loss\n",
+    "def squared_error(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.square(y_pred - y_true)\n",
+    "pb_squared_error=dill.dumps(squared_error)\n",
+    "\n",
+    "# custom metric\n",
+    "def rmse(y_true, y_pred):\n",
+    "    import tensorflow.keras.backend as K\n",
+    "    return K.sqrt(K.mean(K.square(y_pred - y_true), axis=-1))\n",
+    "pb_rmse=dill.dumps(rmse)\n",
+    "\n",
+    "# call load function\n",
+    "cur.execute(\"DROP TABLE IF EXISTS madlib.custom_function_table\")\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'squared_error', 'squared error')\", [p2.Binary(pb_squared_error)])\n",
+    "cur.execute(\"SELECT madlib.load_custom_function('custom_function_table',  %s,'rmse', 'root mean square error')\", [p2.Binary(pb_rmse)])\n",
+    "conn.commit()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 54,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "16 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=64</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'</td>\n",
+       "        <td>epochs=10,batch_size=128</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (3, 1, u\"optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (4, 1, u\"optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (7, 1, u\"optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (8, 1, u\"optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (11, 2, u\"optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (12, 2, u\"optimizer='SGD(lr=0.001)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (13, 2, u\"optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (14, 2, u\"optimizer='Adam(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128'),\n",
+       " (15, 2, u\"optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=64'),\n",
+       " (16, 2, u\"optimizer='SGD(lr=0.01)',metrics=['rmse'],loss='squared_error'\", u'epochs=10,batch_size=128')]"
+      ]
+     },
+     "execution_count": 54,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['squared_error'],\n",
+    "                                             'optimizer_params_list': [ {'optimizer': ['Adam', 'SGD'], 'lr': [0.001, 0.01]} ],\n",
+    "                                             'metrics': ['rmse']}\n",
+    "                                         $$,                  -- compile_param_grid\n",
+    "                                         $$\n",
+    "                                         { 'batch_size': [64, 128],\n",
+    "                                           'epochs': [10]\n",
+    "                                         }\n",
+    "                                         $$,                  -- fit_param_grid\n",
+    "                                         'grid',              -- search_type\n",
+    "                                         NULL,                -- num_configs\n",
+    "                                         NULL,                -- random_state\n",
+    "                                         'custom_function_table'  -- table with custom functions\n",
+    "                                         );\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_model_selection\"></a>\n",
+    "# 6.  Load model selection table [deprecated]\n",
+    "\n",
+    "#### This method is deprecated and replaced by generate_model_configs() method described above.\n",
+    "\n",
+    "Select the model(s) from the model architecture table that you want to run, along with the compile and fit parameters.  Unique combinations will be created for the set of model selection parameters."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 55,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=4,epochs=1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']</td>\n",
+       "        <td>batch_size=8,epochs=1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (2, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (3, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (4, 1, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (5, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (6, 1, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (7, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (8, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (9, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (10, 2, u\"loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']\", u'batch_size=8,epochs=1'),\n",
+       " (11, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=4,epochs=1'),\n",
+       " (12, 2, u\"loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']\", u'batch_size=8,epochs=1')]"
+      ]
+     },
+     "execution_count": 55,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.load_model_selection_table('model_arch_library', -- model architecture table\n",
+    "                                         'mst_table',          -- model selection table output\n",
+    "                                          ARRAY[1,2],              -- model ids from model architecture table\n",
+    "                                          ARRAY[                   -- compile params\n",
+    "                                              $$loss='categorical_crossentropy',optimizer='Adam(lr=0.1)',metrics=['accuracy']$$,\n",
+    "                                              $$loss='categorical_crossentropy', optimizer='Adam(lr=0.01)',metrics=['accuracy']$$,\n",
+    "                                              $$loss='categorical_crossentropy',optimizer='Adam(lr=0.001)',metrics=['accuracy']$$\n",
+    "                                          ],\n",
+    "                                          ARRAY[                    -- fit params\n",
+    "                                              $$batch_size=4,epochs=1$$,\n",
+    "                                              $$batch_size=8,epochs=1$$\n",
+    "                                          ]\n",
+    "                                         );\n",
+    "                                  \n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/MADlib-Keras-model-selection-CNN-cifar10-v1.ipynb b/community-artifacts/Deep-learning/Train-multiple-models/MADlib-Keras-model-selection-CNN-cifar10-v1.ipynb
similarity index 100%
rename from community-artifacts/Deep-learning/MADlib-Keras-model-selection-CNN-cifar10-v1.ipynb
rename to community-artifacts/Deep-learning/Train-multiple-models/MADlib-Keras-model-selection-CNN-cifar10-v1.ipynb
diff --git a/community-artifacts/Deep-learning/Train-multiple-models/MADlib-Keras-model-selection-MLP-v1.ipynb b/community-artifacts/Deep-learning/Train-multiple-models/MADlib-Keras-model-selection-MLP-v1.ipynb
new file mode 100644
index 0000000..4ae9eae
--- /dev/null
+++ b/community-artifacts/Deep-learning/Train-multiple-models/MADlib-Keras-model-selection-MLP-v1.ipynb
@@ -0,0 +1,6279 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Model Selection for Multilayer Perceptron Using Keras and MADlib\n",
+    "\n",
+    "E2E classification example using MADlib calling a Keras MLP for different hyperparameters and model architectures.\n",
+    "\n",
+    "Deep learning works best on very large datasets, but that is not convenient for a quick introduction to the syntax.  So in this workbook we use the well known iris data set from https://archive.ics.uci.edu/ml/datasets/iris to help get you started.  It is similar to the example in user docs http://madlib.apache.org/docs/latest/index.html\n",
+    "\n",
+    "For more realistic examples please refer to the deep learning notebooks at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#class\">Classification</a>\n",
+    "\n",
+    "* <a href=\"#create_input_data\">1. Create input data</a>\n",
+    "\n",
+    "* <a href=\"#pp\">2. Call preprocessor for deep learning</a>\n",
+    "\n",
+    "* <a href=\"#load\">3. Define and load model architecture</a>\n",
+    "\n",
+    "* <a href=\"#def_mst\">4. Define and load model selection tuples</a>\n",
+    "\n",
+    "* <a href=\"#train\">5. Train</a>\n",
+    "\n",
+    "* <a href=\"#eval\">6. Evaluate</a>\n",
+    "\n",
+    "* <a href=\"#pred\">7. Predict</a>\n",
+    "\n",
+    "<a href=\"#class2\">Classification with Other Parameters</a>\n",
+    "\n",
+    "* <a href=\"#val_dataset\">1. Validation dataset</a>\n",
+    "\n",
+    "* <a href=\"#pred_prob\">2. Predict probabilities</a>\n",
+    "\n",
+    "* <a href=\"#warm_start\">3. Warm start</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class\"></a>\n",
+    "# Classification\n",
+    "\n",
+    "<a id=\"create_input_data\"></a>\n",
+    "# 1.  Create input data\n",
+    "\n",
+    "Load iris data set."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "150 rows affected.\n",
+      "150 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>attributes</th>\n",
+       "        <th>class_text</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>[Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>21</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>22</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>25</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>35</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>36</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>37</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>39</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>40</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>42</td>\n",
+       "        <td>[Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>44</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>45</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>46</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>47</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>48</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>[Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>50</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>[Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>53</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>58</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>59</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>62</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>65</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>69</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>79</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>86</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>87</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>88</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>91</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>95</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>97</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>[Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>105</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>106</td>\n",
+       "        <td>[Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>[Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>109</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>110</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>118</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>124</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>129</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>130</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>[Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>[Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>133</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>139</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>140</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>141</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>142</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (2, [Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (3, [Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (4, [Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (5, [Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (6, [Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (7, [Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (8, [Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (9, [Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (10, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (11, [Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (12, [Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (13, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (14, [Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (15, [Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (16, [Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (17, [Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (18, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (19, [Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (20, [Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (21, [Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (22, [Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (23, [Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (24, [Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')], u'Iris-setosa'),\n",
+       " (25, [Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (26, [Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (27, [Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (28, [Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (29, [Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (30, [Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (31, [Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (32, [Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (33, [Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (34, [Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (35, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (36, [Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (37, [Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (38, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (39, [Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (40, [Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (41, [Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (42, [Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (43, [Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (44, [Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')], u'Iris-setosa'),\n",
+       " (45, [Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (46, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (47, [Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (48, [Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (49, [Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (50, [Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (51, [Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (52, [Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (53, [Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (54, [Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (55, [Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (56, [Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (57, [Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (58, [Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (59, [Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (60, [Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (61, [Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (62, [Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (63, [Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (64, [Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (65, [Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (66, [Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (67, [Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (68, [Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (69, [Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (70, [Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (71, [Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')], u'Iris-versicolor'),\n",
+       " (72, [Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (73, [Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (74, [Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (75, [Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (76, [Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (77, [Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (78, [Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')], u'Iris-versicolor'),\n",
+       " (79, [Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (80, [Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (81, [Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (82, [Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (83, [Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (84, [Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (85, [Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (86, [Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (87, [Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (88, [Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (89, [Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (90, [Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (91, [Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (92, [Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (93, [Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (94, [Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (95, [Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (96, [Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (97, [Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (98, [Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (99, [Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (100, [Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (101, [Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (102, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (103, [Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (104, [Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (105, [Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (106, [Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (107, [Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')], u'Iris-virginica'),\n",
+       " (108, [Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (109, [Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (110, [Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (111, [Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (112, [Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (113, [Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (114, [Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (115, [Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (116, [Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (117, [Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (118, [Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (119, [Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (120, [Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (121, [Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (122, [Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (123, [Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (124, [Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (125, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (126, [Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (127, [Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (128, [Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (129, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (130, [Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')], u'Iris-virginica'),\n",
+       " (131, [Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (132, [Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (133, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (134, [Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (135, [Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')], u'Iris-virginica'),\n",
+       " (136, [Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (137, [Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (138, [Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (139, [Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (140, [Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (141, [Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (142, [Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (143, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (144, [Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (145, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (146, [Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (147, [Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (148, [Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (149, [Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (150, [Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')], u'Iris-virginica')]"
+      ]
+     },
+     "execution_count": 4,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql \n",
+    "DROP TABLE IF EXISTS iris_data;\n",
+    "\n",
+    "CREATE TABLE iris_data(\n",
+    "    id serial,\n",
+    "    attributes numeric[],\n",
+    "    class_text varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO iris_data(id, attributes, class_text) VALUES\n",
+    "(1,ARRAY[5.1,3.5,1.4,0.2],'Iris-setosa'),\n",
+    "(2,ARRAY[4.9,3.0,1.4,0.2],'Iris-setosa'),\n",
+    "(3,ARRAY[4.7,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(4,ARRAY[4.6,3.1,1.5,0.2],'Iris-setosa'),\n",
+    "(5,ARRAY[5.0,3.6,1.4,0.2],'Iris-setosa'),\n",
+    "(6,ARRAY[5.4,3.9,1.7,0.4],'Iris-setosa'),\n",
+    "(7,ARRAY[4.6,3.4,1.4,0.3],'Iris-setosa'),\n",
+    "(8,ARRAY[5.0,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(9,ARRAY[4.4,2.9,1.4,0.2],'Iris-setosa'),\n",
+    "(10,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(11,ARRAY[5.4,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(12,ARRAY[4.8,3.4,1.6,0.2],'Iris-setosa'),\n",
+    "(13,ARRAY[4.8,3.0,1.4,0.1],'Iris-setosa'),\n",
+    "(14,ARRAY[4.3,3.0,1.1,0.1],'Iris-setosa'),\n",
+    "(15,ARRAY[5.8,4.0,1.2,0.2],'Iris-setosa'),\n",
+    "(16,ARRAY[5.7,4.4,1.5,0.4],'Iris-setosa'),\n",
+    "(17,ARRAY[5.4,3.9,1.3,0.4],'Iris-setosa'),\n",
+    "(18,ARRAY[5.1,3.5,1.4,0.3],'Iris-setosa'),\n",
+    "(19,ARRAY[5.7,3.8,1.7,0.3],'Iris-setosa'),\n",
+    "(20,ARRAY[5.1,3.8,1.5,0.3],'Iris-setosa'),\n",
+    "(21,ARRAY[5.4,3.4,1.7,0.2],'Iris-setosa'),\n",
+    "(22,ARRAY[5.1,3.7,1.5,0.4],'Iris-setosa'),\n",
+    "(23,ARRAY[4.6,3.6,1.0,0.2],'Iris-setosa'),\n",
+    "(24,ARRAY[5.1,3.3,1.7,0.5],'Iris-setosa'),\n",
+    "(25,ARRAY[4.8,3.4,1.9,0.2],'Iris-setosa'),\n",
+    "(26,ARRAY[5.0,3.0,1.6,0.2],'Iris-setosa'),\n",
+    "(27,ARRAY[5.0,3.4,1.6,0.4],'Iris-setosa'),\n",
+    "(28,ARRAY[5.2,3.5,1.5,0.2],'Iris-setosa'),\n",
+    "(29,ARRAY[5.2,3.4,1.4,0.2],'Iris-setosa'),\n",
+    "(30,ARRAY[4.7,3.2,1.6,0.2],'Iris-setosa'),\n",
+    "(31,ARRAY[4.8,3.1,1.6,0.2],'Iris-setosa'),\n",
+    "(32,ARRAY[5.4,3.4,1.5,0.4],'Iris-setosa'),\n",
+    "(33,ARRAY[5.2,4.1,1.5,0.1],'Iris-setosa'),\n",
+    "(34,ARRAY[5.5,4.2,1.4,0.2],'Iris-setosa'),\n",
+    "(35,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(36,ARRAY[5.0,3.2,1.2,0.2],'Iris-setosa'),\n",
+    "(37,ARRAY[5.5,3.5,1.3,0.2],'Iris-setosa'),\n",
+    "(38,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(39,ARRAY[4.4,3.0,1.3,0.2],'Iris-setosa'),\n",
+    "(40,ARRAY[5.1,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(41,ARRAY[5.0,3.5,1.3,0.3],'Iris-setosa'),\n",
+    "(42,ARRAY[4.5,2.3,1.3,0.3],'Iris-setosa'),\n",
+    "(43,ARRAY[4.4,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(44,ARRAY[5.0,3.5,1.6,0.6],'Iris-setosa'),\n",
+    "(45,ARRAY[5.1,3.8,1.9,0.4],'Iris-setosa'),\n",
+    "(46,ARRAY[4.8,3.0,1.4,0.3],'Iris-setosa'),\n",
+    "(47,ARRAY[5.1,3.8,1.6,0.2],'Iris-setosa'),\n",
+    "(48,ARRAY[4.6,3.2,1.4,0.2],'Iris-setosa'),\n",
+    "(49,ARRAY[5.3,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(50,ARRAY[5.0,3.3,1.4,0.2],'Iris-setosa'),\n",
+    "(51,ARRAY[7.0,3.2,4.7,1.4],'Iris-versicolor'),\n",
+    "(52,ARRAY[6.4,3.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(53,ARRAY[6.9,3.1,4.9,1.5],'Iris-versicolor'),\n",
+    "(54,ARRAY[5.5,2.3,4.0,1.3],'Iris-versicolor'),\n",
+    "(55,ARRAY[6.5,2.8,4.6,1.5],'Iris-versicolor'),\n",
+    "(56,ARRAY[5.7,2.8,4.5,1.3],'Iris-versicolor'),\n",
+    "(57,ARRAY[6.3,3.3,4.7,1.6],'Iris-versicolor'),\n",
+    "(58,ARRAY[4.9,2.4,3.3,1.0],'Iris-versicolor'),\n",
+    "(59,ARRAY[6.6,2.9,4.6,1.3],'Iris-versicolor'),\n",
+    "(60,ARRAY[5.2,2.7,3.9,1.4],'Iris-versicolor'),\n",
+    "(61,ARRAY[5.0,2.0,3.5,1.0],'Iris-versicolor'),\n",
+    "(62,ARRAY[5.9,3.0,4.2,1.5],'Iris-versicolor'),\n",
+    "(63,ARRAY[6.0,2.2,4.0,1.0],'Iris-versicolor'),\n",
+    "(64,ARRAY[6.1,2.9,4.7,1.4],'Iris-versicolor'),\n",
+    "(65,ARRAY[5.6,2.9,3.6,1.3],'Iris-versicolor'),\n",
+    "(66,ARRAY[6.7,3.1,4.4,1.4],'Iris-versicolor'),\n",
+    "(67,ARRAY[5.6,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(68,ARRAY[5.8,2.7,4.1,1.0],'Iris-versicolor'),\n",
+    "(69,ARRAY[6.2,2.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(70,ARRAY[5.6,2.5,3.9,1.1],'Iris-versicolor'),\n",
+    "(71,ARRAY[5.9,3.2,4.8,1.8],'Iris-versicolor'),\n",
+    "(72,ARRAY[6.1,2.8,4.0,1.3],'Iris-versicolor'),\n",
+    "(73,ARRAY[6.3,2.5,4.9,1.5],'Iris-versicolor'),\n",
+    "(74,ARRAY[6.1,2.8,4.7,1.2],'Iris-versicolor'),\n",
+    "(75,ARRAY[6.4,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(76,ARRAY[6.6,3.0,4.4,1.4],'Iris-versicolor'),\n",
+    "(77,ARRAY[6.8,2.8,4.8,1.4],'Iris-versicolor'),\n",
+    "(78,ARRAY[6.7,3.0,5.0,1.7],'Iris-versicolor'),\n",
+    "(79,ARRAY[6.0,2.9,4.5,1.5],'Iris-versicolor'),\n",
+    "(80,ARRAY[5.7,2.6,3.5,1.0],'Iris-versicolor'),\n",
+    "(81,ARRAY[5.5,2.4,3.8,1.1],'Iris-versicolor'),\n",
+    "(82,ARRAY[5.5,2.4,3.7,1.0],'Iris-versicolor'),\n",
+    "(83,ARRAY[5.8,2.7,3.9,1.2],'Iris-versicolor'),\n",
+    "(84,ARRAY[6.0,2.7,5.1,1.6],'Iris-versicolor'),\n",
+    "(85,ARRAY[5.4,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(86,ARRAY[6.0,3.4,4.5,1.6],'Iris-versicolor'),\n",
+    "(87,ARRAY[6.7,3.1,4.7,1.5],'Iris-versicolor'),\n",
+    "(88,ARRAY[6.3,2.3,4.4,1.3],'Iris-versicolor'),\n",
+    "(89,ARRAY[5.6,3.0,4.1,1.3],'Iris-versicolor'),\n",
+    "(90,ARRAY[5.5,2.5,4.0,1.3],'Iris-versicolor'),\n",
+    "(91,ARRAY[5.5,2.6,4.4,1.2],'Iris-versicolor'),\n",
+    "(92,ARRAY[6.1,3.0,4.6,1.4],'Iris-versicolor'),\n",
+    "(93,ARRAY[5.8,2.6,4.0,1.2],'Iris-versicolor'),\n",
+    "(94,ARRAY[5.0,2.3,3.3,1.0],'Iris-versicolor'),\n",
+    "(95,ARRAY[5.6,2.7,4.2,1.3],'Iris-versicolor'),\n",
+    "(96,ARRAY[5.7,3.0,4.2,1.2],'Iris-versicolor'),\n",
+    "(97,ARRAY[5.7,2.9,4.2,1.3],'Iris-versicolor'),\n",
+    "(98,ARRAY[6.2,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(99,ARRAY[5.1,2.5,3.0,1.1],'Iris-versicolor'),\n",
+    "(100,ARRAY[5.7,2.8,4.1,1.3],'Iris-versicolor'),\n",
+    "(101,ARRAY[6.3,3.3,6.0,2.5],'Iris-virginica'),\n",
+    "(102,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(103,ARRAY[7.1,3.0,5.9,2.1],'Iris-virginica'),\n",
+    "(104,ARRAY[6.3,2.9,5.6,1.8],'Iris-virginica'),\n",
+    "(105,ARRAY[6.5,3.0,5.8,2.2],'Iris-virginica'),\n",
+    "(106,ARRAY[7.6,3.0,6.6,2.1],'Iris-virginica'),\n",
+    "(107,ARRAY[4.9,2.5,4.5,1.7],'Iris-virginica'),\n",
+    "(108,ARRAY[7.3,2.9,6.3,1.8],'Iris-virginica'),\n",
+    "(109,ARRAY[6.7,2.5,5.8,1.8],'Iris-virginica'),\n",
+    "(110,ARRAY[7.2,3.6,6.1,2.5],'Iris-virginica'),\n",
+    "(111,ARRAY[6.5,3.2,5.1,2.0],'Iris-virginica'),\n",
+    "(112,ARRAY[6.4,2.7,5.3,1.9],'Iris-virginica'),\n",
+    "(113,ARRAY[6.8,3.0,5.5,2.1],'Iris-virginica'),\n",
+    "(114,ARRAY[5.7,2.5,5.0,2.0],'Iris-virginica'),\n",
+    "(115,ARRAY[5.8,2.8,5.1,2.4],'Iris-virginica'),\n",
+    "(116,ARRAY[6.4,3.2,5.3,2.3],'Iris-virginica'),\n",
+    "(117,ARRAY[6.5,3.0,5.5,1.8],'Iris-virginica'),\n",
+    "(118,ARRAY[7.7,3.8,6.7,2.2],'Iris-virginica'),\n",
+    "(119,ARRAY[7.7,2.6,6.9,2.3],'Iris-virginica'),\n",
+    "(120,ARRAY[6.0,2.2,5.0,1.5],'Iris-virginica'),\n",
+    "(121,ARRAY[6.9,3.2,5.7,2.3],'Iris-virginica'),\n",
+    "(122,ARRAY[5.6,2.8,4.9,2.0],'Iris-virginica'),\n",
+    "(123,ARRAY[7.7,2.8,6.7,2.0],'Iris-virginica'),\n",
+    "(124,ARRAY[6.3,2.7,4.9,1.8],'Iris-virginica'),\n",
+    "(125,ARRAY[6.7,3.3,5.7,2.1],'Iris-virginica'),\n",
+    "(126,ARRAY[7.2,3.2,6.0,1.8],'Iris-virginica'),\n",
+    "(127,ARRAY[6.2,2.8,4.8,1.8],'Iris-virginica'),\n",
+    "(128,ARRAY[6.1,3.0,4.9,1.8],'Iris-virginica'),\n",
+    "(129,ARRAY[6.4,2.8,5.6,2.1],'Iris-virginica'),\n",
+    "(130,ARRAY[7.2,3.0,5.8,1.6],'Iris-virginica'),\n",
+    "(131,ARRAY[7.4,2.8,6.1,1.9],'Iris-virginica'),\n",
+    "(132,ARRAY[7.9,3.8,6.4,2.0],'Iris-virginica'),\n",
+    "(133,ARRAY[6.4,2.8,5.6,2.2],'Iris-virginica'),\n",
+    "(134,ARRAY[6.3,2.8,5.1,1.5],'Iris-virginica'),\n",
+    "(135,ARRAY[6.1,2.6,5.6,1.4],'Iris-virginica'),\n",
+    "(136,ARRAY[7.7,3.0,6.1,2.3],'Iris-virginica'),\n",
+    "(137,ARRAY[6.3,3.4,5.6,2.4],'Iris-virginica'),\n",
+    "(138,ARRAY[6.4,3.1,5.5,1.8],'Iris-virginica'),\n",
+    "(139,ARRAY[6.0,3.0,4.8,1.8],'Iris-virginica'),\n",
+    "(140,ARRAY[6.9,3.1,5.4,2.1],'Iris-virginica'),\n",
+    "(141,ARRAY[6.7,3.1,5.6,2.4],'Iris-virginica'),\n",
+    "(142,ARRAY[6.9,3.1,5.1,2.3],'Iris-virginica'),\n",
+    "(143,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(144,ARRAY[6.8,3.2,5.9,2.3],'Iris-virginica'),\n",
+    "(145,ARRAY[6.7,3.3,5.7,2.5],'Iris-virginica'),\n",
+    "(146,ARRAY[6.7,3.0,5.2,2.3],'Iris-virginica'),\n",
+    "(147,ARRAY[6.3,2.5,5.0,1.9],'Iris-virginica'),\n",
+    "(148,ARRAY[6.5,3.0,5.2,2.0],'Iris-virginica'),\n",
+    "(149,ARRAY[6.2,3.4,5.4,2.3],'Iris-virginica'),\n",
+    "(150,ARRAY[5.9,3.0,5.1,1.8],'Iris-virginica');\n",
+    "\n",
+    "SELECT * FROM iris_data ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Create a test/validation dataset from the training data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(120L,)]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train, iris_test;\n",
+    "\n",
+    "-- Set seed so results are reproducible\n",
+    "SELECT setseed(0);\n",
+    "\n",
+    "SELECT madlib.train_test_split('iris_data',     -- Source table\n",
+    "                               'iris',          -- Output table root name\n",
+    "                                0.8,            -- Train proportion\n",
+    "                                NULL,           -- Test proportion (0.2)\n",
+    "                                NULL,           -- Strata definition\n",
+    "                                NULL,           -- Output all columns\n",
+    "                                NULL,           -- Sample without replacement\n",
+    "                                TRUE            -- Separate output tables\n",
+    "                              );\n",
+    "\n",
+    "SELECT COUNT(*) FROM iris_train;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pp\"></a>\n",
+    "# 2. Call preprocessor for deep learning\n",
+    "Training dataset (uses training preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>attributes_shape</th>\n",
+       "        <th>class_text_shape</th>\n",
+       "        <th>buffer_id</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[60, 4]</td>\n",
+       "        <td>[60, 3]</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[60, 4]</td>\n",
+       "        <td>[60, 3]</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[([60, 4], [60, 3], 0), ([60, 4], [60, 3], 1)]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train_packed, iris_train_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('iris_train',         -- Source table\n",
+    "                                       'iris_train_packed',  -- Output table\n",
+    "                                       'class_text',        -- Dependent variable\n",
+    "                                       'attributes'         -- Independent variable\n",
+    "                                        ); \n",
+    "\n",
+    "SELECT attributes_shape, class_text_shape, buffer_id FROM iris_train_packed ORDER BY buffer_id;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train</td>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>60</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train', u'iris_train_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 60, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_train_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Validation dataset (uses validation preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>attributes_shape</th>\n",
+       "        <th>class_text_shape</th>\n",
+       "        <th>buffer_id</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 4]</td>\n",
+       "        <td>[15, 3]</td>\n",
+       "        <td>0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>[15, 4]</td>\n",
+       "        <td>[15, 3]</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[([15, 4], [15, 3], 0), ([15, 4], [15, 3], 1)]"
+      ]
+     },
+     "execution_count": 8,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_test_packed, iris_test_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('iris_test',          -- Source table\n",
+    "                                         'iris_test_packed',   -- Output table\n",
+    "                                         'class_text',         -- Dependent variable\n",
+    "                                         'attributes',         -- Independent variable\n",
+    "                                         'iris_train_packed'   -- From training preprocessor step\n",
+    "                                          ); \n",
+    "\n",
+    "SELECT attributes_shape, class_text_shape, buffer_id FROM iris_test_packed ORDER BY buffer_id;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_test</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>15</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_test', u'iris_test_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 15, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 9,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_test_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load\"></a>\n",
+    "# 3. Define and load model architecture\n",
+    "Import Keras libraries"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 1 hidden layer:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "WARNING:tensorflow:From /Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/tensorflow/python/ops/init_ops.py:1251: calling __init__ (from tensorflow.python.ops.init_ops) with dtype is deprecated and will be removed in a future version.\n",
+      "Instructions for updating:\n",
+      "Call initializer instance with the dtype argument instead of passing it to the constructor\n",
+      "Model: \"sequential\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense (Dense)                (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_1 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_2 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 193\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model1 = Sequential()\n",
+    "model1.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model1.add(Dense(10, activation='relu'))\n",
+    "model1.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model1.summary()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 12,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 12,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model1.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture with 2 hidden layers:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 13,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_1\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_3 (Dense)              (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_4 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_5 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_6 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 303\n",
+      "Trainable params: 303\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model2 = Sequential()\n",
+    "model2.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(10, activation='relu'))\n",
+    "model2.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model2.summary()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 14,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_1\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 14,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model2.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 15,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>model_weights</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>__internal_madlib_id__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>MLP with 1 hidden layer</td>\n",
+       "        <td>__madlib_temp_4017958_1614991901_4240024__</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_5', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_6', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_7', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>None</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>MLP with 2 hidden layers</td>\n",
+       "        <td>__madlib_temp_28416680_1614991901_72274844__</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Sophie', u'MLP with 1 hidden layer', u'__madlib_temp_4017958_1614991901_4240024__'),\n",
+       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1835 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, None, u'Maria', u'MLP with 2 hidden layers', u'__madlib_temp_28416680_1614991901_72274844__')]"
+      ]
+     },
+     "execution_count": 15,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS model_arch_library;\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Sophie',              -- Name\n",
+    "                               'MLP with 1 hidden layer'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_6\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_7\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Maria',               -- Name\n",
+    "                               'MLP with 2 hidden layers'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT * FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"def_mst\"></a>\n",
+    "# 4.  Define and load model selection tuples"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Generate model configurations using grid search. The output table for grid search contains the unique combinations of model architectures, compile and fit parameters."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 16,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8'),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4'),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8')]"
+      ]
+     },
+     "execution_count": 16,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mst_table, mst_table_summary;\n",
+    "\n",
+    "SELECT madlib.generate_model_configs(\n",
+    "                                        'model_arch_library', -- model architecture table\n",
+    "                                        'mst_table',          -- model selection table output\n",
+    "                                         ARRAY[1,2],          -- model ids from model architecture table\n",
+    "                                         $$\n",
+    "                                            {'loss': ['categorical_crossentropy'],\n",
+    "                                             'optimizer_params_list': [ {'optimizer': ['Adam'], 'lr': [0.001, 0.01, 0.1]} ],\n",
+    "                                             'metrics': ['accuracy']}\n",
+    "                                         $$,                  -- compile_param_grid\n",
+    "                                         $$\n",
+    "                                         { 'batch_size': [4, 8],\n",
+    "                                           'epochs': [1]\n",
+    "                                         }\n",
+    "                                         $$,                  -- fit_param_grid\n",
+    "                                         'grid'               -- search_type\n",
+    "                                         );\n",
+    "\n",
+    "SELECT * FROM mst_table ORDER BY mst_key;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "This is the name of the model architecture table that corresponds to the model selection table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 17,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>object_table</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'model_arch_library', None)]"
+      ]
+     },
+     "execution_count": 17,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM mst_table_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train\"></a>\n",
+    "# 5.  Train\n",
+    "Train multiple models:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 18,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit_multiple_model</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 18,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_multi_model, iris_multi_model_summary, iris_multi_model_info;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
+    "                                              'iris_multi_model',     -- model_output_table\n",
+    "                                              'mst_table',            -- model_selection_table\n",
+    "                                              10,                     -- num_iterations\n",
+    "                                              FALSE                   -- use gpus\n",
+    "                                             );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 19,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>warm_start</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>iris_multi_model</td>\n",
+       "        <td>iris_multi_model_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>mst_table</td>\n",
+       "        <td>None</td>\n",
+       "        <td>10</td>\n",
+       "        <td>10</td>\n",
+       "        <td>False</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2021-03-06 00:51:48.452654</td>\n",
+       "        <td>2021-03-06 00:53:20.221035</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[10]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', None, u'iris_multi_model', u'iris_multi_model_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'mst_table', None, 10, 10, False, None, None, datetime.datetime(2021, 3, 6, 0, 51, 48, 452654), datetime.datetime(2021, 3, 6, 0, 53, 20, 221035), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0, [10])]"
+      ]
+     },
+     "execution_count": 19,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View results for each model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 20,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[90.2427790164948]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.983333349228</td>\n",
+       "        <td>0.201789721847</td>\n",
+       "        <td>[0.983333349227905]</td>\n",
+       "        <td>[0.201789721846581]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[88.9964590072632]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.134730249643</td>\n",
+       "        <td>[0.933333337306976]</td>\n",
+       "        <td>[0.134730249643326]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[88.7690601348877]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.402144879103</td>\n",
+       "        <td>[0.933333337306976]</td>\n",
+       "        <td>[0.402144879102707]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[90.9196391105652]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.416792035103</td>\n",
+       "        <td>[0.933333337306976]</td>\n",
+       "        <td>[0.416792035102844]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[89.534707069397]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.908333361149</td>\n",
+       "        <td>0.19042557478</td>\n",
+       "        <td>[0.908333361148834]</td>\n",
+       "        <td>[0.19042557477951]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[89.273796081543]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.181902274489</td>\n",
+       "        <td>[0.899999976158142]</td>\n",
+       "        <td>[0.181902274489403]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[90.4800100326538]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.824999988079</td>\n",
+       "        <td>0.303107827902</td>\n",
+       "        <td>[0.824999988079071]</td>\n",
+       "        <td>[0.30310782790184]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[89.7936120033264]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.808333337307</td>\n",
+       "        <td>0.300039559603</td>\n",
+       "        <td>[0.808333337306976]</td>\n",
+       "        <td>[0.300039559602737]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[90.0158791542053]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.658333361149</td>\n",
+       "        <td>0.869387447834</td>\n",
+       "        <td>[0.658333361148834]</td>\n",
+       "        <td>[0.869387447834015]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[91.1929490566254]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.558333337307</td>\n",
+       "        <td>0.84612262249</td>\n",
+       "        <td>[0.558333337306976]</td>\n",
+       "        <td>[0.846122622489929]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[91.7660541534424]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.341666668653</td>\n",
+       "        <td>1.10138702393</td>\n",
+       "        <td>[0.341666668653488]</td>\n",
+       "        <td>[1.10138702392578]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[91.5026919841766]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.341666668653</td>\n",
+       "        <td>1.10163521767</td>\n",
+       "        <td>[0.341666668653488]</td>\n",
+       "        <td>[1.10163521766663]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [90.2427790164948], [u'accuracy'], u'categorical_crossentropy', 0.983333349227905, 0.201789721846581, [0.983333349227905], [0.201789721846581], None, None, None, None),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [88.9964590072632], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.134730249643326, [0.933333337306976], [0.134730249643326], None, None, None, None),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [88.7690601348877], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.402144879102707, [0.933333337306976], [0.402144879102707], None, None, None, None),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [90.9196391105652], [u'accuracy'], u'categorical_crossentropy', 0.933333337306976, 0.416792035102844, [0.933333337306976], [0.416792035102844], None, None, None, None),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [89.534707069397], [u'accuracy'], u'categorical_crossentropy', 0.908333361148834, 0.19042557477951, [0.908333361148834], [0.19042557477951], None, None, None, None),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [89.273796081543], [u'accuracy'], u'categorical_crossentropy', 0.899999976158142, 0.181902274489403, [0.899999976158142], [0.181902274489403], None, None, None, None),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [90.4800100326538], [u'accuracy'], u'categorical_crossentropy', 0.824999988079071, 0.30310782790184, [0.824999988079071], [0.30310782790184], None, None, None, None),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [89.7936120033264], [u'accuracy'], u'categorical_crossentropy', 0.808333337306976, 0.300039559602737, [0.808333337306976], [0.300039559602737], None, None, None, None),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [90.0158791542053], [u'accuracy'], u'categorical_crossentropy', 0.658333361148834, 0.869387447834015, [0.658333361148834], [0.869387447834015], None, None, None, None),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [91.1929490566254], [u'accuracy'], u'categorical_crossentropy', 0.558333337306976, 0.846122622489929, [0.558333337306976], [0.846122622489929], None, None, None, None),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [91.7660541534424], [u'accuracy'], u'categorical_crossentropy', 0.341666668653488, 1.10138702392578, [0.341666668653488], [1.10138702392578], None, None, None, None),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [91.5026919841766], [u'accuracy'], u'categorical_crossentropy', 0.341666668653488, 1.10163521766663, [0.341666668653488], [1.10163521766663], None, None, None, None)]"
+      ]
+     },
+     "execution_count": 20,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_info ORDER BY training_metrics_final DESC, training_loss_final;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"eval\"></a>\n",
+    "# 6. Evaluate\n",
+    "\n",
+    "Now run evaluate using model we built above:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 21,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>loss</th>\n",
+       "        <th>metric</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>0.194916069508</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(0.194916069507599, 0.899999976158142, [u'accuracy'], u'categorical_crossentropy')]"
+      ]
+     },
+     "execution_count": 21,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_validate;\n",
+    "SELECT madlib.madlib_keras_evaluate('iris_multi_model',  -- model\n",
+    "                                    'iris_test_packed',  -- test table\n",
+    "                                    'iris_validate',     -- output table\n",
+    "                                     NULL,               -- use gpus\n",
+    "                                     9                   -- mst_key to use\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_validate;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred\"></a>\n",
+    "# 7. Predict\n",
+    "\n",
+    "Now predict using model we built.  We will use the validation data set for prediction as well, which is not usual but serves to show the syntax. The prediction is in the estimated_class_text column:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 22,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "30 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999999</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999999</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999999</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999999</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.0</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.99069124</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9864196</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9983382</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9991603</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9974559</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.60661113</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9940832</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9987955</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7598468</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8414144</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.715776</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9163472</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5081183</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.85080105</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.9842195</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6804195</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.81555897</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.92707217</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7158722</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.55272627</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7662018</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (10, u'class_text', u'Iris-setosa', 0.9999999),\n",
+       " (12, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (14, u'class_text', u'Iris-setosa', 0.9999999),\n",
+       " (18, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (20, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (30, u'class_text', u'Iris-setosa', 0.9999999),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.9999999),\n",
+       " (49, u'class_text', u'Iris-setosa', 1.0),\n",
+       " (55, u'class_text', u'Iris-versicolor', 0.99069124),\n",
+       " (64, u'class_text', u'Iris-versicolor', 0.9864196),\n",
+       " (70, u'class_text', u'Iris-versicolor', 0.9983382),\n",
+       " (76, u'class_text', u'Iris-versicolor', 0.9991603),\n",
+       " (82, u'class_text', u'Iris-versicolor', 0.9974559),\n",
+       " (84, u'class_text', u'Iris-versicolor', 0.60661113),\n",
+       " (92, u'class_text', u'Iris-versicolor', 0.9940832),\n",
+       " (98, u'class_text', u'Iris-versicolor', 0.9987955),\n",
+       " (99, u'class_text', u'Iris-versicolor', 0.7598468),\n",
+       " (102, u'class_text', u'Iris-virginica', 0.8414144),\n",
+       " (107, u'class_text', u'Iris-virginica', 0.715776),\n",
+       " (114, u'class_text', u'Iris-virginica', 0.9163472),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.5081183),\n",
+       " (121, u'class_text', u'Iris-virginica', 0.85080105),\n",
+       " (123, u'class_text', u'Iris-virginica', 0.9842195),\n",
+       " (125, u'class_text', u'Iris-virginica', 0.6804195),\n",
+       " (127, u'class_text', u'Iris-versicolor', 0.81555897),\n",
+       " (145, u'class_text', u'Iris-virginica', 0.92707217),\n",
+       " (147, u'class_text', u'Iris-virginica', 0.7158722),\n",
+       " (148, u'class_text', u'Iris-versicolor', 0.55272627),\n",
+       " (149, u'class_text', u'Iris-virginica', 0.7662018)]"
+      ]
+     },
+     "execution_count": 22,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_multi_model', -- model\n",
+    "                                   'iris_test',        -- test_table\n",
+    "                                   'id',               -- id column\n",
+    "                                   'attributes',       -- independent var\n",
+    "                                   'iris_predict',     -- output table\n",
+    "                                    'response',        -- prediction type\n",
+    "                                    FALSE,             -- use gpus\n",
+    "                                    9                  -- mst_key to use\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 23,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3L,)]"
+      ]
+     },
+     "execution_count": 23,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict JOIN iris_test USING (id) \n",
+    "WHERE iris_predict.class_value != iris_test.class_text;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 24,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90.00</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('90.00'),)]"
+      ]
+     },
+     "execution_count": 24,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict.class_value as estimated\n",
+    "     from iris_predict inner join iris_test\n",
+    "     on iris_test.id=iris_predict.id) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class2\"></a>\n",
+    "# Classification with Other Parameters\n",
+    "\n",
+    "<a id=\"val_dataset\"></a>\n",
+    "# 1.  Validation dataset\n",
+    "\n",
+    "Now use a validation dataset and compute metrics every 2nd iteration using the 'metrics_compute_frequency' parameter.  This can help reduce run time if you do not need metrics computed at every iteration."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 25,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit_multiple_model</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 25,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_multi_model, iris_multi_model_summary, iris_multi_model_info;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
+    "                                              'iris_multi_model',     -- model_output_table\n",
+    "                                              'mst_table',            -- model_selection_table\n",
+    "                                               10,                     -- num_iterations\n",
+    "                                               FALSE,                 -- use gpus\n",
+    "                                              'iris_test_packed',     -- validation dataset\n",
+    "                                               3,                     -- metrics compute frequency\n",
+    "                                               FALSE,                 -- warm start\n",
+    "                                              'Sophie L.',            -- name\n",
+    "                                              'Model selection for iris dataset'  -- description\n",
+    "                                             );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 26,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>warm_start</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>iris_multi_model</td>\n",
+       "        <td>iris_multi_model_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>mst_table</td>\n",
+       "        <td>None</td>\n",
+       "        <td>10</td>\n",
+       "        <td>3</td>\n",
+       "        <td>False</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Model selection for iris dataset</td>\n",
+       "        <td>2021-03-06 00:53:31.218406</td>\n",
+       "        <td>2021-03-06 00:55:25.621208</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3, 6, 9, 10]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_test_packed', u'iris_multi_model', u'iris_multi_model_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'mst_table', None, 10, 3, False, u'Sophie L.', u'Model selection for iris dataset', datetime.datetime(2021, 3, 6, 0, 53, 31, 218406), datetime.datetime(2021, 3, 6, 0, 55, 25, 621208), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0, [3, 6, 9, 10])]"
+      ]
+     },
+     "execution_count": 26,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View performance of each model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 27,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[31.5490398406982, 64.223620891571, 97.8899219036102, 113.156138896942]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.991666674614</td>\n",
+       "        <td>0.177691921592</td>\n",
+       "        <td>[0.824999988079071, 0.975000023841858, 0.933333337306976, 0.991666674613953]</td>\n",
+       "        <td>[0.508709609508514, 0.290052831172943, 0.217903628945351, 0.177691921591759]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.20564225316</td>\n",
+       "        <td>[0.833333313465118, 0.966666638851166, 0.933333337306976, 0.966666638851166]</td>\n",
+       "        <td>[0.516587793827057, 0.316147029399872, 0.228292018175125, 0.205642253160477]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[30.4000718593597, 62.9767029285431, 96.690801858902, 112.145288944244]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.908333361149</td>\n",
+       "        <td>0.203085869551</td>\n",
+       "        <td>[0.933333337306976, 0.808333337306976, 0.958333313465118, 0.908333361148834]</td>\n",
+       "        <td>[0.372362315654755, 0.304766088724136, 0.11820487678051, 0.203085869550705]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.190864190459</td>\n",
+       "        <td>[0.966666638851166, 0.833333313465118, 0.966666638851166, 0.933333337306976]</td>\n",
+       "        <td>[0.347199022769928, 0.290798246860504, 0.110275268554688, 0.190864190459251]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[30.875373840332, 63.4593389034271, 97.1958589553833, 112.702126979828]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.883333325386</td>\n",
+       "        <td>0.692279815674</td>\n",
+       "        <td>[0.533333361148834, 0.616666674613953, 0.875, 0.883333325386047]</td>\n",
+       "        <td>[1.08197057247162, 0.851473987102509, 0.729827761650085, 0.692279815673828]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.674779772758</td>\n",
+       "        <td>[0.600000023841858, 0.666666686534882, 0.899999976158142, 0.899999976158142]</td>\n",
+       "        <td>[1.05298256874084, 0.817528009414673, 0.710631787776947, 0.674779772758484]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[29.8903229236603, 62.4677069187164, 96.1764039993286, 111.539803981781]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.925000011921</td>\n",
+       "        <td>0.176520362496</td>\n",
+       "        <td>[0.833333313465118, 0.925000011920929, 0.774999976158142, 0.925000011920929]</td>\n",
+       "        <td>[0.324734181165695, 0.182637020945549, 0.468331128358841, 0.176520362496376]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.2585529387</td>\n",
+       "        <td>[0.866666674613953, 0.866666674613953, 0.866666674613953, 0.899999976158142]</td>\n",
+       "        <td>[0.341204434633255, 0.261798053979874, 0.45467621088028, 0.258552938699722]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[31.7836039066315, 64.4592599868774, 98.1328208446503, 113.377946853638]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.891666650772</td>\n",
+       "        <td>0.797108471394</td>\n",
+       "        <td>[0.341666668653488, 0.491666674613953, 0.916666686534882, 0.891666650772095]</td>\n",
+       "        <td>[1.09786474704742, 0.967048287391663, 0.838281869888306, 0.797108471393585]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.800795376301</td>\n",
+       "        <td>[0.300000011920929, 0.433333337306976, 0.933333337306976, 0.899999976158142]</td>\n",
+       "        <td>[1.07609903812408, 0.962578594684601, 0.834975183010101, 0.800795376300812]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[30.1456639766693, 62.722916841507, 96.4333670139313, 111.892151832581]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.816666662693</td>\n",
+       "        <td>0.734887838364</td>\n",
+       "        <td>[0.850000023841858, 0.958333313465118, 0.966666638851166, 0.816666662693024]</td>\n",
+       "        <td>[0.335647404193878, 0.0894104242324829, 0.0672163665294647, 0.734887838363647]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.665323019028</td>\n",
+       "        <td>[0.866666674613953, 0.966666638851166, 0.966666638851166, 0.866666674613953]</td>\n",
+       "        <td>[0.320426166057587, 0.154994085431099, 0.204012081027031, 0.66532301902771]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[32.0452349185944, 64.7241299152374, 98.4015560150146, 113.899842977524]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.791666686535</td>\n",
+       "        <td>0.772948563099</td>\n",
+       "        <td>[0.316666662693024, 0.349999994039536, 0.725000023841858, 0.791666686534882]</td>\n",
+       "        <td>[1.01266825199127, 0.905348658561707, 0.807280421257019, 0.772948563098907]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.740880072117</td>\n",
+       "        <td>[0.400000005960464, 0.466666668653488, 0.800000011920929, 0.866666674613953]</td>\n",
+       "        <td>[0.964996755123138, 0.868514597415924, 0.771895349025726, 0.740880072116852]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[30.6602540016174, 63.2428169250488, 96.9531948566437, 112.484740972519]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.691666662693</td>\n",
+       "        <td>0.501820206642</td>\n",
+       "        <td>[0.658333361148834, 0.658333361148834, 0.658333361148834, 0.691666662693024]</td>\n",
+       "        <td>[0.654709756374359, 0.581917643547058, 1.33844769001007, 0.501820206642151]</td>\n",
+       "        <td>0.766666650772</td>\n",
+       "        <td>0.457984447479</td>\n",
+       "        <td>[0.699999988079071, 0.699999988079071, 0.699999988079071, 0.766666650772095]</td>\n",
+       "        <td>[0.592061340808868, 0.525563180446625, 1.17788350582123, 0.457984447479248]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[31.0910878181458, 63.7646949291229, 97.4185988903046, 112.939773797989]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.666666686535</td>\n",
+       "        <td>0.50052946806</td>\n",
+       "        <td>[0.433333337306976, 0.641666650772095, 0.649999976158142, 0.666666686534882]</td>\n",
+       "        <td>[0.850135624408722, 0.611121952533722, 0.509139358997345, 0.50052946805954]</td>\n",
+       "        <td>0.733333349228</td>\n",
+       "        <td>0.459399551153</td>\n",
+       "        <td>[0.466666668653488, 0.699999988079071, 0.699999988079071, 0.733333349227905]</td>\n",
+       "        <td>[0.802468597888947, 0.571285247802734, 0.492577910423279, 0.459399551153183]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[29.6670269966125, 62.2440509796143, 95.9554150104523, 111.311369895935]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.733333349228</td>\n",
+       "        <td>0.821944594383</td>\n",
+       "        <td>[0.341666668653488, 0.341666668653488, 0.658333361148834, 0.733333349227905]</td>\n",
+       "        <td>[1.06431686878204, 0.996406197547913, 0.869706034660339, 0.82194459438324]</td>\n",
+       "        <td>0.699999988079</td>\n",
+       "        <td>0.852133929729</td>\n",
+       "        <td>[0.300000011920929, 0.300000011920929, 0.699999988079071, 0.699999988079071]</td>\n",
+       "        <td>[1.09268116950989, 1.01670277118683, 0.891825795173645, 0.852133929729462]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[32.5322558879852, 65.2217888832092, 98.9477097988129, 114.400418996811]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.683333337307</td>\n",
+       "        <td>0.455871999264</td>\n",
+       "        <td>[0.725000023841858, 0.683333337306976, 0.683333337306976, 0.683333337306976]</td>\n",
+       "        <td>[0.383917421102524, 0.457853585481644, 0.455943495035172, 0.455871999263763]</td>\n",
+       "        <td>0.600000023842</td>\n",
+       "        <td>0.488439053297</td>\n",
+       "        <td>[0.800000011920929, 0.600000023841858, 0.600000023841858, 0.600000023841858]</td>\n",
+       "        <td>[0.388951361179352, 0.50080794095993, 0.487448841333389, 0.488439053297043]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[32.2720308303833, 64.9502189159393, 98.6836059093475, 114.134181976318]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.675000011921</td>\n",
+       "        <td>0.452209770679</td>\n",
+       "        <td>[0.683333337306976, 0.675000011920929, 0.683333337306976, 0.675000011920929]</td>\n",
+       "        <td>[0.492754250764847, 0.469423890113831, 0.571796059608459, 0.452209770679474]</td>\n",
+       "        <td>0.600000023842</td>\n",
+       "        <td>0.464268505573</td>\n",
+       "        <td>[0.733333349227905, 0.766666650772095, 0.600000023841858, 0.600000023841858]</td>\n",
+       "        <td>[0.438488334417343, 0.390993624925613, 0.690678656101227, 0.464268505573273]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [31.5490398406982, 64.223620891571, 97.8899219036102, 113.156138896942], [u'accuracy'], u'categorical_crossentropy', 0.991666674613953, 0.177691921591759, [0.824999988079071, 0.975000023841858, 0.933333337306976, 0.991666674613953], [0.508709609508514, 0.290052831172943, 0.217903628945351, 0.177691921591759], 0.966666638851166, 0.205642253160477, [0.833333313465118, 0.966666638851166, 0.933333337306976, 0.966666638851166], [0.516587793827057, 0.316147029399872, 0.228292018175125, 0.205642253160477]),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [30.4000718593597, 62.9767029285431, 96.690801858902, 112.145288944244], [u'accuracy'], u'categorical_crossentropy', 0.908333361148834, 0.203085869550705, [0.933333337306976, 0.808333337306976, 0.958333313465118, 0.908333361148834], [0.372362315654755, 0.304766088724136, 0.11820487678051, 0.203085869550705], 0.933333337306976, 0.190864190459251, [0.966666638851166, 0.833333313465118, 0.966666638851166, 0.933333337306976], [0.347199022769928, 0.290798246860504, 0.110275268554688, 0.190864190459251]),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [30.875373840332, 63.4593389034271, 97.1958589553833, 112.702126979828], [u'accuracy'], u'categorical_crossentropy', 0.883333325386047, 0.692279815673828, [0.533333361148834, 0.616666674613953, 0.875, 0.883333325386047], [1.08197057247162, 0.851473987102509, 0.729827761650085, 0.692279815673828], 0.899999976158142, 0.674779772758484, [0.600000023841858, 0.666666686534882, 0.899999976158142, 0.899999976158142], [1.05298256874084, 0.817528009414673, 0.710631787776947, 0.674779772758484]),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [29.8903229236603, 62.4677069187164, 96.1764039993286, 111.539803981781], [u'accuracy'], u'categorical_crossentropy', 0.925000011920929, 0.176520362496376, [0.833333313465118, 0.925000011920929, 0.774999976158142, 0.925000011920929], [0.324734181165695, 0.182637020945549, 0.468331128358841, 0.176520362496376], 0.899999976158142, 0.258552938699722, [0.866666674613953, 0.866666674613953, 0.866666674613953, 0.899999976158142], [0.341204434633255, 0.261798053979874, 0.45467621088028, 0.258552938699722]),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [31.7836039066315, 64.4592599868774, 98.1328208446503, 113.377946853638], [u'accuracy'], u'categorical_crossentropy', 0.891666650772095, 0.797108471393585, [0.341666668653488, 0.491666674613953, 0.916666686534882, 0.891666650772095], [1.09786474704742, 0.967048287391663, 0.838281869888306, 0.797108471393585], 0.899999976158142, 0.800795376300812, [0.300000011920929, 0.433333337306976, 0.933333337306976, 0.899999976158142], [1.07609903812408, 0.962578594684601, 0.834975183010101, 0.800795376300812]),\n",
+       " (9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [30.1456639766693, 62.722916841507, 96.4333670139313, 111.892151832581], [u'accuracy'], u'categorical_crossentropy', 0.816666662693024, 0.734887838363647, [0.850000023841858, 0.958333313465118, 0.966666638851166, 0.816666662693024], [0.335647404193878, 0.0894104242324829, 0.0672163665294647, 0.734887838363647], 0.866666674613953, 0.66532301902771, [0.866666674613953, 0.966666638851166, 0.966666638851166, 0.866666674613953], [0.320426166057587, 0.154994085431099, 0.204012081027031, 0.66532301902771]),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [32.0452349185944, 64.7241299152374, 98.4015560150146, 113.899842977524], [u'accuracy'], u'categorical_crossentropy', 0.791666686534882, 0.772948563098907, [0.316666662693024, 0.349999994039536, 0.725000023841858, 0.791666686534882], [1.01266825199127, 0.905348658561707, 0.807280421257019, 0.772948563098907], 0.866666674613953, 0.740880072116852, [0.400000005960464, 0.466666668653488, 0.800000011920929, 0.866666674613953], [0.964996755123138, 0.868514597415924, 0.771895349025726, 0.740880072116852]),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [30.6602540016174, 63.2428169250488, 96.9531948566437, 112.484740972519], [u'accuracy'], u'categorical_crossentropy', 0.691666662693024, 0.501820206642151, [0.658333361148834, 0.658333361148834, 0.658333361148834, 0.691666662693024], [0.654709756374359, 0.581917643547058, 1.33844769001007, 0.501820206642151], 0.766666650772095, 0.457984447479248, [0.699999988079071, 0.699999988079071, 0.699999988079071, 0.766666650772095], [0.592061340808868, 0.525563180446625, 1.17788350582123, 0.457984447479248]),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [31.0910878181458, 63.7646949291229, 97.4185988903046, 112.939773797989], [u'accuracy'], u'categorical_crossentropy', 0.666666686534882, 0.50052946805954, [0.433333337306976, 0.641666650772095, 0.649999976158142, 0.666666686534882], [0.850135624408722, 0.611121952533722, 0.509139358997345, 0.50052946805954], 0.733333349227905, 0.459399551153183, [0.466666668653488, 0.699999988079071, 0.699999988079071, 0.733333349227905], [0.802468597888947, 0.571285247802734, 0.492577910423279, 0.459399551153183]),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [29.6670269966125, 62.2440509796143, 95.9554150104523, 111.311369895935], [u'accuracy'], u'categorical_crossentropy', 0.733333349227905, 0.82194459438324, [0.341666668653488, 0.341666668653488, 0.658333361148834, 0.733333349227905], [1.06431686878204, 0.996406197547913, 0.869706034660339, 0.82194459438324], 0.699999988079071, 0.852133929729462, [0.300000011920929, 0.300000011920929, 0.699999988079071, 0.699999988079071], [1.09268116950989, 1.01670277118683, 0.891825795173645, 0.852133929729462]),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [32.5322558879852, 65.2217888832092, 98.9477097988129, 114.400418996811], [u'accuracy'], u'categorical_crossentropy', 0.683333337306976, 0.455871999263763, [0.725000023841858, 0.683333337306976, 0.683333337306976, 0.683333337306976], [0.383917421102524, 0.457853585481644, 0.455943495035172, 0.455871999263763], 0.600000023841858, 0.488439053297043, [0.800000011920929, 0.600000023841858, 0.600000023841858, 0.600000023841858], [0.388951361179352, 0.50080794095993, 0.487448841333389, 0.488439053297043]),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [32.2720308303833, 64.9502189159393, 98.6836059093475, 114.134181976318], [u'accuracy'], u'categorical_crossentropy', 0.675000011920929, 0.452209770679474, [0.683333337306976, 0.675000011920929, 0.683333337306976, 0.675000011920929], [0.492754250764847, 0.469423890113831, 0.571796059608459, 0.452209770679474], 0.600000023841858, 0.464268505573273, [0.733333349227905, 0.766666650772095, 0.600000023841858, 0.600000023841858], [0.438488334417343, 0.390993624925613, 0.690678656101227, 0.464268505573273])]"
+      ]
+     },
+     "execution_count": 27,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_info ORDER BY validation_metrics_final DESC;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Plot validation results"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 28,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%matplotlib notebook\n",
+    "import matplotlib.pyplot as plt\n",
+    "from matplotlib.ticker import MaxNLocator\n",
+    "from collections import defaultdict\n",
+    "import pandas as pd\n",
+    "import seaborn as sns\n",
+    "sns.set_palette(sns.color_palette(\"hls\", 20))\n",
+    "plt.rcParams.update({'font.size': 12})\n",
+    "pd.set_option('display.max_colwidth', -1)"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 29,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "7 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM iris_multi_model_info ORDER BY validation_loss ASC LIMIT 7;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "df_summary = %sql SELECT * FROM iris_multi_model_summary;\n",
+    "df_summary = df_summary.DataFrame()\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Iteration')\n",
+    "ax_metric.set_ylabel('Metric')\n",
+    "ax_metric.set_title('Validation metric curve')\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Iteration')\n",
+    "ax_loss.set_ylabel('Loss')\n",
+    "ax_loss.set_title('Validation loss curve')\n",
+    "\n",
+    "iters = df_summary['metrics_iters'][0]\n",
+    "\n",
+    "for mst_key in df_results['mst_key']:\n",
+    "    df_output_info = %sql SELECT validation_metrics,validation_loss FROM iris_multi_model_info WHERE mst_key = $mst_key\n",
+    "    df_output_info = df_output_info.DataFrame()\n",
+    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
+    "    validation_loss = df_output_info['validation_loss'][0]\n",
+    "    \n",
+    "    ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
+    "    ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred_prob\"></a>\n",
+    "# 2.  Predict probabilities\n",
+    "\n",
+    "Predict with probabilities for each class:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 30,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "90 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999932</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>6.7611923e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.2535056e-10</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999808</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>1.9209425e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>4.433645e-10</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99998367</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>1.6334934e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>4.3492965e-10</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999931</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>6.9504345e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.9190094e-10</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99999726</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>2.719827e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>2.4018267e-11</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9999982</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>1.8036015e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.515534e-11</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99996376</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>3.623055e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.4014193e-09</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99995685</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>4.3105167e-05</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.541236e-09</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99999833</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>1.6733742e-06</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.0720992e-11</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.97456545</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.025385397</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>4.912654e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8837083</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.11627731</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.4444132e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9832433</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.016161945</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0005947249</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9934144</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.006202936</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00038262276</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9880006</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.01050145</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0014980072</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.743757</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.25624287</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.1804799e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9489498</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.050999135</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>5.1051586e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.9882598</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.011410431</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00032975432</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7122672</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.2864844</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0012483773</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8344315</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.16556835</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>4.9313943e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7617606</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.23823881</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>6.2156596e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.85601324</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.1439867</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.4068247e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.76065344</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.23934652</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>4.0775706e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.65924823</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.34075174</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.7877243e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.968423</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.031577036</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>1.5606285e-11</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.72842705</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.2715729</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.7875385e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8053533</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.19464317</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.5179064e-06</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7297866</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.2702134</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>2.8784607e-08</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5341273</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4658725</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>2.3799986e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.6266347</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3733647</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>5.7692125e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5517554</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4482443</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.1108453e-07</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3, u'class_text', u'Iris-setosa', 0.9999932, 1),\n",
+       " (3, u'class_text', u'Iris-versicolor', 6.7611923e-06, 2),\n",
+       " (3, u'class_text', u'Iris-virginica', 1.2535056e-10, 3),\n",
+       " (10, u'class_text', u'Iris-setosa', 0.9999808, 1),\n",
+       " (10, u'class_text', u'Iris-versicolor', 1.9209425e-05, 2),\n",
+       " (10, u'class_text', u'Iris-virginica', 4.433645e-10, 3),\n",
+       " (12, u'class_text', u'Iris-setosa', 0.99998367, 1),\n",
+       " (12, u'class_text', u'Iris-versicolor', 1.6334934e-05, 2),\n",
+       " (12, u'class_text', u'Iris-virginica', 4.3492965e-10, 3),\n",
+       " (14, u'class_text', u'Iris-setosa', 0.9999931, 1),\n",
+       " (14, u'class_text', u'Iris-versicolor', 6.9504345e-06, 2),\n",
+       " (14, u'class_text', u'Iris-virginica', 1.9190094e-10, 3),\n",
+       " (18, u'class_text', u'Iris-setosa', 0.99999726, 1),\n",
+       " (18, u'class_text', u'Iris-versicolor', 2.719827e-06, 2),\n",
+       " (18, u'class_text', u'Iris-virginica', 2.4018267e-11, 3),\n",
+       " (20, u'class_text', u'Iris-setosa', 0.9999982, 1),\n",
+       " (20, u'class_text', u'Iris-versicolor', 1.8036015e-06, 2),\n",
+       " (20, u'class_text', u'Iris-virginica', 1.515534e-11, 3),\n",
+       " (30, u'class_text', u'Iris-setosa', 0.99996376, 1),\n",
+       " (30, u'class_text', u'Iris-versicolor', 3.623055e-05, 2),\n",
+       " (30, u'class_text', u'Iris-virginica', 1.4014193e-09, 3),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.99995685, 1),\n",
+       " (31, u'class_text', u'Iris-versicolor', 4.3105167e-05, 2),\n",
+       " (31, u'class_text', u'Iris-virginica', 1.541236e-09, 3),\n",
+       " (49, u'class_text', u'Iris-setosa', 0.99999833, 1),\n",
+       " (49, u'class_text', u'Iris-versicolor', 1.6733742e-06, 2),\n",
+       " (49, u'class_text', u'Iris-virginica', 1.0720992e-11, 3),\n",
+       " (55, u'class_text', u'Iris-versicolor', 0.97456545, 1),\n",
+       " (55, u'class_text', u'Iris-virginica', 0.025385397, 2),\n",
+       " (55, u'class_text', u'Iris-setosa', 4.912654e-05, 3),\n",
+       " (64, u'class_text', u'Iris-versicolor', 0.8837083, 1),\n",
+       " (64, u'class_text', u'Iris-virginica', 0.11627731, 2),\n",
+       " (64, u'class_text', u'Iris-setosa', 1.4444132e-05, 3),\n",
+       " (70, u'class_text', u'Iris-versicolor', 0.9832433, 1),\n",
+       " (70, u'class_text', u'Iris-virginica', 0.016161945, 2),\n",
+       " (70, u'class_text', u'Iris-setosa', 0.0005947249, 3),\n",
+       " (76, u'class_text', u'Iris-versicolor', 0.9934144, 1),\n",
+       " (76, u'class_text', u'Iris-virginica', 0.006202936, 2),\n",
+       " (76, u'class_text', u'Iris-setosa', 0.00038262276, 3),\n",
+       " (82, u'class_text', u'Iris-versicolor', 0.9880006, 1),\n",
+       " (82, u'class_text', u'Iris-virginica', 0.01050145, 2),\n",
+       " (82, u'class_text', u'Iris-setosa', 0.0014980072, 3),\n",
+       " (84, u'class_text', u'Iris-virginica', 0.743757, 1),\n",
+       " (84, u'class_text', u'Iris-versicolor', 0.25624287, 2),\n",
+       " (84, u'class_text', u'Iris-setosa', 1.1804799e-07, 3),\n",
+       " (92, u'class_text', u'Iris-versicolor', 0.9489498, 1),\n",
+       " (92, u'class_text', u'Iris-virginica', 0.050999135, 2),\n",
+       " (92, u'class_text', u'Iris-setosa', 5.1051586e-05, 3),\n",
+       " (98, u'class_text', u'Iris-versicolor', 0.9882598, 1),\n",
+       " (98, u'class_text', u'Iris-virginica', 0.011410431, 2),\n",
+       " (98, u'class_text', u'Iris-setosa', 0.00032975432, 3),\n",
+       " (99, u'class_text', u'Iris-versicolor', 0.7122672, 1),\n",
+       " (99, u'class_text', u'Iris-setosa', 0.2864844, 2),\n",
+       " (99, u'class_text', u'Iris-virginica', 0.0012483773, 3),\n",
+       " (102, u'class_text', u'Iris-virginica', 0.8344315, 1),\n",
+       " (102, u'class_text', u'Iris-versicolor', 0.16556835, 2),\n",
+       " (102, u'class_text', u'Iris-setosa', 4.9313943e-08, 3),\n",
+       " (107, u'class_text', u'Iris-virginica', 0.7617606, 1),\n",
+       " (107, u'class_text', u'Iris-versicolor', 0.23823881, 2),\n",
+       " (107, u'class_text', u'Iris-setosa', 6.2156596e-07, 3),\n",
+       " (114, u'class_text', u'Iris-virginica', 0.85601324, 1),\n",
+       " (114, u'class_text', u'Iris-versicolor', 0.1439867, 2),\n",
+       " (114, u'class_text', u'Iris-setosa', 3.4068247e-08, 3),\n",
+       " (117, u'class_text', u'Iris-virginica', 0.76065344, 1),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.23934652, 2),\n",
+       " (117, u'class_text', u'Iris-setosa', 4.0775706e-08, 3),\n",
+       " (121, u'class_text', u'Iris-virginica', 0.65924823, 1),\n",
+       " (121, u'class_text', u'Iris-versicolor', 0.34075174, 2),\n",
+       " (121, u'class_text', u'Iris-setosa', 3.7877243e-08, 3),\n",
+       " (123, u'class_text', u'Iris-virginica', 0.968423, 1),\n",
+       " (123, u'class_text', u'Iris-versicolor', 0.031577036, 2),\n",
+       " (123, u'class_text', u'Iris-setosa', 1.5606285e-11, 3),\n",
+       " (125, u'class_text', u'Iris-virginica', 0.72842705, 1),\n",
+       " (125, u'class_text', u'Iris-versicolor', 0.2715729, 2),\n",
+       " (125, u'class_text', u'Iris-setosa', 3.7875385e-08, 3),\n",
+       " (127, u'class_text', u'Iris-versicolor', 0.8053533, 1),\n",
+       " (127, u'class_text', u'Iris-virginica', 0.19464317, 2),\n",
+       " (127, u'class_text', u'Iris-setosa', 3.5179064e-06, 3),\n",
+       " (145, u'class_text', u'Iris-virginica', 0.7297866, 1),\n",
+       " (145, u'class_text', u'Iris-versicolor', 0.2702134, 2),\n",
+       " (145, u'class_text', u'Iris-setosa', 2.8784607e-08, 3),\n",
+       " (147, u'class_text', u'Iris-virginica', 0.5341273, 1),\n",
+       " (147, u'class_text', u'Iris-versicolor', 0.4658725, 2),\n",
+       " (147, u'class_text', u'Iris-setosa', 2.3799986e-07, 3),\n",
+       " (148, u'class_text', u'Iris-versicolor', 0.6266347, 1),\n",
+       " (148, u'class_text', u'Iris-virginica', 0.3733647, 2),\n",
+       " (148, u'class_text', u'Iris-setosa', 5.7692125e-07, 3),\n",
+       " (149, u'class_text', u'Iris-virginica', 0.5517554, 1),\n",
+       " (149, u'class_text', u'Iris-versicolor', 0.4482443, 2),\n",
+       " (149, u'class_text', u'Iris-setosa', 3.1108453e-07, 3)]"
+      ]
+     },
+     "execution_count": 30,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_multi_model', -- model\n",
+    "                                   'iris_test',        -- test_table\n",
+    "                                   'id',               -- id column\n",
+    "                                   'attributes',       -- independent var\n",
+    "                                   'iris_predict',     -- output table\n",
+    "                                    'prob',            -- prediction type\n",
+    "                                    FALSE,             -- use gpus\n",
+    "                                    3                  -- mst_key to use\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id, rank;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"warm_start\"></a>\n",
+    "# 3.  Warm start\n",
+    "\n",
+    "Next, use the warm_start parameter to continue learning, using the coefficients from the run above. Note that we don't drop the model table or model summary table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 31,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit_multiple_model</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 31,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.madlib_keras_fit_multiple_model('iris_train_packed',    -- source_table\n",
+    "                                              'iris_multi_model',     -- model_output_table\n",
+    "                                              'mst_table',            -- model_selection_table\n",
+    "                                               3,                     -- num_iterations\n",
+    "                                               FALSE,                 -- use gpus\n",
+    "                                              'iris_test_packed',     -- validation dataset\n",
+    "                                               1,                     -- metrics compute frequency\n",
+    "                                               TRUE,                  -- warm start\n",
+    "                                              'Sophie L.',            -- name\n",
+    "                                              'Simple MLP for iris dataset'  -- description\n",
+    "                                             );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 32,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>model_info</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_selection_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>warm_start</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>iris_multi_model</td>\n",
+       "        <td>iris_multi_model_info</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>mst_table</td>\n",
+       "        <td>None</td>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>True</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Simple MLP for iris dataset</td>\n",
+       "        <td>2021-03-06 00:55:34.010762</td>\n",
+       "        <td>2021-03-06 00:56:20.576330</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[1]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[1, 2, 3]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_test_packed', u'iris_multi_model', u'iris_multi_model_info', [u'class_text'], [u'attributes'], u'model_arch_library', u'mst_table', None, 3, 1, True, u'Sophie L.', u'Simple MLP for iris dataset', datetime.datetime(2021, 3, 6, 0, 55, 34, 10762), datetime.datetime(2021, 3, 6, 0, 56, 20, 576330), u'1.18.0-dev', [1], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], [u'character varying'], 1.0, [1, 2, 3])]"
+      ]
+     },
+     "execution_count": 32,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View performance of each model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 33,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "12 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>mst_key</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[12.8246030807495, 28.3149819374084, 43.8511519432068]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.125932246447</td>\n",
+       "        <td>[0.983333349227905, 0.908333361148834, 0.949999988079071]</td>\n",
+       "        <td>[0.0759517326951027, 0.280529856681824, 0.125932246446609]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.262804627419</td>\n",
+       "        <td>[0.966666638851166, 0.933333337306976, 0.966666638851166]</td>\n",
+       "        <td>[0.115140154957771, 0.282798647880554, 0.262804627418518]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[12.3267669677734, 27.5790538787842, 43.3719210624695]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.958333313465</td>\n",
+       "        <td>0.646220803261</td>\n",
+       "        <td>[0.916666686534882, 0.774999976158142, 0.958333313465118]</td>\n",
+       "        <td>[0.760809063911438, 0.70676600933075, 0.646220803260803]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.676706075668</td>\n",
+       "        <td>[0.899999976158142, 0.699999988079071, 0.966666638851166]</td>\n",
+       "        <td>[0.789911270141602, 0.741125166416168, 0.676706075668335]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[13.8655989170074, 29.3921880722046, 45.186311006546]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.161019146442</td>\n",
+       "        <td>[0.608333349227905, 0.975000023841858, 0.966666638851166]</td>\n",
+       "        <td>[0.656926870346069, 0.154457986354828, 0.161019146442413]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.184286847711</td>\n",
+       "        <td>[0.666666686534882, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.60343611240387, 0.166501134634018, 0.184286847710609]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[12.5584180355072, 27.7957689762115, 43.5938129425049]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.925000011921</td>\n",
+       "        <td>0.125614732504</td>\n",
+       "        <td>[0.850000023841858, 0.908333361148834, 0.925000011920929]</td>\n",
+       "        <td>[0.311796188354492, 0.228279903531075, 0.125614732503891]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.205575048923</td>\n",
+       "        <td>[0.699999988079071, 0.899999976158142, 0.933333337306976]</td>\n",
+       "        <td>[0.434732705354691, 0.278642177581787, 0.205575048923492]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[14.3016650676727, 29.8289239406586, 45.6773319244385]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.916666686535</td>\n",
+       "        <td>0.680241525173</td>\n",
+       "        <td>[0.899999976158142, 0.899999976158142, 0.916666686534882]</td>\n",
+       "        <td>[0.75947380065918, 0.717410624027252, 0.680241525173187]</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>0.685820519924</td>\n",
+       "        <td>[0.933333337306976, 0.933333337306976, 0.933333337306976]</td>\n",
+       "        <td>[0.764581918716431, 0.718774557113647, 0.685820519924164]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[13.6457929611206, 29.1624140739441, 44.9534199237823]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.891666650772</td>\n",
+       "        <td>0.590237081051</td>\n",
+       "        <td>[0.824999988079071, 0.783333361148834, 0.891666650772095]</td>\n",
+       "        <td>[0.666068911552429, 0.633061707019806, 0.590237081050873]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.576045572758</td>\n",
+       "        <td>[0.866666674613953, 0.866666674613953, 0.899999976158142]</td>\n",
+       "        <td>[0.645683944225311, 0.608498632907867, 0.576045572757721]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[14.0837008953094, 29.6097829341888, 45.4142129421234]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.916666686535</td>\n",
+       "        <td>0.174454689026</td>\n",
+       "        <td>[0.949999988079071, 0.958333313465118, 0.916666686534882]</td>\n",
+       "        <td>[0.166735425591469, 0.141851797699928, 0.174454689025879]</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>0.219132959843</td>\n",
+       "        <td>[0.966666638851166, 0.933333337306976, 0.899999976158142]</td>\n",
+       "        <td>[0.186790466308594, 0.176578417420387, 0.219132959842682]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[13.1594960689545, 28.5860660076141, 44.1881170272827]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.285291582346</td>\n",
+       "        <td>[0.774999976158142, 0.949999988079071, 0.866666674613953]</td>\n",
+       "        <td>[0.441815197467804, 0.140827313065529, 0.285291582345963]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.246576815844</td>\n",
+       "        <td>[0.766666650772095, 0.966666638851166, 0.866666674613953]</td>\n",
+       "        <td>[0.4128278195858, 0.146319955587387, 0.246576815843582]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[14.5546190738678, 30.0798380374908, 45.94082903862]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.850000023842</td>\n",
+       "        <td>0.675731360912</td>\n",
+       "        <td>[0.791666686534882, 0.841666638851166, 0.850000023841858]</td>\n",
+       "        <td>[0.746130049228668, 0.706377267837524, 0.675731360912323]</td>\n",
+       "        <td>0.866666674614</td>\n",
+       "        <td>0.650432705879</td>\n",
+       "        <td>[0.866666674613953, 0.866666674613953, 0.866666674613953]</td>\n",
+       "        <td>[0.712817847728729, 0.677974581718445, 0.650432705879211]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[15.3575170040131, 30.5435180664062, 46.5635209083557]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.658333361149</td>\n",
+       "        <td>0.45723798871</td>\n",
+       "        <td>[0.658333361148834, 0.683333337306976, 0.658333361148834]</td>\n",
+       "        <td>[0.457635939121246, 0.455960959196091, 0.457237988710403]</td>\n",
+       "        <td>0.699999988079</td>\n",
+       "        <td>0.48275628686</td>\n",
+       "        <td>[0.699999988079071, 0.600000023841858, 0.699999988079071]</td>\n",
+       "        <td>[0.48207613825798, 0.491984754800797, 0.482756286859512]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>1</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=4</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.75390625</td>\n",
+       "        <td>[14.8466219902039, 30.2953569889069, 46.1656670570374]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.683333337307</td>\n",
+       "        <td>0.456283688545</td>\n",
+       "        <td>[0.925000011920929, 0.899999976158142, 0.683333337306976]</td>\n",
+       "        <td>[0.224153310060501, 0.295417010784149, 0.456283688545227]</td>\n",
+       "        <td>0.600000023842</td>\n",
+       "        <td>0.494575560093</td>\n",
+       "        <td>[0.966666638851166, 0.899999976158142, 0.600000023841858]</td>\n",
+       "        <td>[0.227903217077255, 0.345975488424301, 0.494575560092926]</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>2</td>\n",
+       "        <td>optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'</td>\n",
+       "        <td>epochs=1,batch_size=8</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>1.18359375</td>\n",
+       "        <td>[13.4095330238342, 28.938658952713, 44.7153990268707]</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.691666662693</td>\n",
+       "        <td>0.528191685677</td>\n",
+       "        <td>[0.708333313465118, 0.966666638851166, 0.691666662693024]</td>\n",
+       "        <td>[0.395545929670334, 0.100506067276001, 0.528191685676575]</td>\n",
+       "        <td>0.566666662693</td>\n",
+       "        <td>0.720313131809</td>\n",
+       "        <td>[0.633333325386047, 0.966666638851166, 0.566666662693024]</td>\n",
+       "        <td>[0.508394777774811, 0.130626574158669, 0.720313131809235]</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(9, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [12.8246030807495, 28.3149819374084, 43.8511519432068], [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.125932246446609, [0.983333349227905, 0.908333361148834, 0.949999988079071], [0.0759517326951027, 0.280529856681824, 0.125932246446609], 0.966666638851166, 0.262804627418518, [0.966666638851166, 0.933333337306976, 0.966666638851166], [0.115140154957771, 0.282798647880554, 0.262804627418518]),\n",
+       " (7, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [12.3267669677734, 27.5790538787842, 43.3719210624695], [u'accuracy'], u'categorical_crossentropy', 0.958333313465118, 0.646220803260803, [0.916666686534882, 0.774999976158142, 0.958333313465118], [0.760809063911438, 0.70676600933075, 0.646220803260803], 0.966666638851166, 0.676706075668335, [0.899999976158142, 0.699999988079071, 0.966666638851166], [0.789911270141602, 0.741125166416168, 0.676706075668335]),\n",
+       " (6, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [13.8655989170074, 29.3921880722046, 45.186311006546], [u'accuracy'], u'categorical_crossentropy', 0.966666638851166, 0.161019146442413, [0.608333349227905, 0.975000023841858, 0.966666638851166], [0.656926870346069, 0.154457986354828, 0.161019146442413], 0.966666638851166, 0.184286847710609, [0.666666686534882, 0.966666638851166, 0.966666638851166], [0.60343611240387, 0.166501134634018, 0.184286847710609]),\n",
+       " (3, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [12.5584180355072, 27.7957689762115, 43.5938129425049], [u'accuracy'], u'categorical_crossentropy', 0.925000011920929, 0.125614732503891, [0.850000023841858, 0.908333361148834, 0.925000011920929], [0.311796188354492, 0.228279903531075, 0.125614732503891], 0.933333337306976, 0.205575048923492, [0.699999988079071, 0.899999976158142, 0.933333337306976], [0.434732705354691, 0.278642177581787, 0.205575048923492]),\n",
+       " (1, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [14.3016650676727, 29.8289239406586, 45.6773319244385], [u'accuracy'], u'categorical_crossentropy', 0.916666686534882, 0.680241525173187, [0.899999976158142, 0.899999976158142, 0.916666686534882], [0.75947380065918, 0.717410624027252, 0.680241525173187], 0.933333337306976, 0.685820519924164, [0.933333337306976, 0.933333337306976, 0.933333337306976], [0.764581918716431, 0.718774557113647, 0.685820519924164]),\n",
+       " (2, 1, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [13.6457929611206, 29.1624140739441, 44.9534199237823], [u'accuracy'], u'categorical_crossentropy', 0.891666650772095, 0.590237081050873, [0.824999988079071, 0.783333361148834, 0.891666650772095], [0.666068911552429, 0.633061707019806, 0.590237081050873], 0.899999976158142, 0.576045572757721, [0.866666674613953, 0.866666674613953, 0.899999976158142], [0.645683944225311, 0.608498632907867, 0.576045572757721]),\n",
+       " (4, 1, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 0.75390625, [14.0837008953094, 29.6097829341888, 45.4142129421234], [u'accuracy'], u'categorical_crossentropy', 0.916666686534882, 0.174454689025879, [0.949999988079071, 0.958333313465118, 0.916666686534882], [0.166735425591469, 0.141851797699928, 0.174454689025879], 0.899999976158142, 0.219132959842682, [0.966666638851166, 0.933333337306976, 0.899999976158142], [0.186790466308594, 0.176578417420387, 0.219132959842682]),\n",
+       " (10, 2, u\"optimizer='Adam(lr=0.01)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [13.1594960689545, 28.5860660076141, 44.1881170272827], [u'accuracy'], u'categorical_crossentropy', 0.866666674613953, 0.285291582345963, [0.774999976158142, 0.949999988079071, 0.866666674613953], [0.441815197467804, 0.140827313065529, 0.285291582345963], 0.866666674613953, 0.246576815843582, [0.766666650772095, 0.966666638851166, 0.866666674613953], [0.4128278195858, 0.146319955587387, 0.246576815843582]),\n",
+       " (8, 2, u\"optimizer='Adam(lr=0.001)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [14.5546190738678, 30.0798380374908, 45.94082903862], [u'accuracy'], u'categorical_crossentropy', 0.850000023841858, 0.675731360912323, [0.791666686534882, 0.841666638851166, 0.850000023841858], [0.746130049228668, 0.706377267837524, 0.675731360912323], 0.866666674613953, 0.650432705879211, [0.866666674613953, 0.866666674613953, 0.866666674613953], [0.712817847728729, 0.677974581718445, 0.650432705879211]),\n",
+       " (11, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 1.18359375, [15.3575170040131, 30.5435180664062, 46.5635209083557], [u'accuracy'], u'categorical_crossentropy', 0.658333361148834, 0.457237988710403, [0.658333361148834, 0.683333337306976, 0.658333361148834], [0.457635939121246, 0.455960959196091, 0.457237988710403], 0.699999988079071, 0.482756286859512, [0.699999988079071, 0.600000023841858, 0.699999988079071], [0.48207613825798, 0.491984754800797, 0.482756286859512]),\n",
+       " (5, 1, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=4', u'madlib_keras', 0.75390625, [14.8466219902039, 30.2953569889069, 46.1656670570374], [u'accuracy'], u'categorical_crossentropy', 0.683333337306976, 0.456283688545227, [0.925000011920929, 0.899999976158142, 0.683333337306976], [0.224153310060501, 0.295417010784149, 0.456283688545227], 0.600000023841858, 0.494575560092926, [0.966666638851166, 0.899999976158142, 0.600000023841858], [0.227903217077255, 0.345975488424301, 0.494575560092926]),\n",
+       " (12, 2, u\"optimizer='Adam(lr=0.1)',metrics=['accuracy'],loss='categorical_crossentropy'\", u'epochs=1,batch_size=8', u'madlib_keras', 1.18359375, [13.4095330238342, 28.938658952713, 44.7153990268707], [u'accuracy'], u'categorical_crossentropy', 0.691666662693024, 0.528191685676575, [0.708333313465118, 0.966666638851166, 0.691666662693024], [0.395545929670334, 0.100506067276001, 0.528191685676575], 0.566666662693024, 0.720313131809235, [0.633333325386047, 0.966666638851166, 0.566666662693024], [0.508394777774811, 0.130626574158669, 0.720313131809235])]"
+      ]
+     },
+     "execution_count": 33,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_multi_model_info ORDER BY validation_metrics_final DESC;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Plot validation results:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 34,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "7 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "application/javascript": [
+       "/* Put everything inside the global mpl namespace */\n",
+       "window.mpl = {};\n",
+       "\n",
+       "\n",
+       "mpl.get_websocket_type = function() {\n",
+       "    if (typeof(WebSocket) !== 'undefined') {\n",
+       "        return WebSocket;\n",
+       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
+       "        return MozWebSocket;\n",
+       "    } else {\n",
+       "        alert('Your browser does not have WebSocket support.' +\n",
+       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
+       "              'Firefox 4 and 5 are also supported but you ' +\n",
+       "              'have to enable WebSockets in about:config.');\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
+       "    this.id = figure_id;\n",
+       "\n",
+       "    this.ws = websocket;\n",
+       "\n",
+       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
+       "\n",
+       "    if (!this.supports_binary) {\n",
+       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
+       "        if (warnings) {\n",
+       "            warnings.style.display = 'block';\n",
+       "            warnings.textContent = (\n",
+       "                \"This browser does not support binary websocket messages. \" +\n",
+       "                    \"Performance may be slow.\");\n",
+       "        }\n",
+       "    }\n",
+       "\n",
+       "    this.imageObj = new Image();\n",
+       "\n",
+       "    this.context = undefined;\n",
+       "    this.message = undefined;\n",
+       "    this.canvas = undefined;\n",
+       "    this.rubberband_canvas = undefined;\n",
+       "    this.rubberband_context = undefined;\n",
+       "    this.format_dropdown = undefined;\n",
+       "\n",
+       "    this.image_mode = 'full';\n",
+       "\n",
+       "    this.root = $('<div/>');\n",
+       "    this._root_extra_style(this.root)\n",
+       "    this.root.attr('style', 'display: inline-block');\n",
+       "\n",
+       "    $(parent_element).append(this.root);\n",
+       "\n",
+       "    this._init_header(this);\n",
+       "    this._init_canvas(this);\n",
+       "    this._init_toolbar(this);\n",
+       "\n",
+       "    var fig = this;\n",
+       "\n",
+       "    this.waiting = false;\n",
+       "\n",
+       "    this.ws.onopen =  function () {\n",
+       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
+       "            fig.send_message(\"send_image_mode\", {});\n",
+       "            if (mpl.ratio != 1) {\n",
+       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
+       "            }\n",
+       "            fig.send_message(\"refresh\", {});\n",
+       "        }\n",
+       "\n",
+       "    this.imageObj.onload = function() {\n",
+       "            if (fig.image_mode == 'full') {\n",
+       "                // Full images could contain transparency (where diff images\n",
+       "                // almost always do), so we need to clear the canvas so that\n",
+       "                // there is no ghosting.\n",
+       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "            }\n",
+       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
+       "        };\n",
+       "\n",
+       "    this.imageObj.onunload = function() {\n",
+       "        fig.ws.close();\n",
+       "    }\n",
+       "\n",
+       "    this.ws.onmessage = this._make_on_message_function(this);\n",
+       "\n",
+       "    this.ondownload = ondownload;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_header = function() {\n",
+       "    var titlebar = $(\n",
+       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
+       "        'ui-helper-clearfix\"/>');\n",
+       "    var titletext = $(\n",
+       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
+       "        'text-align: center; padding: 3px;\"/>');\n",
+       "    titlebar.append(titletext)\n",
+       "    this.root.append(titlebar);\n",
+       "    this.header = titletext[0];\n",
+       "}\n",
+       "\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_canvas = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var canvas_div = $('<div/>');\n",
+       "\n",
+       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
+       "\n",
+       "    function canvas_keyboard_event(event) {\n",
+       "        return fig.key_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
+       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
+       "    this.canvas_div = canvas_div\n",
+       "    this._canvas_extra_style(canvas_div)\n",
+       "    this.root.append(canvas_div);\n",
+       "\n",
+       "    var canvas = $('<canvas/>');\n",
+       "    canvas.addClass('mpl-canvas');\n",
+       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
+       "\n",
+       "    this.canvas = canvas[0];\n",
+       "    this.context = canvas[0].getContext(\"2d\");\n",
+       "\n",
+       "    var backingStore = this.context.backingStorePixelRatio ||\n",
+       "\tthis.context.webkitBackingStorePixelRatio ||\n",
+       "\tthis.context.mozBackingStorePixelRatio ||\n",
+       "\tthis.context.msBackingStorePixelRatio ||\n",
+       "\tthis.context.oBackingStorePixelRatio ||\n",
+       "\tthis.context.backingStorePixelRatio || 1;\n",
+       "\n",
+       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
+       "\n",
+       "    var rubberband = $('<canvas/>');\n",
+       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
+       "\n",
+       "    var pass_mouse_events = true;\n",
+       "\n",
+       "    canvas_div.resizable({\n",
+       "        start: function(event, ui) {\n",
+       "            pass_mouse_events = false;\n",
+       "        },\n",
+       "        resize: function(event, ui) {\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "        stop: function(event, ui) {\n",
+       "            pass_mouse_events = true;\n",
+       "            fig.request_resize(ui.size.width, ui.size.height);\n",
+       "        },\n",
+       "    });\n",
+       "\n",
+       "    function mouse_event_fn(event) {\n",
+       "        if (pass_mouse_events)\n",
+       "            return fig.mouse_event(event, event['data']);\n",
+       "    }\n",
+       "\n",
+       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
+       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
+       "    // Throttle sequential mouse events to 1 every 20ms.\n",
+       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
+       "\n",
+       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
+       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
+       "\n",
+       "    canvas_div.on(\"wheel\", function (event) {\n",
+       "        event = event.originalEvent;\n",
+       "        event['data'] = 'scroll'\n",
+       "        if (event.deltaY < 0) {\n",
+       "            event.step = 1;\n",
+       "        } else {\n",
+       "            event.step = -1;\n",
+       "        }\n",
+       "        mouse_event_fn(event);\n",
+       "    });\n",
+       "\n",
+       "    canvas_div.append(canvas);\n",
+       "    canvas_div.append(rubberband);\n",
+       "\n",
+       "    this.rubberband = rubberband;\n",
+       "    this.rubberband_canvas = rubberband[0];\n",
+       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
+       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
+       "\n",
+       "    this._resize_canvas = function(width, height) {\n",
+       "        // Keep the size of the canvas, canvas container, and rubber band\n",
+       "        // canvas in synch.\n",
+       "        canvas_div.css('width', width)\n",
+       "        canvas_div.css('height', height)\n",
+       "\n",
+       "        canvas.attr('width', width * mpl.ratio);\n",
+       "        canvas.attr('height', height * mpl.ratio);\n",
+       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
+       "\n",
+       "        rubberband.attr('width', width);\n",
+       "        rubberband.attr('height', height);\n",
+       "    }\n",
+       "\n",
+       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
+       "    // upon first draw.\n",
+       "    this._resize_canvas(600, 600);\n",
+       "\n",
+       "    // Disable right mouse context menu.\n",
+       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
+       "        return false;\n",
+       "    });\n",
+       "\n",
+       "    function set_focus () {\n",
+       "        canvas.focus();\n",
+       "        canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    window.setTimeout(set_focus, 100);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) {\n",
+       "            // put a spacer in here.\n",
+       "            continue;\n",
+       "        }\n",
+       "        var button = $('<button/>');\n",
+       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
+       "                        'ui-button-icon-only');\n",
+       "        button.attr('role', 'button');\n",
+       "        button.attr('aria-disabled', 'false');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "\n",
+       "        var icon_img = $('<span/>');\n",
+       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
+       "        icon_img.addClass(image);\n",
+       "        icon_img.addClass('ui-corner-all');\n",
+       "\n",
+       "        var tooltip_span = $('<span/>');\n",
+       "        tooltip_span.addClass('ui-button-text');\n",
+       "        tooltip_span.html(tooltip);\n",
+       "\n",
+       "        button.append(icon_img);\n",
+       "        button.append(tooltip_span);\n",
+       "\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    var fmt_picker_span = $('<span/>');\n",
+       "\n",
+       "    var fmt_picker = $('<select/>');\n",
+       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
+       "    fmt_picker_span.append(fmt_picker);\n",
+       "    nav_element.append(fmt_picker_span);\n",
+       "    this.format_dropdown = fmt_picker[0];\n",
+       "\n",
+       "    for (var ind in mpl.extensions) {\n",
+       "        var fmt = mpl.extensions[ind];\n",
+       "        var option = $(\n",
+       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
+       "        fmt_picker.append(option)\n",
+       "    }\n",
+       "\n",
+       "    // Add hover states to the ui-buttons\n",
+       "    $( \".ui-button\" ).hover(\n",
+       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
+       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
+       "    );\n",
+       "\n",
+       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
+       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
+       "    // which will in turn request a refresh of the image.\n",
+       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_message = function(type, properties) {\n",
+       "    properties['type'] = type;\n",
+       "    properties['figure_id'] = this.id;\n",
+       "    this.ws.send(JSON.stringify(properties));\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.send_draw_message = function() {\n",
+       "    if (!this.waiting) {\n",
+       "        this.waiting = true;\n",
+       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    var format_dropdown = fig.format_dropdown;\n",
+       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
+       "    fig.ondownload(fig, format);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
+       "    var size = msg['size'];\n",
+       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
+       "        fig._resize_canvas(size[0], size[1]);\n",
+       "        fig.send_message(\"refresh\", {});\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
+       "    var x0 = msg['x0'] / mpl.ratio;\n",
+       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
+       "    var x1 = msg['x1'] / mpl.ratio;\n",
+       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
+       "    x0 = Math.floor(x0) + 0.5;\n",
+       "    y0 = Math.floor(y0) + 0.5;\n",
+       "    x1 = Math.floor(x1) + 0.5;\n",
+       "    y1 = Math.floor(y1) + 0.5;\n",
+       "    var min_x = Math.min(x0, x1);\n",
+       "    var min_y = Math.min(y0, y1);\n",
+       "    var width = Math.abs(x1 - x0);\n",
+       "    var height = Math.abs(y1 - y0);\n",
+       "\n",
+       "    fig.rubberband_context.clearRect(\n",
+       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
+       "\n",
+       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
+       "    // Updates the figure title.\n",
+       "    fig.header.textContent = msg['label'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
+       "    var cursor = msg['cursor'];\n",
+       "    switch(cursor)\n",
+       "    {\n",
+       "    case 0:\n",
+       "        cursor = 'pointer';\n",
+       "        break;\n",
+       "    case 1:\n",
+       "        cursor = 'default';\n",
+       "        break;\n",
+       "    case 2:\n",
+       "        cursor = 'crosshair';\n",
+       "        break;\n",
+       "    case 3:\n",
+       "        cursor = 'move';\n",
+       "        break;\n",
+       "    }\n",
+       "    fig.rubberband_canvas.style.cursor = cursor;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
+       "    fig.message.textContent = msg['message'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
+       "    // Request the server to send over a new figure.\n",
+       "    fig.send_draw_message();\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
+       "    fig.image_mode = msg['mode'];\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Called whenever the canvas gets updated.\n",
+       "    this.send_message(\"ack\", {});\n",
+       "}\n",
+       "\n",
+       "// A function to construct a web socket function for onmessage handling.\n",
+       "// Called in the figure constructor.\n",
+       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
+       "    return function socket_on_message(evt) {\n",
+       "        if (evt.data instanceof Blob) {\n",
+       "            /* FIXME: We get \"Resource interpreted as Image but\n",
+       "             * transferred with MIME type text/plain:\" errors on\n",
+       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
+       "             * to be part of the websocket stream */\n",
+       "            evt.data.type = \"image/png\";\n",
+       "\n",
+       "            /* Free the memory for the previous frames */\n",
+       "            if (fig.imageObj.src) {\n",
+       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
+       "                    fig.imageObj.src);\n",
+       "            }\n",
+       "\n",
+       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
+       "                evt.data);\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
+       "            fig.imageObj.src = evt.data;\n",
+       "            fig.updated_canvas_event();\n",
+       "            fig.waiting = false;\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        var msg = JSON.parse(evt.data);\n",
+       "        var msg_type = msg['type'];\n",
+       "\n",
+       "        // Call the  \"handle_{type}\" callback, which takes\n",
+       "        // the figure and JSON message as its only arguments.\n",
+       "        try {\n",
+       "            var callback = fig[\"handle_\" + msg_type];\n",
+       "        } catch (e) {\n",
+       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
+       "            return;\n",
+       "        }\n",
+       "\n",
+       "        if (callback) {\n",
+       "            try {\n",
+       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
+       "                callback(fig, msg);\n",
+       "            } catch (e) {\n",
+       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
+       "            }\n",
+       "        }\n",
+       "    };\n",
+       "}\n",
+       "\n",
+       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
+       "mpl.findpos = function(e) {\n",
+       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
+       "    var targ;\n",
+       "    if (!e)\n",
+       "        e = window.event;\n",
+       "    if (e.target)\n",
+       "        targ = e.target;\n",
+       "    else if (e.srcElement)\n",
+       "        targ = e.srcElement;\n",
+       "    if (targ.nodeType == 3) // defeat Safari bug\n",
+       "        targ = targ.parentNode;\n",
+       "\n",
+       "    // jQuery normalizes the pageX and pageY\n",
+       "    // pageX,Y are the mouse positions relative to the document\n",
+       "    // offset() returns the position of the element relative to the document\n",
+       "    var x = e.pageX - $(targ).offset().left;\n",
+       "    var y = e.pageY - $(targ).offset().top;\n",
+       "\n",
+       "    return {\"x\": x, \"y\": y};\n",
+       "};\n",
+       "\n",
+       "/*\n",
+       " * return a copy of an object with only non-object keys\n",
+       " * we need this to avoid circular references\n",
+       " * http://stackoverflow.com/a/24161582/3208463\n",
+       " */\n",
+       "function simpleKeys (original) {\n",
+       "  return Object.keys(original).reduce(function (obj, key) {\n",
+       "    if (typeof original[key] !== 'object')\n",
+       "        obj[key] = original[key]\n",
+       "    return obj;\n",
+       "  }, {});\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
+       "    var canvas_pos = mpl.findpos(event)\n",
+       "\n",
+       "    if (name === 'button_press')\n",
+       "    {\n",
+       "        this.canvas.focus();\n",
+       "        this.canvas_div.focus();\n",
+       "    }\n",
+       "\n",
+       "    var x = canvas_pos.x * mpl.ratio;\n",
+       "    var y = canvas_pos.y * mpl.ratio;\n",
+       "\n",
+       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
+       "                             step: event.step,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "\n",
+       "    /* This prevents the web browser from automatically changing to\n",
+       "     * the text insertion cursor when the button is pressed.  We want\n",
+       "     * to control all of the cursor setting manually through the\n",
+       "     * 'cursor' event from matplotlib */\n",
+       "    event.preventDefault();\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    // Handle any extra behaviour associated with a key event\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.key_event = function(event, name) {\n",
+       "\n",
+       "    // Prevent repeat events\n",
+       "    if (name == 'key_press')\n",
+       "    {\n",
+       "        if (event.which === this._key)\n",
+       "            return;\n",
+       "        else\n",
+       "            this._key = event.which;\n",
+       "    }\n",
+       "    if (name == 'key_release')\n",
+       "        this._key = null;\n",
+       "\n",
+       "    var value = '';\n",
+       "    if (event.ctrlKey && event.which != 17)\n",
+       "        value += \"ctrl+\";\n",
+       "    if (event.altKey && event.which != 18)\n",
+       "        value += \"alt+\";\n",
+       "    if (event.shiftKey && event.which != 16)\n",
+       "        value += \"shift+\";\n",
+       "\n",
+       "    value += 'k';\n",
+       "    value += event.which.toString();\n",
+       "\n",
+       "    this._key_event_extra(event, name);\n",
+       "\n",
+       "    this.send_message(name, {key: value,\n",
+       "                             guiEvent: simpleKeys(event)});\n",
+       "    return false;\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
+       "    if (name == 'download') {\n",
+       "        this.handle_save(this, null);\n",
+       "    } else {\n",
+       "        this.send_message(\"toolbar_button\", {name: name});\n",
+       "    }\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
+       "    this.message.textContent = tooltip;\n",
+       "};\n",
+       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
+       "\n",
+       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
+       "\n",
+       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
+       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
+       "    // object with the appropriate methods. Currently this is a non binary\n",
+       "    // socket, so there is still some room for performance tuning.\n",
+       "    var ws = {};\n",
+       "\n",
+       "    ws.close = function() {\n",
+       "        comm.close()\n",
+       "    };\n",
+       "    ws.send = function(m) {\n",
+       "        //console.log('sending', m);\n",
+       "        comm.send(m);\n",
+       "    };\n",
+       "    // Register the callback with on_msg.\n",
+       "    comm.on_msg(function(msg) {\n",
+       "        //console.log('receiving', msg['content']['data'], msg);\n",
+       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
+       "        ws.onmessage(msg['content']['data'])\n",
+       "    });\n",
+       "    return ws;\n",
+       "}\n",
+       "\n",
+       "mpl.mpl_figure_comm = function(comm, msg) {\n",
+       "    // This is the function which gets called when the mpl process\n",
+       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
+       "\n",
+       "    var id = msg.content.data.id;\n",
+       "    // Get hold of the div created by the display call when the Comm\n",
+       "    // socket was opened in Python.\n",
+       "    var element = $(\"#\" + id);\n",
+       "    var ws_proxy = comm_websocket_adapter(comm)\n",
+       "\n",
+       "    function ondownload(figure, format) {\n",
+       "        window.open(figure.imageObj.src);\n",
+       "    }\n",
+       "\n",
+       "    var fig = new mpl.figure(id, ws_proxy,\n",
+       "                           ondownload,\n",
+       "                           element.get(0));\n",
+       "\n",
+       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
+       "    // web socket which is closed, not our websocket->open comm proxy.\n",
+       "    ws_proxy.onopen();\n",
+       "\n",
+       "    fig.parent_element = element.get(0);\n",
+       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
+       "    if (!fig.cell_info) {\n",
+       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
+       "        return;\n",
+       "    }\n",
+       "\n",
+       "    var output_index = fig.cell_info[2]\n",
+       "    var cell = fig.cell_info[0];\n",
+       "\n",
+       "};\n",
+       "\n",
+       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
+       "    var width = fig.canvas.width/mpl.ratio\n",
+       "    fig.root.unbind('remove')\n",
+       "\n",
+       "    // Update the output cell to use the data from the current canvas.\n",
+       "    fig.push_to_output();\n",
+       "    var dataURL = fig.canvas.toDataURL();\n",
+       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
+       "    // the notebook keyboard shortcuts fail.\n",
+       "    IPython.keyboard_manager.enable()\n",
+       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
+       "    fig.close_ws(fig, msg);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
+       "    fig.send_message('closing', msg);\n",
+       "    // fig.ws.close()\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
+       "    // Turn the data on the canvas into data in the output cell.\n",
+       "    var width = this.canvas.width/mpl.ratio\n",
+       "    var dataURL = this.canvas.toDataURL();\n",
+       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.updated_canvas_event = function() {\n",
+       "    // Tell IPython that the notebook contents must change.\n",
+       "    IPython.notebook.set_dirty(true);\n",
+       "    this.send_message(\"ack\", {});\n",
+       "    var fig = this;\n",
+       "    // Wait a second, then push the new image to the DOM so\n",
+       "    // that it is saved nicely (might be nice to debounce this).\n",
+       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._init_toolbar = function() {\n",
+       "    var fig = this;\n",
+       "\n",
+       "    var nav_element = $('<div/>')\n",
+       "    nav_element.attr('style', 'width: 100%');\n",
+       "    this.root.append(nav_element);\n",
+       "\n",
+       "    // Define a callback function for later on.\n",
+       "    function toolbar_event(event) {\n",
+       "        return fig.toolbar_button_onclick(event['data']);\n",
+       "    }\n",
+       "    function toolbar_mouse_event(event) {\n",
+       "        return fig.toolbar_button_onmouseover(event['data']);\n",
+       "    }\n",
+       "\n",
+       "    for(var toolbar_ind in mpl.toolbar_items){\n",
+       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
+       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
+       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
+       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
+       "\n",
+       "        if (!name) { continue; };\n",
+       "\n",
+       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
+       "        button.click(method_name, toolbar_event);\n",
+       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
+       "        nav_element.append(button);\n",
+       "    }\n",
+       "\n",
+       "    // Add the status bar.\n",
+       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
+       "    nav_element.append(status_bar);\n",
+       "    this.message = status_bar[0];\n",
+       "\n",
+       "    // Add the close button to the window.\n",
+       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
+       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
+       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
+       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
+       "    buttongrp.append(button);\n",
+       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
+       "    titlebar.prepend(buttongrp);\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._root_extra_style = function(el){\n",
+       "    var fig = this\n",
+       "    el.on(\"remove\", function(){\n",
+       "\tfig.close_ws(fig, {});\n",
+       "    });\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
+       "    // this is important to make the div 'focusable\n",
+       "    el.attr('tabindex', 0)\n",
+       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
+       "    // off when our div gets focus\n",
+       "\n",
+       "    // location in version 3\n",
+       "    if (IPython.notebook.keyboard_manager) {\n",
+       "        IPython.notebook.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "    else {\n",
+       "        // location in version 2\n",
+       "        IPython.keyboard_manager.register_events(el);\n",
+       "    }\n",
+       "\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
+       "    var manager = IPython.notebook.keyboard_manager;\n",
+       "    if (!manager)\n",
+       "        manager = IPython.keyboard_manager;\n",
+       "\n",
+       "    // Check for shift+enter\n",
+       "    if (event.shiftKey && event.which == 13) {\n",
+       "        this.canvas_div.blur();\n",
+       "        // select the cell after this one\n",
+       "        var index = IPython.notebook.find_cell_index(this.cell_info[0]);\n",
+       "        IPython.notebook.select(index + 1);\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
+       "    fig.ondownload(fig, null);\n",
+       "}\n",
+       "\n",
+       "\n",
+       "mpl.find_output_cell = function(html_output) {\n",
+       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
+       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
+       "    // IPython event is triggered only after the cells have been serialised, which for\n",
+       "    // our purposes (turning an active figure into a static one), is too late.\n",
+       "    var cells = IPython.notebook.get_cells();\n",
+       "    var ncells = cells.length;\n",
+       "    for (var i=0; i<ncells; i++) {\n",
+       "        var cell = cells[i];\n",
+       "        if (cell.cell_type === 'code'){\n",
+       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
+       "                var data = cell.output_area.outputs[j];\n",
+       "                if (data.data) {\n",
+       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
+       "                    data = data.data;\n",
+       "                }\n",
+       "                if (data['text/html'] == html_output) {\n",
+       "                    return [cell, data, j];\n",
+       "                }\n",
+       "            }\n",
+       "        }\n",
+       "    }\n",
+       "}\n",
+       "\n",
+       "// Register the function which deals with the matplotlib target/channel.\n",
+       "// The kernel may be null if the page has been refreshed.\n",
+       "if (IPython.notebook.kernel != null) {\n",
+       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
+       "}\n"
+      ],
+      "text/plain": [
+       "<IPython.core.display.Javascript object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "data": {
+      "text/html": [
+       "<img src=\"\" width=\"720\">"
+      ],
+      "text/plain": [
+       "<IPython.core.display.HTML object>"
+      ]
+     },
+     "metadata": {},
+     "output_type": "display_data"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    }
+   ],
+   "source": [
+    "df_results = %sql SELECT * FROM iris_multi_model_info ORDER BY validation_loss ASC LIMIT 7;\n",
+    "df_results = df_results.DataFrame()\n",
+    "\n",
+    "df_summary = %sql SELECT * FROM iris_multi_model_summary;\n",
+    "df_summary = df_summary.DataFrame()\n",
+    "\n",
+    "#set up plots\n",
+    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
+    "fig.legend(ncol=4)\n",
+    "fig.tight_layout()\n",
+    "\n",
+    "ax_metric = axs[0]\n",
+    "ax_loss = axs[1]\n",
+    "\n",
+    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_metric.set_xlabel('Iteration')\n",
+    "ax_metric.set_ylabel('Metric')\n",
+    "ax_metric.set_title('Validation metric curve')\n",
+    "\n",
+    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
+    "ax_loss.set_xlabel('Iteration')\n",
+    "ax_loss.set_ylabel('Loss')\n",
+    "ax_loss.set_title('Validation loss curve')\n",
+    "\n",
+    "iters = df_summary['metrics_iters'][0]\n",
+    "\n",
+    "for mst_key in df_results['mst_key']:\n",
+    "    df_output_info = %sql SELECT validation_metrics,validation_loss FROM iris_multi_model_info WHERE mst_key = $mst_key\n",
+    "    df_output_info = df_output_info.DataFrame()\n",
+    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
+    "    validation_loss = df_output_info['validation_loss'][0]\n",
+    "    \n",
+    "    ax_metric.plot(iters, validation_metrics, label=mst_key, marker='o')\n",
+    "    ax_loss.plot(iters, validation_loss, label=mst_key, marker='o')\n",
+    "\n",
+    "plt.legend();\n",
+    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Train-single-model/.ipynb_checkpoints/MADlib-Keras-MLP-v2-checkpoint.ipynb b/community-artifacts/Deep-learning/Train-single-model/.ipynb_checkpoints/MADlib-Keras-MLP-v2-checkpoint.ipynb
new file mode 100644
index 0000000..8dfa6cd
--- /dev/null
+++ b/community-artifacts/Deep-learning/Train-single-model/.ipynb_checkpoints/MADlib-Keras-MLP-v2-checkpoint.ipynb
@@ -0,0 +1,5025 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Multilayer Perceptron Using Keras and MADlib\n",
+    "\n",
+    "E2E classification example using MADlib calling a Keras MLP.\n",
+    "\n",
+    "Deep learning works best on very large datasets, but that is not convenient for a quick introduction to the syntax.  So in this workbook we use the well known iris data set from https://archive.ics.uci.edu/ml/datasets/iris to help get you started.  It is similar to the example in user docs http://madlib.apache.org/docs/latest/index.html\n",
+    "\n",
+    "For more realistic examples with images please refer to the deep learning notebooks at\n",
+    "https://github.com/apache/madlib-site/tree/asf-site/community-artifacts\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#class\">Classification</a>\n",
+    "\n",
+    "* <a href=\"#create_input_data\">1. Create input data</a>\n",
+    "\n",
+    "* <a href=\"#pp\">2. Call preprocessor for deep learning</a>\n",
+    "\n",
+    "* <a href=\"#load\">3. Define and load model architecture</a>\n",
+    "\n",
+    "* <a href=\"#train\">4. Train</a>\n",
+    "\n",
+    "* <a href=\"#eval\">5. Evaluate</a>\n",
+    "\n",
+    "* <a href=\"#pred\">6. Predict</a>\n",
+    "\n",
+    "* <a href=\"#pred_byom\">7. Predict BYOM</a>\n",
+    "\n",
+    "<a href=\"#class2\">Classification with Other Parameters</a>\n",
+    "\n",
+    "* <a href=\"#val_dataset\">1. Validation dataset</a>\n",
+    "\n",
+    "* <a href=\"#pred_prob\">2. Predict probabilities</a>\n",
+    "\n",
+    "* <a href=\"#warm_start\">3. Warm start</a>\n",
+    "\n",
+    "<a href=\"#transfer_learn\">Transfer learning</a>\n",
+    "\n",
+    "* <a href=\"#load2\">1. Define and load model architecture with some layers frozen</a>\n",
+    "\n",
+    "* <a href=\"#train2\">2. Train transfer model</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class\"></a>\n",
+    "# Classification\n",
+    "\n",
+    "<a id=\"create_input_data\"></a>\n",
+    "# 1.  Create input data\n",
+    "\n",
+    "Load iris data set."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "150 rows affected.\n",
+      "150 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>attributes</th>\n",
+       "        <th>class_text</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>[Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>21</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>22</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>25</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>35</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>36</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>37</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>39</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>40</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>42</td>\n",
+       "        <td>[Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>44</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>45</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>46</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>47</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>48</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>[Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>50</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>[Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>53</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>58</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>59</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>62</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>65</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>69</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>79</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>86</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>87</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>88</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>91</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>95</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>97</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>[Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>105</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>106</td>\n",
+       "        <td>[Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>[Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>109</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>110</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>118</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>124</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>129</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>130</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>[Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>[Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>133</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>139</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>140</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>141</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>142</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (2, [Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (3, [Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (4, [Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (5, [Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (6, [Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (7, [Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (8, [Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (9, [Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (10, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (11, [Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (12, [Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (13, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (14, [Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (15, [Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (16, [Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (17, [Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (18, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (19, [Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (20, [Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (21, [Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (22, [Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (23, [Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (24, [Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')], u'Iris-setosa'),\n",
+       " (25, [Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (26, [Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (27, [Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (28, [Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (29, [Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (30, [Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (31, [Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (32, [Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (33, [Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (34, [Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (35, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (36, [Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (37, [Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (38, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (39, [Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (40, [Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (41, [Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (42, [Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (43, [Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (44, [Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')], u'Iris-setosa'),\n",
+       " (45, [Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (46, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (47, [Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (48, [Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (49, [Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (50, [Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (51, [Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (52, [Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (53, [Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (54, [Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (55, [Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (56, [Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (57, [Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (58, [Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (59, [Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (60, [Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (61, [Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (62, [Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (63, [Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (64, [Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (65, [Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (66, [Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (67, [Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (68, [Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (69, [Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (70, [Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (71, [Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')], u'Iris-versicolor'),\n",
+       " (72, [Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (73, [Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (74, [Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (75, [Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (76, [Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (77, [Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (78, [Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')], u'Iris-versicolor'),\n",
+       " (79, [Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (80, [Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (81, [Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (82, [Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (83, [Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (84, [Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (85, [Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (86, [Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (87, [Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (88, [Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (89, [Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (90, [Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (91, [Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (92, [Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (93, [Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (94, [Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (95, [Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (96, [Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (97, [Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (98, [Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (99, [Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (100, [Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (101, [Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (102, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (103, [Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (104, [Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (105, [Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (106, [Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (107, [Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')], u'Iris-virginica'),\n",
+       " (108, [Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (109, [Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (110, [Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (111, [Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (112, [Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (113, [Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (114, [Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (115, [Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (116, [Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (117, [Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (118, [Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (119, [Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (120, [Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (121, [Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (122, [Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (123, [Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (124, [Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (125, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (126, [Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (127, [Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (128, [Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (129, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (130, [Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')], u'Iris-virginica'),\n",
+       " (131, [Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (132, [Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (133, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (134, [Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (135, [Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')], u'Iris-virginica'),\n",
+       " (136, [Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (137, [Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (138, [Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (139, [Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (140, [Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (141, [Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (142, [Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (143, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (144, [Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (145, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (146, [Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (147, [Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (148, [Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (149, [Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (150, [Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')], u'Iris-virginica')]"
+      ]
+     },
+     "execution_count": 4,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql \n",
+    "DROP TABLE IF EXISTS iris_data;\n",
+    "\n",
+    "CREATE TABLE iris_data(\n",
+    "    id serial,\n",
+    "    attributes numeric[],\n",
+    "    class_text varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO iris_data(id, attributes, class_text) VALUES\n",
+    "(1,ARRAY[5.1,3.5,1.4,0.2],'Iris-setosa'),\n",
+    "(2,ARRAY[4.9,3.0,1.4,0.2],'Iris-setosa'),\n",
+    "(3,ARRAY[4.7,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(4,ARRAY[4.6,3.1,1.5,0.2],'Iris-setosa'),\n",
+    "(5,ARRAY[5.0,3.6,1.4,0.2],'Iris-setosa'),\n",
+    "(6,ARRAY[5.4,3.9,1.7,0.4],'Iris-setosa'),\n",
+    "(7,ARRAY[4.6,3.4,1.4,0.3],'Iris-setosa'),\n",
+    "(8,ARRAY[5.0,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(9,ARRAY[4.4,2.9,1.4,0.2],'Iris-setosa'),\n",
+    "(10,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(11,ARRAY[5.4,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(12,ARRAY[4.8,3.4,1.6,0.2],'Iris-setosa'),\n",
+    "(13,ARRAY[4.8,3.0,1.4,0.1],'Iris-setosa'),\n",
+    "(14,ARRAY[4.3,3.0,1.1,0.1],'Iris-setosa'),\n",
+    "(15,ARRAY[5.8,4.0,1.2,0.2],'Iris-setosa'),\n",
+    "(16,ARRAY[5.7,4.4,1.5,0.4],'Iris-setosa'),\n",
+    "(17,ARRAY[5.4,3.9,1.3,0.4],'Iris-setosa'),\n",
+    "(18,ARRAY[5.1,3.5,1.4,0.3],'Iris-setosa'),\n",
+    "(19,ARRAY[5.7,3.8,1.7,0.3],'Iris-setosa'),\n",
+    "(20,ARRAY[5.1,3.8,1.5,0.3],'Iris-setosa'),\n",
+    "(21,ARRAY[5.4,3.4,1.7,0.2],'Iris-setosa'),\n",
+    "(22,ARRAY[5.1,3.7,1.5,0.4],'Iris-setosa'),\n",
+    "(23,ARRAY[4.6,3.6,1.0,0.2],'Iris-setosa'),\n",
+    "(24,ARRAY[5.1,3.3,1.7,0.5],'Iris-setosa'),\n",
+    "(25,ARRAY[4.8,3.4,1.9,0.2],'Iris-setosa'),\n",
+    "(26,ARRAY[5.0,3.0,1.6,0.2],'Iris-setosa'),\n",
+    "(27,ARRAY[5.0,3.4,1.6,0.4],'Iris-setosa'),\n",
+    "(28,ARRAY[5.2,3.5,1.5,0.2],'Iris-setosa'),\n",
+    "(29,ARRAY[5.2,3.4,1.4,0.2],'Iris-setosa'),\n",
+    "(30,ARRAY[4.7,3.2,1.6,0.2],'Iris-setosa'),\n",
+    "(31,ARRAY[4.8,3.1,1.6,0.2],'Iris-setosa'),\n",
+    "(32,ARRAY[5.4,3.4,1.5,0.4],'Iris-setosa'),\n",
+    "(33,ARRAY[5.2,4.1,1.5,0.1],'Iris-setosa'),\n",
+    "(34,ARRAY[5.5,4.2,1.4,0.2],'Iris-setosa'),\n",
+    "(35,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(36,ARRAY[5.0,3.2,1.2,0.2],'Iris-setosa'),\n",
+    "(37,ARRAY[5.5,3.5,1.3,0.2],'Iris-setosa'),\n",
+    "(38,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(39,ARRAY[4.4,3.0,1.3,0.2],'Iris-setosa'),\n",
+    "(40,ARRAY[5.1,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(41,ARRAY[5.0,3.5,1.3,0.3],'Iris-setosa'),\n",
+    "(42,ARRAY[4.5,2.3,1.3,0.3],'Iris-setosa'),\n",
+    "(43,ARRAY[4.4,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(44,ARRAY[5.0,3.5,1.6,0.6],'Iris-setosa'),\n",
+    "(45,ARRAY[5.1,3.8,1.9,0.4],'Iris-setosa'),\n",
+    "(46,ARRAY[4.8,3.0,1.4,0.3],'Iris-setosa'),\n",
+    "(47,ARRAY[5.1,3.8,1.6,0.2],'Iris-setosa'),\n",
+    "(48,ARRAY[4.6,3.2,1.4,0.2],'Iris-setosa'),\n",
+    "(49,ARRAY[5.3,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(50,ARRAY[5.0,3.3,1.4,0.2],'Iris-setosa'),\n",
+    "(51,ARRAY[7.0,3.2,4.7,1.4],'Iris-versicolor'),\n",
+    "(52,ARRAY[6.4,3.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(53,ARRAY[6.9,3.1,4.9,1.5],'Iris-versicolor'),\n",
+    "(54,ARRAY[5.5,2.3,4.0,1.3],'Iris-versicolor'),\n",
+    "(55,ARRAY[6.5,2.8,4.6,1.5],'Iris-versicolor'),\n",
+    "(56,ARRAY[5.7,2.8,4.5,1.3],'Iris-versicolor'),\n",
+    "(57,ARRAY[6.3,3.3,4.7,1.6],'Iris-versicolor'),\n",
+    "(58,ARRAY[4.9,2.4,3.3,1.0],'Iris-versicolor'),\n",
+    "(59,ARRAY[6.6,2.9,4.6,1.3],'Iris-versicolor'),\n",
+    "(60,ARRAY[5.2,2.7,3.9,1.4],'Iris-versicolor'),\n",
+    "(61,ARRAY[5.0,2.0,3.5,1.0],'Iris-versicolor'),\n",
+    "(62,ARRAY[5.9,3.0,4.2,1.5],'Iris-versicolor'),\n",
+    "(63,ARRAY[6.0,2.2,4.0,1.0],'Iris-versicolor'),\n",
+    "(64,ARRAY[6.1,2.9,4.7,1.4],'Iris-versicolor'),\n",
+    "(65,ARRAY[5.6,2.9,3.6,1.3],'Iris-versicolor'),\n",
+    "(66,ARRAY[6.7,3.1,4.4,1.4],'Iris-versicolor'),\n",
+    "(67,ARRAY[5.6,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(68,ARRAY[5.8,2.7,4.1,1.0],'Iris-versicolor'),\n",
+    "(69,ARRAY[6.2,2.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(70,ARRAY[5.6,2.5,3.9,1.1],'Iris-versicolor'),\n",
+    "(71,ARRAY[5.9,3.2,4.8,1.8],'Iris-versicolor'),\n",
+    "(72,ARRAY[6.1,2.8,4.0,1.3],'Iris-versicolor'),\n",
+    "(73,ARRAY[6.3,2.5,4.9,1.5],'Iris-versicolor'),\n",
+    "(74,ARRAY[6.1,2.8,4.7,1.2],'Iris-versicolor'),\n",
+    "(75,ARRAY[6.4,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(76,ARRAY[6.6,3.0,4.4,1.4],'Iris-versicolor'),\n",
+    "(77,ARRAY[6.8,2.8,4.8,1.4],'Iris-versicolor'),\n",
+    "(78,ARRAY[6.7,3.0,5.0,1.7],'Iris-versicolor'),\n",
+    "(79,ARRAY[6.0,2.9,4.5,1.5],'Iris-versicolor'),\n",
+    "(80,ARRAY[5.7,2.6,3.5,1.0],'Iris-versicolor'),\n",
+    "(81,ARRAY[5.5,2.4,3.8,1.1],'Iris-versicolor'),\n",
+    "(82,ARRAY[5.5,2.4,3.7,1.0],'Iris-versicolor'),\n",
+    "(83,ARRAY[5.8,2.7,3.9,1.2],'Iris-versicolor'),\n",
+    "(84,ARRAY[6.0,2.7,5.1,1.6],'Iris-versicolor'),\n",
+    "(85,ARRAY[5.4,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(86,ARRAY[6.0,3.4,4.5,1.6],'Iris-versicolor'),\n",
+    "(87,ARRAY[6.7,3.1,4.7,1.5],'Iris-versicolor'),\n",
+    "(88,ARRAY[6.3,2.3,4.4,1.3],'Iris-versicolor'),\n",
+    "(89,ARRAY[5.6,3.0,4.1,1.3],'Iris-versicolor'),\n",
+    "(90,ARRAY[5.5,2.5,4.0,1.3],'Iris-versicolor'),\n",
+    "(91,ARRAY[5.5,2.6,4.4,1.2],'Iris-versicolor'),\n",
+    "(92,ARRAY[6.1,3.0,4.6,1.4],'Iris-versicolor'),\n",
+    "(93,ARRAY[5.8,2.6,4.0,1.2],'Iris-versicolor'),\n",
+    "(94,ARRAY[5.0,2.3,3.3,1.0],'Iris-versicolor'),\n",
+    "(95,ARRAY[5.6,2.7,4.2,1.3],'Iris-versicolor'),\n",
+    "(96,ARRAY[5.7,3.0,4.2,1.2],'Iris-versicolor'),\n",
+    "(97,ARRAY[5.7,2.9,4.2,1.3],'Iris-versicolor'),\n",
+    "(98,ARRAY[6.2,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(99,ARRAY[5.1,2.5,3.0,1.1],'Iris-versicolor'),\n",
+    "(100,ARRAY[5.7,2.8,4.1,1.3],'Iris-versicolor'),\n",
+    "(101,ARRAY[6.3,3.3,6.0,2.5],'Iris-virginica'),\n",
+    "(102,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(103,ARRAY[7.1,3.0,5.9,2.1],'Iris-virginica'),\n",
+    "(104,ARRAY[6.3,2.9,5.6,1.8],'Iris-virginica'),\n",
+    "(105,ARRAY[6.5,3.0,5.8,2.2],'Iris-virginica'),\n",
+    "(106,ARRAY[7.6,3.0,6.6,2.1],'Iris-virginica'),\n",
+    "(107,ARRAY[4.9,2.5,4.5,1.7],'Iris-virginica'),\n",
+    "(108,ARRAY[7.3,2.9,6.3,1.8],'Iris-virginica'),\n",
+    "(109,ARRAY[6.7,2.5,5.8,1.8],'Iris-virginica'),\n",
+    "(110,ARRAY[7.2,3.6,6.1,2.5],'Iris-virginica'),\n",
+    "(111,ARRAY[6.5,3.2,5.1,2.0],'Iris-virginica'),\n",
+    "(112,ARRAY[6.4,2.7,5.3,1.9],'Iris-virginica'),\n",
+    "(113,ARRAY[6.8,3.0,5.5,2.1],'Iris-virginica'),\n",
+    "(114,ARRAY[5.7,2.5,5.0,2.0],'Iris-virginica'),\n",
+    "(115,ARRAY[5.8,2.8,5.1,2.4],'Iris-virginica'),\n",
+    "(116,ARRAY[6.4,3.2,5.3,2.3],'Iris-virginica'),\n",
+    "(117,ARRAY[6.5,3.0,5.5,1.8],'Iris-virginica'),\n",
+    "(118,ARRAY[7.7,3.8,6.7,2.2],'Iris-virginica'),\n",
+    "(119,ARRAY[7.7,2.6,6.9,2.3],'Iris-virginica'),\n",
+    "(120,ARRAY[6.0,2.2,5.0,1.5],'Iris-virginica'),\n",
+    "(121,ARRAY[6.9,3.2,5.7,2.3],'Iris-virginica'),\n",
+    "(122,ARRAY[5.6,2.8,4.9,2.0],'Iris-virginica'),\n",
+    "(123,ARRAY[7.7,2.8,6.7,2.0],'Iris-virginica'),\n",
+    "(124,ARRAY[6.3,2.7,4.9,1.8],'Iris-virginica'),\n",
+    "(125,ARRAY[6.7,3.3,5.7,2.1],'Iris-virginica'),\n",
+    "(126,ARRAY[7.2,3.2,6.0,1.8],'Iris-virginica'),\n",
+    "(127,ARRAY[6.2,2.8,4.8,1.8],'Iris-virginica'),\n",
+    "(128,ARRAY[6.1,3.0,4.9,1.8],'Iris-virginica'),\n",
+    "(129,ARRAY[6.4,2.8,5.6,2.1],'Iris-virginica'),\n",
+    "(130,ARRAY[7.2,3.0,5.8,1.6],'Iris-virginica'),\n",
+    "(131,ARRAY[7.4,2.8,6.1,1.9],'Iris-virginica'),\n",
+    "(132,ARRAY[7.9,3.8,6.4,2.0],'Iris-virginica'),\n",
+    "(133,ARRAY[6.4,2.8,5.6,2.2],'Iris-virginica'),\n",
+    "(134,ARRAY[6.3,2.8,5.1,1.5],'Iris-virginica'),\n",
+    "(135,ARRAY[6.1,2.6,5.6,1.4],'Iris-virginica'),\n",
+    "(136,ARRAY[7.7,3.0,6.1,2.3],'Iris-virginica'),\n",
+    "(137,ARRAY[6.3,3.4,5.6,2.4],'Iris-virginica'),\n",
+    "(138,ARRAY[6.4,3.1,5.5,1.8],'Iris-virginica'),\n",
+    "(139,ARRAY[6.0,3.0,4.8,1.8],'Iris-virginica'),\n",
+    "(140,ARRAY[6.9,3.1,5.4,2.1],'Iris-virginica'),\n",
+    "(141,ARRAY[6.7,3.1,5.6,2.4],'Iris-virginica'),\n",
+    "(142,ARRAY[6.9,3.1,5.1,2.3],'Iris-virginica'),\n",
+    "(143,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(144,ARRAY[6.8,3.2,5.9,2.3],'Iris-virginica'),\n",
+    "(145,ARRAY[6.7,3.3,5.7,2.5],'Iris-virginica'),\n",
+    "(146,ARRAY[6.7,3.0,5.2,2.3],'Iris-virginica'),\n",
+    "(147,ARRAY[6.3,2.5,5.0,1.9],'Iris-virginica'),\n",
+    "(148,ARRAY[6.5,3.0,5.2,2.0],'Iris-virginica'),\n",
+    "(149,ARRAY[6.2,3.4,5.4,2.3],'Iris-virginica'),\n",
+    "(150,ARRAY[5.9,3.0,5.1,1.8],'Iris-virginica');\n",
+    "\n",
+    "SELECT * FROM iris_data ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Create a test/validation dataset from the training data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(120L,)]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train, iris_test;\n",
+    "\n",
+    "-- Set seed so results are reproducible\n",
+    "SELECT setseed(0);\n",
+    "\n",
+    "SELECT madlib.train_test_split('iris_data',     -- Source table\n",
+    "                               'iris',          -- Output table root name\n",
+    "                                0.8,            -- Train proportion\n",
+    "                                NULL,           -- Test proportion (0.2)\n",
+    "                                NULL,           -- Strata definition\n",
+    "                                NULL,           -- Output all columns\n",
+    "                                NULL,           -- Sample without replacement\n",
+    "                                TRUE            -- Separate output tables\n",
+    "                              );\n",
+    "\n",
+    "SELECT COUNT(*) FROM iris_train;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pp\"></a>\n",
+    "# 2. Call preprocessor for deep learning\n",
+    "Training dataset (uses training preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train</td>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>60</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train', u'iris_train_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 60, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train_packed, iris_train_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('iris_train',         -- Source table\n",
+    "                                       'iris_train_packed',  -- Output table\n",
+    "                                       'class_text',         -- Dependent variable\n",
+    "                                       'attributes'          -- Independent variable\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM iris_train_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Validation dataset (uses validation preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_test</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>15</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_test', u'iris_test_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 15, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_test_packed, iris_test_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('iris_test',          -- Source table\n",
+    "                                         'iris_test_packed',   -- Output table\n",
+    "                                         'class_text',         -- Dependent variable\n",
+    "                                         'attributes',         -- Independent variable\n",
+    "                                         'iris_train_packed'   -- From training preprocessor step\n",
+    "                                          ); \n",
+    "\n",
+    "SELECT * FROM iris_test_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load\"></a>\n",
+    "# 3. Define and load model architecture\n",
+    "Import Keras libraries"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "WARNING:tensorflow:From /Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/tensorflow/python/ops/init_ops.py:1251: calling __init__ (from tensorflow.python.ops.init_ops) with dtype is deprecated and will be removed in a future version.\n",
+      "Instructions for updating:\n",
+      "Call initializer instance with the dtype argument instead of passing it to the constructor\n",
+      "Model: \"sequential\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense (Dense)                (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_1 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_2 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 193\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model_simple = Sequential()\n",
+    "model_simple.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model_simple.add(Dense(10, activation='relu'))\n",
+    "model_simple.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model_simple.summary();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model_simple.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Sophie', u'A simple model')]"
+      ]
+     },
+     "execution_count": 11,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS model_arch_library;\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Sophie',              -- Name\n",
+    "                               'A simple model'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT model_id, model_arch, name, description FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train\"></a>\n",
+    "# 4.  Train\n",
+    "Train the model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 12,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 12,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10                    -- num_iterations\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 13,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>1</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>10</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-06 00:27:28.144705</td>\n",
+       "        <td>2021-03-06 00:27:31.754147</td>\n",
+       "        <td>[3.60936093330383]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.916666686535</td>\n",
+       "        <td>0.463008254766</td>\n",
+       "        <td>[0.916666686534882]</td>\n",
+       "        <td>[0.463008254766464]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>[10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, None, None, 10, None, None, u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 6, 0, 27, 28, 144705), datetime.datetime(2021, 3, 6, 0, 27, 31, 754147), [3.60936093330383], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.916666686534882, 0.463008254766464, [0.916666686534882], [0.463008254766464], None, None, None, None, [10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 13,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"eval\"></a>\n",
+    "# 5. Evaluate\n",
+    "\n",
+    "Now run evaluate using model we built above:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 14,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>loss</th>\n",
+       "        <th>metric</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>0.523572981358</td>\n",
+       "        <td>0.933333337307</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(0.523572981357574, 0.933333337306976, [u'accuracy'], u'categorical_crossentropy')]"
+      ]
+     },
+     "execution_count": 14,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_validate;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_evaluate('iris_model',       -- model\n",
+    "                                   'iris_test_packed',  -- test table\n",
+    "                                   'iris_validate'      -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_validate;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred\"></a>\n",
+    "# 6. Predict\n",
+    "\n",
+    "Now predict using model we built.  We will use the validation data set for prediction as well, which is not usual but serves to show the syntax. The prediction is in the estimated_class_text column:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 15,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "90 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.83670896</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.14060013</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.022690918</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8369735</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.14013577</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.022890732</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.87973696</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.10638312</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.013879963</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.93740743</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.056862056</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0057305074</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.83670896</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.14060013</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.022690918</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8709096</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.11054307</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.018547323</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.4681935</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4571225</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.07468399</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.45466852</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.4470526</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.09827888</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.47486252</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.46100235</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.064135045</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.47181308</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.43595785</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.09222904</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.47956672</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.41212082</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.10831244</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.50861007</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3626588</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.12873109</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5061021</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3914343</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.102463536</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.49345753</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.35755217</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.14899038</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4796765</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.4385325</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0817909</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.47809058</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.34930265</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.17260681</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6172143</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3620455</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.020740215</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5837618</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3847274</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.03151086</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.61951214</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3637118</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.016776035</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5954762</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.37995332</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.024570476</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.571379</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4039808</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.024640195</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.57040656</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3980587</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.03153468</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.52341586</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.43971062</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.036873452</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5800313</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3929817</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.026986998</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.72622484</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.26773784</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0060372944</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5089497</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.44541556</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.045634773</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.62922823</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.35819018</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.012581516</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6017383</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3781529</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.020108894</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5293082</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4390557</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.031636048</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.58249867</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.39045528</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.027046034</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(10, u'class_text', u'Iris-setosa', 0.83670896, 1),\n",
+       " (10, u'class_text', u'Iris-versicolor', 0.14060013, 2),\n",
+       " (10, u'class_text', u'Iris-virginica', 0.022690918, 3),\n",
+       " (13, u'class_text', u'Iris-setosa', 0.8369735, 1),\n",
+       " (13, u'class_text', u'Iris-versicolor', 0.14013577, 2),\n",
+       " (13, u'class_text', u'Iris-virginica', 0.022890732, 3),\n",
+       " (29, u'class_text', u'Iris-setosa', 0.87973696, 1),\n",
+       " (29, u'class_text', u'Iris-versicolor', 0.10638312, 2),\n",
+       " (29, u'class_text', u'Iris-virginica', 0.013879963, 3),\n",
+       " (34, u'class_text', u'Iris-setosa', 0.93740743, 1),\n",
+       " (34, u'class_text', u'Iris-versicolor', 0.056862056, 2),\n",
+       " (34, u'class_text', u'Iris-virginica', 0.0057305074, 3),\n",
+       " (38, u'class_text', u'Iris-setosa', 0.83670896, 1),\n",
+       " (38, u'class_text', u'Iris-versicolor', 0.14060013, 2),\n",
+       " (38, u'class_text', u'Iris-virginica', 0.022690918, 3),\n",
+       " (43, u'class_text', u'Iris-setosa', 0.8709096, 1),\n",
+       " (43, u'class_text', u'Iris-versicolor', 0.11054307, 2),\n",
+       " (43, u'class_text', u'Iris-virginica', 0.018547323, 3),\n",
+       " (56, u'class_text', u'Iris-virginica', 0.4681935, 1),\n",
+       " (56, u'class_text', u'Iris-versicolor', 0.4571225, 2),\n",
+       " (56, u'class_text', u'Iris-setosa', 0.07468399, 3),\n",
+       " (61, u'class_text', u'Iris-versicolor', 0.45466852, 1),\n",
+       " (61, u'class_text', u'Iris-virginica', 0.4470526, 2),\n",
+       " (61, u'class_text', u'Iris-setosa', 0.09827888, 3),\n",
+       " (64, u'class_text', u'Iris-virginica', 0.47486252, 1),\n",
+       " (64, u'class_text', u'Iris-versicolor', 0.46100235, 2),\n",
+       " (64, u'class_text', u'Iris-setosa', 0.064135045, 3),\n",
+       " (67, u'class_text', u'Iris-versicolor', 0.47181308, 1),\n",
+       " (67, u'class_text', u'Iris-virginica', 0.43595785, 2),\n",
+       " (67, u'class_text', u'Iris-setosa', 0.09222904, 3),\n",
+       " (70, u'class_text', u'Iris-versicolor', 0.47956672, 1),\n",
+       " (70, u'class_text', u'Iris-virginica', 0.41212082, 2),\n",
+       " (70, u'class_text', u'Iris-setosa', 0.10831244, 3),\n",
+       " (72, u'class_text', u'Iris-versicolor', 0.50861007, 1),\n",
+       " (72, u'class_text', u'Iris-virginica', 0.3626588, 2),\n",
+       " (72, u'class_text', u'Iris-setosa', 0.12873109, 3),\n",
+       " (75, u'class_text', u'Iris-versicolor', 0.5061021, 1),\n",
+       " (75, u'class_text', u'Iris-virginica', 0.3914343, 2),\n",
+       " (75, u'class_text', u'Iris-setosa', 0.102463536, 3),\n",
+       " (89, u'class_text', u'Iris-versicolor', 0.49345753, 1),\n",
+       " (89, u'class_text', u'Iris-virginica', 0.35755217, 2),\n",
+       " (89, u'class_text', u'Iris-setosa', 0.14899038, 3),\n",
+       " (92, u'class_text', u'Iris-versicolor', 0.4796765, 1),\n",
+       " (92, u'class_text', u'Iris-virginica', 0.4385325, 2),\n",
+       " (92, u'class_text', u'Iris-setosa', 0.0817909, 3),\n",
+       " (94, u'class_text', u'Iris-versicolor', 0.47809058, 1),\n",
+       " (94, u'class_text', u'Iris-virginica', 0.34930265, 2),\n",
+       " (94, u'class_text', u'Iris-setosa', 0.17260681, 3),\n",
+       " (101, u'class_text', u'Iris-virginica', 0.6172143, 1),\n",
+       " (101, u'class_text', u'Iris-versicolor', 0.3620455, 2),\n",
+       " (101, u'class_text', u'Iris-setosa', 0.020740215, 3),\n",
+       " (102, u'class_text', u'Iris-virginica', 0.5837618, 1),\n",
+       " (102, u'class_text', u'Iris-versicolor', 0.3847274, 2),\n",
+       " (102, u'class_text', u'Iris-setosa', 0.03151086, 3),\n",
+       " (103, u'class_text', u'Iris-virginica', 0.61951214, 1),\n",
+       " (103, u'class_text', u'Iris-versicolor', 0.3637118, 2),\n",
+       " (103, u'class_text', u'Iris-setosa', 0.016776035, 3),\n",
+       " (112, u'class_text', u'Iris-virginica', 0.5954762, 1),\n",
+       " (112, u'class_text', u'Iris-versicolor', 0.37995332, 2),\n",
+       " (112, u'class_text', u'Iris-setosa', 0.024570476, 3),\n",
+       " (113, u'class_text', u'Iris-virginica', 0.571379, 1),\n",
+       " (113, u'class_text', u'Iris-versicolor', 0.4039808, 2),\n",
+       " (113, u'class_text', u'Iris-setosa', 0.024640195, 3),\n",
+       " (115, u'class_text', u'Iris-virginica', 0.57040656, 1),\n",
+       " (115, u'class_text', u'Iris-versicolor', 0.3980587, 2),\n",
+       " (115, u'class_text', u'Iris-setosa', 0.03153468, 3),\n",
+       " (116, u'class_text', u'Iris-virginica', 0.52341586, 1),\n",
+       " (116, u'class_text', u'Iris-versicolor', 0.43971062, 2),\n",
+       " (116, u'class_text', u'Iris-setosa', 0.036873452, 3),\n",
+       " (117, u'class_text', u'Iris-virginica', 0.5800313, 1),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.3929817, 2),\n",
+       " (117, u'class_text', u'Iris-setosa', 0.026986998, 3),\n",
+       " (119, u'class_text', u'Iris-virginica', 0.72622484, 1),\n",
+       " (119, u'class_text', u'Iris-versicolor', 0.26773784, 2),\n",
+       " (119, u'class_text', u'Iris-setosa', 0.0060372944, 3),\n",
+       " (127, u'class_text', u'Iris-virginica', 0.5089497, 1),\n",
+       " (127, u'class_text', u'Iris-versicolor', 0.44541556, 2),\n",
+       " (127, u'class_text', u'Iris-setosa', 0.045634773, 3),\n",
+       " (136, u'class_text', u'Iris-virginica', 0.62922823, 1),\n",
+       " (136, u'class_text', u'Iris-versicolor', 0.35819018, 2),\n",
+       " (136, u'class_text', u'Iris-setosa', 0.012581516, 3),\n",
+       " (144, u'class_text', u'Iris-virginica', 0.6017383, 1),\n",
+       " (144, u'class_text', u'Iris-versicolor', 0.3781529, 2),\n",
+       " (144, u'class_text', u'Iris-setosa', 0.020108894, 3),\n",
+       " (146, u'class_text', u'Iris-virginica', 0.5293082, 1),\n",
+       " (146, u'class_text', u'Iris-versicolor', 0.4390557, 2),\n",
+       " (146, u'class_text', u'Iris-setosa', 0.031636048, 3),\n",
+       " (147, u'class_text', u'Iris-virginica', 0.58249867, 1),\n",
+       " (147, u'class_text', u'Iris-versicolor', 0.39045528, 2),\n",
+       " (147, u'class_text', u'Iris-setosa', 0.027046034, 3)]"
+      ]
+     },
+     "execution_count": 15,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_model', -- model\n",
+    "                                   'iris_test',  -- test_table\n",
+    "                                   'id',  -- id column\n",
+    "                                   'attributes', -- independent var\n",
+    "                                   'iris_predict'  -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id, rank;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 16,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2L,)]"
+      ]
+     },
+     "execution_count": 16,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict JOIN iris_test USING (id)\n",
+    "WHERE iris_predict.class_value != iris_test.class_text AND iris_predict.rank = 1;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 17,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93.33</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('93.33'),)]"
+      ]
+     },
+     "execution_count": 17,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict.class_value as estimated\n",
+    "     from iris_predict inner join iris_test\n",
+    "     on iris_test.id=iris_predict.id where iris_predict.rank = 1) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred_byom\"></a>\n",
+    "# 7. Predict BYOM\n",
+    "The predict BYOM function allows you to do inference on models that have not been trained on MADlib, but rather imported from elsewhere.  \n",
+    "\n",
+    "We will use the validation dataset for prediction as well, which is not usual but serves to show the syntax.\n",
+    "\n",
+    "See load_keras_model()\n",
+    "http://madlib.apache.org/docs/latest/group__grp__keras__model__arch.html\n",
+    "for details on how to load the model architecture and weights.  In this example we will use weights we already have:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 18,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 18,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "UPDATE model_arch_library \n",
+    "SET model_weights = iris_model.model_weights \n",
+    "FROM iris_model \n",
+    "WHERE model_arch_library.model_id = 1;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Now train using a model from the model architecture table directly without referencing the model table from the MADlib training.  \n",
+    "\n",
+    "Note that if you specify the class values parameter as we do below, it must reflect how the dependent variable was 1-hot encoded for training.  In this example the 'training_preprocessor_dl()' in Step 2 above encoded in the order {'Iris-setosa', 'Iris-versicolor', 'Iris-virginica'} so this is the order we pass in the parameter.  If we accidently picked another order that did not match the 1-hot encoding, the predictions would be wrong."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 19,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "30 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.83670896</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8369735</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.87973696</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.93740743</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.83670896</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8709096</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.4681935</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.45466852</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.47486252</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.47181308</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.47956672</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.50861007</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5061021</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.49345753</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4796765</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.47809058</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6172143</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5837618</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.61951214</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5954762</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.571379</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.57040656</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.52341586</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5800313</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.72622484</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5089497</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.62922823</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6017383</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5293082</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.58249867</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(10, u'dependent_var', u'Iris-setosa', 0.83670896),\n",
+       " (13, u'dependent_var', u'Iris-setosa', 0.8369735),\n",
+       " (29, u'dependent_var', u'Iris-setosa', 0.87973696),\n",
+       " (34, u'dependent_var', u'Iris-setosa', 0.93740743),\n",
+       " (38, u'dependent_var', u'Iris-setosa', 0.83670896),\n",
+       " (43, u'dependent_var', u'Iris-setosa', 0.8709096),\n",
+       " (56, u'dependent_var', u'Iris-virginica', 0.4681935),\n",
+       " (61, u'dependent_var', u'Iris-versicolor', 0.45466852),\n",
+       " (64, u'dependent_var', u'Iris-virginica', 0.47486252),\n",
+       " (67, u'dependent_var', u'Iris-versicolor', 0.47181308),\n",
+       " (70, u'dependent_var', u'Iris-versicolor', 0.47956672),\n",
+       " (72, u'dependent_var', u'Iris-versicolor', 0.50861007),\n",
+       " (75, u'dependent_var', u'Iris-versicolor', 0.5061021),\n",
+       " (89, u'dependent_var', u'Iris-versicolor', 0.49345753),\n",
+       " (92, u'dependent_var', u'Iris-versicolor', 0.4796765),\n",
+       " (94, u'dependent_var', u'Iris-versicolor', 0.47809058),\n",
+       " (101, u'dependent_var', u'Iris-virginica', 0.6172143),\n",
+       " (102, u'dependent_var', u'Iris-virginica', 0.5837618),\n",
+       " (103, u'dependent_var', u'Iris-virginica', 0.61951214),\n",
+       " (112, u'dependent_var', u'Iris-virginica', 0.5954762),\n",
+       " (113, u'dependent_var', u'Iris-virginica', 0.571379),\n",
+       " (115, u'dependent_var', u'Iris-virginica', 0.57040656),\n",
+       " (116, u'dependent_var', u'Iris-virginica', 0.52341586),\n",
+       " (117, u'dependent_var', u'Iris-virginica', 0.5800313),\n",
+       " (119, u'dependent_var', u'Iris-virginica', 0.72622484),\n",
+       " (127, u'dependent_var', u'Iris-virginica', 0.5089497),\n",
+       " (136, u'dependent_var', u'Iris-virginica', 0.62922823),\n",
+       " (144, u'dependent_var', u'Iris-virginica', 0.6017383),\n",
+       " (146, u'dependent_var', u'Iris-virginica', 0.5293082),\n",
+       " (147, u'dependent_var', u'Iris-virginica', 0.58249867)]"
+      ]
+     },
+     "execution_count": 19,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict_byom;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict_byom('model_arch_library',  -- model arch table\n",
+    "                                         1,                    -- model arch id\n",
+    "                                        'iris_test',           -- test_table\n",
+    "                                        'id',                  -- id column\n",
+    "                                        'attributes',          -- independent var\n",
+    "                                        'iris_predict_byom',   -- output table\n",
+    "                                        'response',            -- prediction type\n",
+    "                                         FALSE,                -- use GPUs\n",
+    "                                         ARRAY[ARRAY['Iris-setosa', 'Iris-versicolor', 'Iris-virginica']], -- class values\n",
+    "                                         1.0                   -- normalizing const\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict_byom ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 20,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2L,)]"
+      ]
+     },
+     "execution_count": 20,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict_byom JOIN iris_test USING (id)\n",
+    "WHERE iris_predict_byom.class_value != iris_test.class_text;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 21,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93.33</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('93.33'),)]"
+      ]
+     },
+     "execution_count": 21,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict_byom.class_value as estimated\n",
+    "     from iris_predict_byom inner join iris_test\n",
+    "     on iris_test.id=iris_predict_byom.id) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class2\"></a>\n",
+    "# Classification with Other Parameters\n",
+    "\n",
+    "<a id=\"val_dataset\"></a>\n",
+    "# 1.  Validation dataset\n",
+    "Now use a validation dataset and compute metrics every 2nd iteration using the 'metrics_compute_frequency' parameter.  This can help reduce run time if you do not need metrics computed at every iteration."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 22,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 22,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10,                   -- num_iterations\n",
+    "                                FALSE,                -- use GPUs\n",
+    "                                'iris_test_packed',   -- validation dataset\n",
+    "                                2,                    -- metrics compute frequency\n",
+    "                                FALSE,                -- warm start\n",
+    "                               'Sophie L.',           -- name\n",
+    "                               'Simple MLP for iris dataset'  -- description\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 23,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>1</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Simple MLP for iris dataset</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-06 00:27:42.910502</td>\n",
+       "        <td>2021-03-06 00:27:44.171209</td>\n",
+       "        <td>[0.706467866897583, 0.850914001464844, 0.988704919815063, 1.12321996688843, 1.26061987876892]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.286152273417</td>\n",
+       "        <td>[0.933333337306976, 0.941666662693024, 0.941666662693024, 0.958333313465118, 0.966666638851166]</td>\n",
+       "        <td>[0.410510897636414, 0.371806919574738, 0.339208543300629, 0.310610443353653, 0.286152273416519]</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>0.312809795141</td>\n",
+       "        <td>[1.0, 1.0, 1.0, 1.0, 1.0]</td>\n",
+       "        <td>[0.478174388408661, 0.426770567893982, 0.391106754541397, 0.351149171590805, 0.31280979514122]</td>\n",
+       "        <td>[2, 4, 6, 8, 10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', None, 2, u'Sophie L.', u'Simple MLP for iris dataset', u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 6, 0, 27, 42, 910502), datetime.datetime(2021, 3, 6, 0, 27, 44, 171209), [0.706467866897583, 0.850914001464844, 0.988704919815063, 1.12321996688843, 1.26061987876892], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.966666638851166, 0.286152273416519, [0.933333337306976, 0.941666662693024, 0.941666662693024, 0.958333313465118, 0.966666638851166], [0.410510897636414, 0.371806919574738, 0.339208543300629, 0.310610443353653, 0.286152273416519], 1.0, 0.31280979514122, [1.0, 1.0, 1.0, 1.0, 1.0], [0.478174388408661, 0.426770567893982, 0.391106754541397, 0.351149171590805, 0.31280979514122], [2, 4, 6, 8, 10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 23,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Accuracy by iteration"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 24,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np\n",
+    "import sys\n",
+    "import os\n",
+    "from matplotlib import pyplot as plt\n",
+    "\n",
+    "# get accuracy and iteration number\n",
+    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
+    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
+    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
+    "\n",
+    "# get number of points\n",
+    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
+    "num_points = num_points_proxy[0]\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "iters = np.array(iters_proxy).reshape(num_points)\n",
+    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
+    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by iteration')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Loss by iteration"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 25,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get loss\n",
+    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
+    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
+    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation loss by iteration')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Loss')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Accuracy by time"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 26,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get time\n",
+    "time_proxy = %sql SELECT metrics_elapsed_time FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "time = np.array(time_proxy).reshape(num_points)/60.0\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by time')\n",
+    "plt.xlabel('Time (min)')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(time, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(time, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Time to achieve a given accuracy"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 27,
+   "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": [
+    "#plot\n",
+    "plt.title('Iris time by validation accuracy')\n",
+    "plt.xlabel('Accuracy')\n",
+    "plt.ylabel('Time (min)')\n",
+    "plt.grid(True)\n",
+    "plt.plot(train_accuracy, time, 'g.-', label='Train')\n",
+    "plt.plot(test_accuracy, time, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred_prob\"></a>\n",
+    "# 2. Predict probabilities\n",
+    "Predict with probabilities for each class:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 28,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "90 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.95964456</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.040107667</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.00024777954</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9597473</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.039995104</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.00025748153</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9761629</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.02375229</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>8.479464e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.99167526</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.008311711</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>1.30546e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.95964456</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.040107667</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.00024777954</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9731986</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.026633002</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0001682964</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5465453</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.44023818</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.01321647</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.58524394</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.38831925</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.026436739</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.54376984</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.4466499</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.009580206</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.57931274</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.4024986</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.018188644</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.6546029</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.31436205</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.031035094</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7274642</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.23212723</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.04040851</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.70608836</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.2678585</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.026053142</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7004706</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.24725808</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.052271266</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.60884434</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.37558457</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.015571073</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.6937521</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.22628777</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.07996013</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8121722</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.18745965</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00036821415</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.732954</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.26589376</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0011522635</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7756057</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.22408752</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00030681648</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.73672587</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.26255292</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00072121713</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6946963</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.30448684</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0008168273</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7407642</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.2582139</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0010218392</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.6248408</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.37318283</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0019763925</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.69742286</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.30152085</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0010562533</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8924382</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.10752802</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>3.379417e-05</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5494711</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.44651195</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.004016916</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7880493</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.21177953</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0001712008</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.76935935</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.23023891</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.00040176214</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.62273574</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.37572634</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0015378947</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.71288556</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.2861904</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0009240419</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(10, u'class_text', u'Iris-setosa', 0.95964456, 1),\n",
+       " (10, u'class_text', u'Iris-versicolor', 0.040107667, 2),\n",
+       " (10, u'class_text', u'Iris-virginica', 0.00024777954, 3),\n",
+       " (13, u'class_text', u'Iris-setosa', 0.9597473, 1),\n",
+       " (13, u'class_text', u'Iris-versicolor', 0.039995104, 2),\n",
+       " (13, u'class_text', u'Iris-virginica', 0.00025748153, 3),\n",
+       " (29, u'class_text', u'Iris-setosa', 0.9761629, 1),\n",
+       " (29, u'class_text', u'Iris-versicolor', 0.02375229, 2),\n",
+       " (29, u'class_text', u'Iris-virginica', 8.479464e-05, 3),\n",
+       " (34, u'class_text', u'Iris-setosa', 0.99167526, 1),\n",
+       " (34, u'class_text', u'Iris-versicolor', 0.008311711, 2),\n",
+       " (34, u'class_text', u'Iris-virginica', 1.30546e-05, 3),\n",
+       " (38, u'class_text', u'Iris-setosa', 0.95964456, 1),\n",
+       " (38, u'class_text', u'Iris-versicolor', 0.040107667, 2),\n",
+       " (38, u'class_text', u'Iris-virginica', 0.00024777954, 3),\n",
+       " (43, u'class_text', u'Iris-setosa', 0.9731986, 1),\n",
+       " (43, u'class_text', u'Iris-versicolor', 0.026633002, 2),\n",
+       " (43, u'class_text', u'Iris-virginica', 0.0001682964, 3),\n",
+       " (56, u'class_text', u'Iris-versicolor', 0.5465453, 1),\n",
+       " (56, u'class_text', u'Iris-virginica', 0.44023818, 2),\n",
+       " (56, u'class_text', u'Iris-setosa', 0.01321647, 3),\n",
+       " (61, u'class_text', u'Iris-versicolor', 0.58524394, 1),\n",
+       " (61, u'class_text', u'Iris-virginica', 0.38831925, 2),\n",
+       " (61, u'class_text', u'Iris-setosa', 0.026436739, 3),\n",
+       " (64, u'class_text', u'Iris-versicolor', 0.54376984, 1),\n",
+       " (64, u'class_text', u'Iris-virginica', 0.4466499, 2),\n",
+       " (64, u'class_text', u'Iris-setosa', 0.009580206, 3),\n",
+       " (67, u'class_text', u'Iris-versicolor', 0.57931274, 1),\n",
+       " (67, u'class_text', u'Iris-virginica', 0.4024986, 2),\n",
+       " (67, u'class_text', u'Iris-setosa', 0.018188644, 3),\n",
+       " (70, u'class_text', u'Iris-versicolor', 0.6546029, 1),\n",
+       " (70, u'class_text', u'Iris-virginica', 0.31436205, 2),\n",
+       " (70, u'class_text', u'Iris-setosa', 0.031035094, 3),\n",
+       " (72, u'class_text', u'Iris-versicolor', 0.7274642, 1),\n",
+       " (72, u'class_text', u'Iris-virginica', 0.23212723, 2),\n",
+       " (72, u'class_text', u'Iris-setosa', 0.04040851, 3),\n",
+       " (75, u'class_text', u'Iris-versicolor', 0.70608836, 1),\n",
+       " (75, u'class_text', u'Iris-virginica', 0.2678585, 2),\n",
+       " (75, u'class_text', u'Iris-setosa', 0.026053142, 3),\n",
+       " (89, u'class_text', u'Iris-versicolor', 0.7004706, 1),\n",
+       " (89, u'class_text', u'Iris-virginica', 0.24725808, 2),\n",
+       " (89, u'class_text', u'Iris-setosa', 0.052271266, 3),\n",
+       " (92, u'class_text', u'Iris-versicolor', 0.60884434, 1),\n",
+       " (92, u'class_text', u'Iris-virginica', 0.37558457, 2),\n",
+       " (92, u'class_text', u'Iris-setosa', 0.015571073, 3),\n",
+       " (94, u'class_text', u'Iris-versicolor', 0.6937521, 1),\n",
+       " (94, u'class_text', u'Iris-virginica', 0.22628777, 2),\n",
+       " (94, u'class_text', u'Iris-setosa', 0.07996013, 3),\n",
+       " (101, u'class_text', u'Iris-virginica', 0.8121722, 1),\n",
+       " (101, u'class_text', u'Iris-versicolor', 0.18745965, 2),\n",
+       " (101, u'class_text', u'Iris-setosa', 0.00036821415, 3),\n",
+       " (102, u'class_text', u'Iris-virginica', 0.732954, 1),\n",
+       " (102, u'class_text', u'Iris-versicolor', 0.26589376, 2),\n",
+       " (102, u'class_text', u'Iris-setosa', 0.0011522635, 3),\n",
+       " (103, u'class_text', u'Iris-virginica', 0.7756057, 1),\n",
+       " (103, u'class_text', u'Iris-versicolor', 0.22408752, 2),\n",
+       " (103, u'class_text', u'Iris-setosa', 0.00030681648, 3),\n",
+       " (112, u'class_text', u'Iris-virginica', 0.73672587, 1),\n",
+       " (112, u'class_text', u'Iris-versicolor', 0.26255292, 2),\n",
+       " (112, u'class_text', u'Iris-setosa', 0.00072121713, 3),\n",
+       " (113, u'class_text', u'Iris-virginica', 0.6946963, 1),\n",
+       " (113, u'class_text', u'Iris-versicolor', 0.30448684, 2),\n",
+       " (113, u'class_text', u'Iris-setosa', 0.0008168273, 3),\n",
+       " (115, u'class_text', u'Iris-virginica', 0.7407642, 1),\n",
+       " (115, u'class_text', u'Iris-versicolor', 0.2582139, 2),\n",
+       " (115, u'class_text', u'Iris-setosa', 0.0010218392, 3),\n",
+       " (116, u'class_text', u'Iris-virginica', 0.6248408, 1),\n",
+       " (116, u'class_text', u'Iris-versicolor', 0.37318283, 2),\n",
+       " (116, u'class_text', u'Iris-setosa', 0.0019763925, 3),\n",
+       " (117, u'class_text', u'Iris-virginica', 0.69742286, 1),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.30152085, 2),\n",
+       " (117, u'class_text', u'Iris-setosa', 0.0010562533, 3),\n",
+       " (119, u'class_text', u'Iris-virginica', 0.8924382, 1),\n",
+       " (119, u'class_text', u'Iris-versicolor', 0.10752802, 2),\n",
+       " (119, u'class_text', u'Iris-setosa', 3.379417e-05, 3),\n",
+       " (127, u'class_text', u'Iris-virginica', 0.5494711, 1),\n",
+       " (127, u'class_text', u'Iris-versicolor', 0.44651195, 2),\n",
+       " (127, u'class_text', u'Iris-setosa', 0.004016916, 3),\n",
+       " (136, u'class_text', u'Iris-virginica', 0.7880493, 1),\n",
+       " (136, u'class_text', u'Iris-versicolor', 0.21177953, 2),\n",
+       " (136, u'class_text', u'Iris-setosa', 0.0001712008, 3),\n",
+       " (144, u'class_text', u'Iris-virginica', 0.76935935, 1),\n",
+       " (144, u'class_text', u'Iris-versicolor', 0.23023891, 2),\n",
+       " (144, u'class_text', u'Iris-setosa', 0.00040176214, 3),\n",
+       " (146, u'class_text', u'Iris-virginica', 0.62273574, 1),\n",
+       " (146, u'class_text', u'Iris-versicolor', 0.37572634, 2),\n",
+       " (146, u'class_text', u'Iris-setosa', 0.0015378947, 3),\n",
+       " (147, u'class_text', u'Iris-virginica', 0.71288556, 1),\n",
+       " (147, u'class_text', u'Iris-versicolor', 0.2861904, 2),\n",
+       " (147, u'class_text', u'Iris-setosa', 0.0009240419, 3)]"
+      ]
+     },
+     "execution_count": 28,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_model',      -- model\n",
+    "                                   'iris_test',       -- test_table\n",
+    "                                   'id',              -- id column\n",
+    "                                   'attributes',      -- independent var\n",
+    "                                   'iris_predict',    -- output table\n",
+    "                                   'prob'             -- response type\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id, rank;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"warm_start\"></a>\n",
+    "# 3. Warm start\n",
+    "Next, use the warm_start parameter to continue learning, using the coefficients from the run above. Note that we don't drop the model table or model summary table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 29,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 29,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10,                   -- num_iterations\n",
+    "                                FALSE,                -- use GPUs\n",
+    "                                'iris_test_packed',   -- validation dataset\n",
+    "                                2,                    -- metrics compute frequency\n",
+    "                                TRUE,                 -- warm start\n",
+    "                               'Sophie L.',           -- name \n",
+    "                               'Simple MLP for iris dataset'  -- description\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "In the summary table and plots below note that the loss and accuracy values pick up from where the previous run left off:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 30,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>1</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Simple MLP for iris dataset</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-06 00:27:51.102558</td>\n",
+       "        <td>2021-03-06 00:27:52.451185</td>\n",
+       "        <td>[0.781347990036011, 0.923561096191406, 1.06405401229858, 1.20302820205688, 1.34854102134705]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.975000023842</td>\n",
+       "        <td>0.194891035557</td>\n",
+       "        <td>[0.941666662693024, 0.966666638851166, 0.958333313465118, 0.949999988079071, 0.975000023841858]</td>\n",
+       "        <td>[0.262409120798111, 0.24169448018074, 0.222953796386719, 0.207046672701836, 0.194891035556793]</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>0.188044458628</td>\n",
+       "        <td>[0.966666638851166, 1.0, 1.0, 1.0, 1.0]</td>\n",
+       "        <td>[0.293483078479767, 0.254781544208527, 0.232207864522934, 0.212682083249092, 0.188044458627701]</td>\n",
+       "        <td>[2, 4, 6, 8, 10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', None, 2, u'Sophie L.', u'Simple MLP for iris dataset', u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 6, 0, 27, 51, 102558), datetime.datetime(2021, 3, 6, 0, 27, 52, 451185), [0.781347990036011, 0.923561096191406, 1.06405401229858, 1.20302820205688, 1.34854102134705], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.975000023841858, 0.194891035556793, [0.941666662693024, 0.966666638851166, 0.958333313465118, 0.949999988079071, 0.975000023841858], [0.262409120798111, 0.24169448018074, 0.222953796386719, 0.207046672701836, 0.194891035556793], 1.0, 0.188044458627701, [0.966666638851166, 1.0, 1.0, 1.0, 1.0], [0.293483078479767, 0.254781544208527, 0.232207864522934, 0.212682083249092, 0.188044458627701], [2, 4, 6, 8, 10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 30,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 31,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np\n",
+    "import sys\n",
+    "import os\n",
+    "from matplotlib import pyplot as plt\n",
+    "\n",
+    "# get accuracy and iteration number\n",
+    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
+    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
+    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
+    "\n",
+    "# get number of points\n",
+    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
+    "num_points = num_points_proxy[0]\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "iters = np.array(iters_proxy).reshape(num_points)\n",
+    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
+    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by iteration - warm start')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 32,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get loss\n",
+    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
+    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
+    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation loss by iteration - warm start')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Loss')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"transfer_learn\"></a>\n",
+    "# Transfer learning\n",
+    "\n",
+    "<a id=\"load2\"></a>\n",
+    "# 1. Define and load model architecture with some layers frozen\n",
+    "Here we want to start with initial weights from a pre-trained model rather than training from scratch.  We also want to use a model architecture with the earlier feature layer(s) frozen to save on training time.  The example below is somewhat contrived but gives you the idea of the steps.\n",
+    "\n",
+    "First define a model architecture with the 1st hidden layer frozen:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 33,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_1\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_3 (Dense)              (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_4 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_5 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 143\n",
+      "Non-trainable params: 50\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model_transfer = Sequential()\n",
+    "model_transfer.add(Dense(10, activation='relu', input_shape=(4,), trainable=False))\n",
+    "model_transfer.add(Dense(10, activation='relu'))\n",
+    "model_transfer.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model_transfer.summary()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 34,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": false, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_1\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 34,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model_transfer.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load transfer model into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 35,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': False, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>A transfer model</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Sophie', u'A simple model'),\n",
+       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1341 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Maria', u'A transfer model')]"
+      ]
+     },
+     "execution_count": 35,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,                      \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": false, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Maria',               -- Name\n",
+    "                               'A transfer model'     -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT model_id, model_arch, name, description FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train2\"></a>\n",
+    "# 2. Train transfer model\n",
+    "\n",
+    "Fetch the weights from a previous MADlib run.  (Normally these would be downloaded from a source that trained the same model architecture on a related dataset.)"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 36,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 36,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "UPDATE model_arch_library \n",
+    "SET model_weights = iris_model.model_weights \n",
+    "FROM iris_model \n",
+    "WHERE model_arch_library.model_id = 2;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Now train the model using the transfer model and the pre-trained weights:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 37,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 37,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                2,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10,                   -- num_iterations\n",
+    "                                FALSE,                -- use GPUs\n",
+    "                                'iris_test_packed',   -- validation dataset\n",
+    "                                2                     -- metrics compute frequency\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 38,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>2</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-06 00:27:56.293667</td>\n",
+       "        <td>2021-03-06 00:27:57.661243</td>\n",
+       "        <td>[0.832237005233765, 0.965812921524048, 1.09816098213196, 1.22954201698303, 1.3674840927124]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.949999988079</td>\n",
+       "        <td>0.153273612261</td>\n",
+       "        <td>[0.949999988079071, 0.958333313465118, 0.949999988079071, 0.949999988079071, 0.949999988079071]</td>\n",
+       "        <td>[0.182110622525215, 0.173247531056404, 0.165094882249832, 0.158673033118248, 0.153273612260818]</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>0.134765788913</td>\n",
+       "        <td>[1.0, 1.0, 1.0, 1.0, 1.0]</td>\n",
+       "        <td>[0.177851542830467, 0.161254957318306, 0.152191400527954, 0.142795532941818, 0.134765788912773]</td>\n",
+       "        <td>[2, 4, 6, 8, 10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 2, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', None, 2, None, None, u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 6, 0, 27, 56, 293667), datetime.datetime(2021, 3, 6, 0, 27, 57, 661243), [0.832237005233765, 0.965812921524048, 1.09816098213196, 1.22954201698303, 1.3674840927124], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.949999988079071, 0.153273612260818, [0.949999988079071, 0.958333313465118, 0.949999988079071, 0.949999988079071, 0.949999988079071], [0.182110622525215, 0.173247531056404, 0.165094882249832, 0.158673033118248, 0.153273612260818], 1.0, 0.134765788912773, [1.0, 1.0, 1.0, 1.0, 1.0], [0.177851542830467, 0.161254957318306, 0.152191400527954, 0.142795532941818, 0.134765788912773], [2, 4, 6, 8, 10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 38,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Note loss picks up from where the last training left off:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 39,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYsAAAEWCAYAAACXGLsWAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADl0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uIDIuMi41LCBodHRwOi8vbWF0cGxvdGxpYi5vcmcvSM8oowAAIABJREFUeJzt3Xl4FeX5//H3TQgQAVltXIKA4gYVQRCMKyAqqEVFSnHBilq0FcG2/qy2Vi3u1qpBqKiAgkWoxeVLW3eEahXZFxVEUVkiLoCCArKE3L8/ZkKHkGTOCTmckHxe13WuM8szM/fMmZn7zDObuTsiIiJlqZHuAEREpPJTshARkVhKFiIiEkvJQkREYilZiIhILCULERGJpWSRADO72Mxe3YPTa2FmbmY1w/aXzOzniZQtx7R+b2ajdife6szMuphZfgWNa6SZ/bEixrUbMXxgZl3SGUM6mNkvzewrM9tgZk0qYHzTzOzKioitsjDdZwFmtgy40t1fT3csECQA4DMg090LKrBsF+Bv7p5TEXFK6pbpnvitzOxJIN/db07VNJKI5TaglbtfkoZpZwLfAce7+4IKGuc0gt+vyvwR05FFjPL+Y5f0qu6/W1Wbfwukan+VDdQBPijPwGaWUbHhlDmt9P2u7l7tP8AyoHvYfBnwNvAgsBa4I+z237C/hf2+Jvg38h7w4xLG+TNgdrFuvwYmh81nA/PCcawEbouUawE4UDNsn0Zw5AOQAdwPrAE+Ba4pVnYAsBj4Pux/Vdi9LvADUAhsCD8HArcR/AMqmnYvgo1mXTjdo4otp+uBhcB64O9AnVKW6aHAG+EyXAOMBxpG+jcDngNWh2WGR/r9IjIPi4Bjw+5O8O+zqNyTwB1hcxcgH/gd8CXwFNAI+Fc4jW/D5pzI8I2BJ4BVYf8Xwu7vAz+JlMsM56F9CfNZNN3fh2WWAReH/Y4DvgIyIuV7AwtKWWZPEqxvpf1WNYAbgU/CZfYM0LjYOnMFsAJ4M+z+j3B5rAfeBNqE3QcC24Ct4fj/WcK2UBt4KFw+q8Lm2sXm+7cE28IXwIBybn89wji2hbEsiKz3dxJsjz8ArShl/U4kJuAsgvXpe+BzgnX5cGBjuOw2AG+EZY8EXgO+AZYAfYv9To8AL4bDdi9hnqYRbrNh++Vh3N8CrwDNI/3yCPYB3wFzgJMj/W4DJgF/C/tfGXZ7BhgXzssHQMeU7ydTPYG94cOuyaIAuBaoCWSxc7I4M/xBGxIkjqOAA0oY5z7hD3lYpNssoF9kxT6aYAfQlmCncl7YrwWlJ4urgQ8JdraNganFyp5NsKM24FRgE//b2XYhqHaIxnkbYbKIbDinE+wgbwCWArUiy2kmwY6rcbjyX13KMm0Vjqc2sB/BjuqhsF8GsIAg6dYl+Fd3UtjvpwQb8nHhPLQq2rCITxYFwL3hNLOAJsAF4W9Rn2DH+UJk+H8TJLxG4fyeGna/Afh7pNy5wHulzGfRdB8Ip3tquAyPCPsvAnpGyj8P/LaUcRWfn+K/1RDgXSAnnNajwIRi68y4cJlmhd0vD+e9aMc/v6TplbItDA2n96PwN3wHuL3YfA8Nl91ZBOtao3Jug7cR+dMSWe9XAG0ItsVM4tfvUmMiSB4nh82NIsMVLbuibaguwc57QDjd9gR/BFpHltt64ESC7XeXP0zsvM2eS7AdHRWO72bgnUjZSwjW1ZoEie7LonGGy2UbcF44rayw2+Zw/jKAu4F3U76fTPUE9oYPuyaLFcX6X8b/kkU34CPgeKBGzHj/BtwSNh9GkDz2KaXsQ8CDpay80RXvDSI7aOCMaNkSxvsCMCRs7kLZyeKPwDORfjUIdtxdIsvpkkj/+4CRCS7j84B5YXMuwb/9XWIm+Nc1pJRxxCWLrSVtuJHy7YBvw+YDCP6577JzI0iG3wP7hu2TgBtKGWcXgh1U3Ui3Z4A/hs2/A8aHzY0Jdl67/LkoZX6K/1aLgdMi7QcQ7EhqRtaZQ8qY/4ZhmQbFp1fKtvAJcFak35nAskh8P0R/Q4J/88eXcxvcsR5Guk0DhsYMV3z9LjUmgsRzVdHvGilTtOyKtrefAW8VK/MocGtkuY2LiWsa/9tmXwKuKLZdbSJydFFs2G+BYyLL5c0SltXrkfbWwA/lWe7JfHTOomQrS+vh7m8Aw4ERwNdm9piZ7VtK8aeBC8Pmiwj+1W4CMLPOZjbVzFab2XqCI4amCcR2YLH4lkd7mllPM3vXzL4xs3UE/z4SGW/RuHeMz90Lw2kdFCnzZaR5E1CvpBGZWbaZTTSzz83sO4LEWRRHM2C5l3xCvhnBTqo8Vrv75kgM+5jZo2a2PIzhTaBhWMfcDPjG3b8tPhJ3X0VQ9XGBmTUEehJUo5XmW3ffGGlfTrAsIZjvn5hZXaAvwU7oi3LOX3PgeTNbF/62i4HtBHXuRXasG2aWYWb3mNkn4fwvC3uVa31g5/kCWFvsNyxxfTCzk8OrjDaYWbLnBXbaFhNYv8uK6YKw/HIz+4+Z5ZYyzeZA56LlHE7nYmD/0uKK0RzIi4zrG4Ijo4PCebrezBab2fqwf4Ni81TStIpvh3VSfT5DyaJkXmZP92Hu3oEgox8O/L9Sir4G7Gdm7QiSxtORfk8Dk4Fm7t4AGEmwAsX5gmBHV+TgogYzqw08S3BOI9vdGxLUqxaNt8z5IqiXbh4Zn4XT+jyBuIq7K5ze0e6+L8GhdlEcK4GDS1m5VxJUM5RkE0GVUpH9i/UvPn+/BY4AOocxnBJ2t3A6jcNkUJKxYcw/Baa7e1nLoFGYDIocTLAsCYebTnCuoj/BuZRElPRbrSSo0moY+dQpFlt0uIsIqkC6E+yAWoTdy7U+EJmvZLj7W+5eL/y0Ka1YXPcE1u+4OGa5+7kE1WovEBwBlmQl8J9iy7meu/8ygXhLG99VxcaX5e7vmNnJBNWefQmOchsSVHFF5ymZaaWMkkWSzOy48Kggk6BuejNBdcYu3H0bQT35nwmqIF6L9K5P8M92s5l1ItiwE/EMMNjMcsysEcEJzyK1COqmVwMFZtaToJqqyFdAEzNrUMa4zzaz08L5+y2whaCuOln1CU4Yrjezg9g5oc4kSHr3mFldM6tjZieG/UYB15tZh/AKmFZmVrTDmg9cFP5j7kFQZx0Xww/AOjNrDNxa1CP8d/8S8Fcza2RmmWZ2SmTYF4BjCc4TjEtgfv9kZrXCjf8cgt+9yDiCHcLRBCf1E1HSbzUSuLNoeZjZfmZ2bhnjqE/w+60lSLJ3lTCNQ8oYfgJwczidpsAtBEdKqfAV0CLmiqe49btU4W9zsZk1CLfL7yhluyW4EOJwM+sfrheZ4XZ/VOKzs5ORwE1m1iaMpYGZ/TTsV5+gGnM1UNPMbgFKq6lIKyWL5O0LPE5Qr7icYEP8cxnlnyb4Z/ePYofHvwKGmtn3BBthaf9yinucoF5/ATCXyM7H3b8HBofj+pYgAU2O9P+QYAfwaXhIHK1SwN2XEPybfpjghN5PCK4K2ppgbFF/ItjZric4kRyNc3s47lYE9cj5BPXEuPs/CK6AeZrgvMELBIkWgh33Twiu1Lo47FeWhwhOCK4hOFH7crH+/Qnq/D8kqNu+LhLjDwT/YlsSv4P/kmB5ryKorro6XNZFniesQiqqhoxTym+VR/B7vhquN+8CncsYzTiCdfRzghPt7xbrPxpoHY6/pGV5BzCb4Oq39wjWtzsSib8cipLrWjObW1KBuPU7Af2BZWGV3NUE61Bp0zkD6Efwm37J/y6cSJq7Px8OPzGc9vsEVZsQbMsvE5wHXU7w5zOZKq49RjfliZQi/Jd3uFfAjWJm9glBVUSluPFTJFlV6sYdkYoSVltdQfBvdHfHdQFBvfMbuzsukXRRNZRIMWb2C4KqgJfc/c3dHNc0ghu4rgmvLhPZK6kaSkREYunIQkREYlWZcxZNmzb1Fi1alHv4jRs3Urdu3fiCe5jiSo7iSo7iSk5VjGvOnDlr3H2/2IKpvkV8T306dOjgu2Pq1Km7NXyqKK7kKK7kKK7kVMW4KPbA09I+qoYSEZFYShYiIhJLyUJERGIpWYiISCwlCxERiZWyZGFmY8zsazN7v5T+ZmbDzGypmS00s2Mj/X5uZh+Hn5+nKkYREUlMKo8sniR4t25pehK8Pe4wgvcBPwI7nslzK8HTNDsBt4aP4k6d6dM5ePx4mD49pZOpMrS8kqPllRwtr+TsoeWV0sd9mFkL4F/u/uMS+j0KTHP3CWH7EoLXInYheI3nVSWVK03Hjh199uzZyQc5fTqccgpeUIDVqAFt20KD0l73sOetW7eOhg1Lez9PGqxfDwsX4oWFWl6J0PJKjpZXcqLLKysLpkyB3NJeAFgyM5vj7h3jyqXzDu6D2Pm57flht9K678LMBhIclZCdnc20adOSDuLg8eNpWVCAAV5YyOYvv2RLJXpe1vbt21m3bl26w9ih9ldfUaewUMsrQVpeydHySk50eRVu2cKyMWNYsWVLaiaWyJ175f0QvMbx/VL6/Qs4KdI+BegIXA/cHOn+R+D6uGmV+w7ud95xz8ry7TVquGdlBe2VSKW7Y1TLKzlaXsnR8kpOBSwv9oI7uD9n53dJ54TdSuueGrm5MGUKyy6/vFyHcNWOlldytLySo+WVnD24vNJZDTUZGGRmEwlOZq939y/M7BXgrshJ7TOAm1IaSW4uK7Zs4RCtmInR8kqOlldytLySs4eWV8qShZlNIDhZ3dTM8gmucMoEcPeRwIvAWcBSYBMwIOz3jZndDswKRzXU3b9JVZwiIhIvZcnC3S+M6e/ANaX0GwOMSUVcIiKSPN3BLSIisZQsREQklpKFiIjEUrIQEZFYShYiIhJLyUJERGIpWYiISCwlCxERiaVkISIisZQsREQklpKFiIjEUrIQEZFYShYiIhJLyUJERGIpWYiISCwlCxERiaVkISIisZQsREQklpKFiIjEUrIQEZFYShYiIhJLyUJERGIpWYiISCwlCxERiaVkISIisZQsREQklpKFiIjEUrIQEZFYShYiIhJLyUJERGIpWYiISKyUJgsz62FmS8xsqZndWEL/5mY2xcwWmtk0M8uJ9LvXzN4PPz9LZZwiIlK2lCULM8sARgA9gdbAhWbWulix+4Fx7t4WGArcHQ57NnAs0A7oDFxvZvumKlYRESlbKo8sOgFL3f1Td98KTATOLVamNfBG2Dw10r818Ka7F7j7RmAh0COFsYqISBnM3VMzYrM+QA93vzJs7w90dvdBkTJPAzPcPc/MegPPAk2BDsCtwOnAPsBMYIS7/6XYNAYCAwGys7M7TJw4sdzxbtiwgXr16pV7+FRRXMlRXMlRXMmpinF17dp1jrt3jC3o7in5AH2AUZH2/sDwYmUOBJ4D5gF5QD7QMOz3B2A+8BowHriurOl16NDBd8fUqVN3a/hUUVzJUVzJUVzJqYpxAbM9gX16KquhPgeaRdpzwm47uPsqd+/t7u3D5IC7rwu/73T3du5+OmDARymMVUREypDKZDELOMzMWppZLaAfMDlawMyamllRDDcBY8LuGWbWJGxuC7QFXk1hrCIiUoaaqRqxuxeY2SDgFSADGOPuH5jZUILDnslAF+BuM3PgTeCacPBM4C0zA/gOuMTdC1IVq4iIlC1lyQLA3V8EXizW7ZZI8yRgUgnDbSa4IkpERCoB3cEtIiKxlCxERCSWkoWIiMRSshARkVhKFiIiEkvJQkREYilZiIhILCULERGJpWQhIiKxlCxERCSWkoWIiMRSshARkVhKFiIiEkvJQkREYilZiIhILCULERGJpWQhIiKxlCxERCSWkoWIiMRSshARkVhKFiIiEkvJQkREYilZiIhILCULERGJpWQhIiKxlCxERCSWkoWIiMRSshARkVhKFiIiEkvJQkREYqU0WZhZDzNbYmZLzezGEvo3N7MpZrbQzKaZWU6k331m9oGZLTazYWZmqYxVRERKl7JkYWYZwAigJ9AauNDMWhcrdj8wzt3bAkOBu8NhTwBOBNoCPwaOA05NVawiIlK2VB5ZdAKWuvun7r4VmAicW6xMa+CNsHlqpL8DdYBaQG0gE/gqhbGKiEgZzN3LLmB2LfA3d/82qRGb9QF6uPuVYXt/oLO7D4qUeRqY4e55ZtYbeBZo6u5rzex+4ErAgOHu/ocSpjEQGAiQnZ3dYeLEicmEuJMNGzZQr169cg+fKoorOYorOYorOVUxrq5du85x946xBd29zA9wB7AUeAboQZhgEhiuDzAq0t6fYKcfLXMg8BwwD8gD8oGGQCvg30C98DMdOLms6XXo0MF3x9SpU3dr+FRRXMlRXMlRXMmpinEBsz2BfXpsNZS73wwcBowGLgM+NrO7zOzQmEE/B5pF2nPCbtFxr3L33u7eHvhD2G0dcD7wrrtvcPcNwEtAblysIiKSGgmdswizz5fhpwBoBEwys/vKGGwWcJiZtTSzWkA/YHK0gJk1NbOiGG4CxoTNK4BTzaymmWUSnNxenOA8iYhIBasZV8DMhgCXAmuAUcD/c/dt4U7+Y+CGkoZz9wIzGwS8AmQAY9z9AzMbSnDYMxnoAtxtZg68CVwTDj4J6Aa8R3Cy+2V3/2f5Z1NEZGfbtm0jPz+fzZs3JzxMgwYNWLy48v1vTSSuOnXqkJOTQ2ZmZrmmEZssgMZAb3dfHu3o7oVmdk5ZA7r7i8CLxbrdEmmeRJAYig+3HbgqgdhERMolPz+f+vXr06JFCxK9jev777+nfv36KY4seXFxuTtr164lPz+fli1blmsaiVRDvQR8U9RiZvuaWecwgMqXYkVEErB582aaNGmScKLYm5kZTZo0SeooqrhEksUjwIZI+4awm4jIXq06JIoiuzuviSQLC09wA0H1E4lVX4mISCnWrl1Lu3btaNeuHfvvvz8HHXTQjvatW7cmNI4BAwawZMmSFEcaSGSn/6mZDeZ/RxO/Aj5NXUgiIlVfkyZNmD9/PgC33XYb9erV4/rrr9+pzI57HGqU/L/+iSeeAIJzFqmWyJHF1cAJBPdI5AOdCe+aFhGpTmasmsHdb93N9JXTUzaNpUuX0rp1ay6++GLatGnDF198wcCBA+nYsSNt2rRh6NChO8qedNJJzJ8/n4KCAho2bMiNN97IMcccQ25uLl9//XWFxhV7ZOHuXxPcIyEiUiVd9/J1zP9yfpll1m9Zz8KvFlLohdSwGrTNbkuD2g1KLd9u/3Y81OOhcsXz4YcfMm7cODp2DJ7Ccc8999C4cWMKCgro2rUrffr0oXXrnZ/Lun79ek499VTuuecefvOb3zBmzBhuvHGXh32XWyL3WdQBrgDaEDzcDwB3v7zCohARqeTWb15PoRcCUOiFrN+8vsxksTsOPfTQHYkCYMKECYwePZqCggJWrVrFokWLdkkWWVlZ9OzZE4AOHTrw1ltvVWhMiZyzeAr4EDiT4DHiF6O7qUWkCknkCGD6yumcNu40tm7fSq2MWozvPZ7cZql5ClHdunV3NH/88cfk5eUxc+ZMGjZsyCWXXFLiJbC1atXa0ZyRkUFBQUGFxpTIOYtW7v5HYKO7jwXOJjhvISJSbeQ2y2Vyn8nc3vV2plw6JWWJorjvvvuO+vXrs++++/LFF1/wyiuv7JHpFpfIkcW28Hudmf2Y4PlQP0pdSCIilVPnAzvT/Yjue3Saxx57LK1bt+bII4+kefPmnHjiiXt0+kUSSRaPmVkj4GaCBwHWA/6Y0qhERKqR2267bUdzq1atdlxSC8HNdE899VSJw/33v/8Fgktn161bt6N7v3796NevYq9LKjNZhA8L/M6DFx+9CRxSoVMXEZG9QpnnLMK7tUt8qqyIiFQfiZzgft3MrjezZmbWuOiT8shERKTSSOScxc/C72si3RxVSYmIVBuJ3MFdvoefi4hIlZHIHdyXltTd3cdVfDgiIlIZJVINdVykuQ5wGjAXULIQESmntWvXctpppwHw5ZdfkpGRwX777QfAzJkzd7ojuyxjxozhlFNOSfkb/BKphro22m5mDYGJKYtIRKQaSOQR5YkYM2YMRxxxBK1ataroEHdSnpcYbQR0HkNEqp0aM2bArFnQpQvkpu5xH2PHjmXEiBFs3bqVE044geHDh1NYWMiAAQOYP38+7s7AgQPJzs5m/vz5XHbZZdStWzepI5JkJXLO4p8EVz9BcKlta+CZlEQjIpIO110H88t+RDnr17PPwoVQWAg1akDbttCgjKfOtmsHDyX/iPL333+f559/nnfeeYeaNWsycOBAJk6cyKGHHsqaNWt47733AFi3bh0NGzbk4Ycf5t577035Y0ASObK4P9JcACx39/wUxSMiUjmtXx8kCgi+168vO1mU0+uvv86sWbN2PKL8hx9+oFmzZpx55pksWbKEwYMHc/bZZ3PGGWdU+LTLkkiyWAF84e6bAcwsy8xauPuylEYmIrKnJHIEMH06nHYabN0KtWrB+PEpqYpydy6//HJuv/32XfotXLiQl156iREjRvDss8/y2GOPVfj0S5PIHdz/AAoj7dvDbiIi1UduLpsmT4bbb4cpU1J2zqJ79+4888wzrFmzBgiumlqxYgWrV6/G3fnpT3/K0KFDmTt3LgD169dnw4YNKYklKpEji5ruvrWoxd23mllqzqCIiFRihZ07Q/fUPqL86KOP5tZbb6V79+4UFhaSmZnJyJEjycjI4IorrsDdMTPuvfdeAAYMGMCgQYPSf4IbWG1mvdx9MoCZnQusSUk0IiLVUPQR5QAXXXQRF1100S7l5s2bt0u3vn370rNnz/TfZwFcDYw3s+Fhez5Q4l3dIiJSNSVyU94nwPFmVi9sT33lmIiIVCqxJ7jN7C4za+juG9x9g5k1MrM79kRwIiJSOSRyNVRPd9/xvr7wrXlnpS4kEZE9w93jC1URuzuviSSLDDOrXdRiZllA7TLKi4hUenXq1GHt2rXVImG4O2vXrqVOnTrlHkciJ7jHA1PM7AnAgMuAsYmM3Mx6AHlABjDK3e8p1r85MAbYD/gGuMTd882sK/BgpOiRQD93fyGR6YqIxMnJySE/P5/Vq1cnPMzmzZt3a4ebKonEVadOHXJycso9jUROcN9rZguA7gTPiHoFaB43nJllACOA0wmuoJplZpPdfVGk2P3AOHcfa2bdgLuB/u4+FWgXjqcxsBR4Nak5ExEpQ2ZmJi1bJvdM1GnTptG+ffsURVR+eyKuRKqhAL4iSBQ/BboBixMYphOw1N0/DW/qmwicW6xMa+CNsHlqCf0B+gAvufumBGMVEZEKZqXV15nZ4cCF4WcN8HfgenePPaoIh+8D9HD3K8P2/kBndx8UKfM0MMPd88ysN/As0NTd10bKvAE84O7/KmEaA4GBANnZ2R0mTiz/azY2bNhAvXr1yj18qiiu5Ciu5Ciu5FTFuLp27TrH3TvGFnT3Ej8Ez4P6D9Aq0u3T0sqXMHwfgvMURe39geHFyhwIPAfMIzi3kQ80jPQ/AFgNZMZNr0OHDr47pk6dulvDp4riSo7iSo7iSk5VjAuY7Qns08s6Z9Eb6AdMNbOXCaqRLImE9TnQLNKeE3aLJqpV4XQIb/q7wCOX6QJ9gefdfVsS0xURkQpW6jkLd3/B3fsRXIk0FbgO+JGZPWJmiTxIfRZwmJm1DB882A+YHC1gZk3NrCiGmwiujIq6EJiQ2KyIiEiqxJ7gdveN7v60u/+E4OhgHvC7BIYrAAYRXD21GHjG3T8ws6Fm1iss1gVYYmYfAdnAnUXDm1kLgiOT/yQzQyIiUvGSege3B3dvPxZ+Ein/IvBisW63RJonAZNKGXYZcFAy8YmISGokeumsiIhUY0oWIiISS8lCRERiKVmIiEgsJQsREYmlZCEiIrGULEREJJaShYiIxFKyEBGRWEoWIiISS8lCRERiKVmIiEgsJQsREYmlZCEiIrGULEREJJaShYiIxFKyEBGRWEoWIiISS8lCRERiKVmIiEgsJQsREYmlZCEiIrGULEREJJaShYiIxFKyEBGRWEoWIiISS8lCRERiKVmIiEgsJQsREYmlZCEiIrFSmizMrIeZLTGzpWZ2Ywn9m5vZFDNbaGbTzCwn0u9gM3vVzBab2SIza5HKWEVEpHQpSxZmlgGMAHoCrYELzax1sWL3A+PcvS0wFLg70m8c8Gd3PwroBHydqlhFRKRsqTyy6AQsdfdP3X0rMBE4t1iZ1sAbYfPUov5hUqnp7q8BuPsGd9+UwlhFRKQM5u6pGbFZH6CHu18ZtvcHOrv7oEiZp4EZ7p5nZr2BZ4GmwMnAlcBWoCXwOnCju28vNo2BwECA7OzsDhMnTix3vBs2bKBevXrlHj5VFFdyFFdyFFdyqmJcXbt2nePuHWMLuntKPkAfYFSkvT8wvFiZA4HngHlAHpAPNAyHXQ8cAtQkSCJXlDW9Dh06+O6YOnXqbg2fKoorOYorOYorOVUxLmC2J7BPT2U11OdAs0h7TthtB3df5e693b098Iew27owacz3oAqrAHgBODaFsYqISBlSmSxmAYeZWUszqwX0AyZHC5hZUzMriuEmYExk2IZmtl/Y3g1YlMJYRUSkDClLFuERwSDgFWAx8Iy7f2BmQ82sV1isC7DEzD4CsoE7w2G3A9cDU8zsPcCAx1MVq4iIlK1mKkfu7i8CLxbrdkukeRIwqZRhXwPapjI+KZ/pK6czfsV4aq+sTW6z3HSHIyJ7gO7gloRtL9zOfW/fx8lPnMyoz0bRbVw3pq+cnu6wRGQPULKQWOs3r+fB6Q9y+PDD+d3rv2N7eAXz5oLNDPzXQN5e8XbR1W0iUkUpWUipPlr7Ede+eC05D+bwm1d/wwH1DuCOrneQVTOLGtSgZo2aLFu3jJOeOInjHj+OcQvGsaVgS7rDFpEUSOk5C9n7uDuvfvIqeTPyeGnpS2TWyKTfj/sxpPMQOhzYAYBuLbsxZuoYLu96OUdnH81TC55i2Mxh/PyFn3PDazdwdcerubrj1exfb/80z42IVBQlCwFg49aNjFswjmEzh/Hhmg/JrpvNbafexlUdr9plp5/bLJctB2/ZcXL7l8f9kqs6XsVrn7zGsJnD+NN//sRdb921S5IRkb2XkkU1t3zdcobPHM6oeaNYt3kdHQ7owLjzxtG3TV9q16yd8Hho7dZMAAASsElEQVRqWA3ObHUmZ7Y6k4/WfsTDMx7miflP8NTCpzix2YkM6TyE8486n5o1tMqJ7I205VZD7s6by99k2MxhvPDhCxjGBa0vYHCnwZzQ7ATMbLfGf3iTw3n4rIe5o9sdjJk3hodnPkzfSX3J2TeHa467hl8c+wua7NOkguZGRPYEneCuRjYXbOaJeU/Q/tH2dBnbhWnLpnHDCTfw2ZDP+Hufv3PiwSfudqKIalCnAb/O/TUfX/sx/9fv/zi8yeHcNOUmmj3YjIH/HMj7X79fYdMSkdTSkUU1sOr7VTwy6xEenfMoqzetps1+bXjsnMe4uO3F7JO5T8qnn1Ejg15H9KLXEb1476v3GDZjGE8tfIrH5z5Ot5bdGNJ5CGcfdjYZNTJSHouIlI+SRRU2I38GeTPy+Meif7C9cDvnHH4OQzoPoVvLbhV6BJGMo7OP5vFej3N397t5fM7jjJg1gnMnnsshjQ7h2k7Xcnn7y9m39r5piU1ESqdqqCpm2/ZtTHhvAsePOp7jRx/Pvz/+N4OOG8TH137M5Asnc9ohp6UtUUQ13acpN518044qsP3r7c+vX/k1Bz1wEINfGszHaz9Od4giEqEjiypi9cbVPDbnMf46+6+s+n4VhzU+jGE9hnFZu8uoX7t+usMrVWZGJn3b9KVvm77MXjWbvBl5jJw9kuEzh3PWYWcxpPMQuh/SvVIkOJHqTMliL7fgywXkzcjj6feeZsv2LZxx6Bk8/pPH6dGqBzVs7zpw7HhgR546/ynu634fI2ePZOSckZzxtzM4qulRDO48mP5t+1O3Vt10hylSLe1dexMBggf6Pb/4ebo82YV2j7bj7x/8nQHtBvDBrz7glUte4azDztrrEkXUAfUP4E9d/8SK61Yw9ryxZGVm8ct//5KcB3O44bUbWL5uebpDFKl2dGSxF1m3eR2j545m+KzhLFu3jOYNmvPn0//MFe2voFFWo3SHV+Fq16zNpcdcSv+2/Xl75dvkzcjjgekP8Jfpf+H8I89ncOfBnHzwyaqiEtkDlCz2Ah+u+ZBhM4YxdsFYNm3bxCnNT+EvZ/yFXkf0qhZ3RJsZJx18EicdfBIr1q/gr7P+ymNzHuPZxc/Sfv/2DO48mH4/7kedmnXSHapIlbX31lVUcYVeyEsfv8QNC2/gqBFHMXreaPq26cvcgXP5z2X/ofdRvatFoiju4AYHc0/3e8j/TT6PnvMoW7ZvYcD/DeDgBw/mlqm3sHbL2nSHKFIlVb+9TSW3YesGnpz/JA/PfJiP1n5Ek1pNuL3r7QzsMJAf1f1RusOrNPbJ3IeBHQbyi2N/wZTPppA3I4873ryDDMvgZxt/xpDOQzjuoOPSHaZIlaFkUUl8+u2nDJ85nNHzRvPdlu/odFAnxvcez36r9+P0U05Pd3iVlpnR/ZDudD+kO0u/WcqNz93I5CWTGf/eeHJzchnceTAXHHUBmRmZ6Q5VZK+maqg0cnemfjaV8yaeR6thrXh45sOcddhZTL9iOjOunMFFR19EZg3t5BLVqnErBrUaRP5v8snrkcfqTau58NkLaZnXkrveuos1m9akO0SRvZaSRRr8sO0HRs8dTbtH29FtXDfeXvk2vz/59ywbsowJF0zg+Jzj0x3iXm3f2vsyuPNglgxawj8v/CdH7XcUf3jjDzR7sBlXTr6ShV8tTHeIInsdVUPtQfnf5e+4kmftD2tpm92W0b1Gc+GPLyQrMyvd4VU5NawG5xx+Duccfg4ffP0BD898mHELxjF63mi6tOjCkM5D+MnhP9EDDEUSoGSRYu7Ou/nvkjcjj0mLJuE4vY7oxZDOQzi1+am6R2APafOjNow8ZyR3nXYXo+aOYvjM4Zz/9/Np2bAlgzoN4vL2l9OwTsN0hylSaakaKkW2bt/K+IXj6TyqMyeMOYGXl77Mdcdfx9Jrl/L8z56nS4suShRp0DirMTeceAOfDvmUf/z0H+Tsm8NvX/0tOQ/kMOjFQSxZsyTdIYpUSjqyqGBfb/yakbNH8sjsR/hyw5cc0eQIRpw1gkuPuZR6teqlOzwJ1axRkz6t+9CndR/mfjGXYTOG8fjc4JHpPVr1YEjnIZxx6Bl79WNTRCqSkkUFmffFPPJm5DHh/Qls3b6Vnq16MqTzEE4/9HTtcCq5Yw84lifPe5J7u9/Lo3Me5ZHZj9BzfE+OaHIEgzsPVqIXQdVQu6WgsIBnFz3LKU+cwrGPHcukRZP4xbG/4MNrPuTFi1/kzFZnKlHsRbLrZXPLqbew/LrlPHX+U9SvXZ9rXryGnAdyuP7V6/ns28/SHaJI2ujIohy++eEbRs0dxYhZI1ixfgUtGrbgL2f8RSdJq4haGbW4pO0lXHz0xTsuTnjo3Yd48N0HdXGCVFtKFklYtHoRw2YMY9yCcfxQ8ANdW3RlWI9hnHP4Obr8sgoyM3Kb5ZLbLHeny55f+PAFjsk+hsGdB3PR0RfpAYZSLaiOJEahF/Lvj/7NGU+dQZu/tmHsgrFcdPRFLLh6AW/8/A3OPfJcJYpqIGffHO467S5W/nolj//kcbb7dq6YfAXNHmzGzW/czOfffZ7uEEVSSsmiFN9v+Z5hM4ZxxPAjOGfCOSxavYg7u93Jyl+vZFSvUbTNbpvuECUNsjKzuPLYK1l49ULeuPQNTmx2Ine9dRct8lpw4bMX8m7+u+kOUSQlUloNZWY9gDwgAxjl7vcU698cGAPsB3wDXOLu+WG/7cB7YdEV7t4rlbEW+eSbT3h45sOMmTeG77d+T25OLnd0vYPeR/XWw+hkBzOja8uudG3ZdaeHQE58fyKdDurEkM5D6NO6D7UyaqU7VJEKkbJkYWYZwAjgdCAfmGVmk919UaTY/cA4dx9rZt2Au4H+Yb8f3L1dquKLemfFO9y35D5u/vRm3ln5DjVr1KRvm756zLUk5JBGh/DAmQ8wtOtQxs4fy7CZw7j4uYu5/tXr+dVxv6L9Ae15YcUL1F5Zm9xmuekOt9KbvnI641eM1/JK0J5aXqk8sugELHX3TwHMbCJwLhBNFq2B34TNU4EXUhhPiZ5b/Bx9numD4wAMaDeAO7vdyQH1D9jTocherl6telzT6Rp+edwveWXpK+TNyOOPU/+4o//oz0bTvGFz9sncJ41R7mzjxo3UXVQ33WHssGnbJpavW47jWl4JiC6v8SvHM+XSKSlLGKlMFgcBKyPt+UDnYmUWAL0JqqrOB+qbWRN3XwvUMbPZQAFwj7vvkkjMbCAwECA7O5tp06YlHeS/l/97R6KoQQ0y1mewZM4SllA5HvuwYcOGcs1XqimusmWRxY05N1J/c30mfT4JAMexrcZ+NfdLc3T/06hWI2p65bkocsXWFTu2Ry2veNHltaVgC2OmjmHLwVtSMzF3T8kH6ENwnqKovT8wvFiZA4HngHkECSMfaBj2Oyj8PgRYBhxa1vQ6dOjg5fHOinc8644sr3FbDc+6I8vfWfFOucaTKlOnTk13CCVSXInR+pUcLa/kVMTyAmZ7Avv0VF4N9TnQLNKeE3bbwd1XuXtvd28P/CHsti78/jz8/hSYBrRPRZC5zXKZcukULm95eUoP4aR60vqVHC2v5OzJ5ZXK46lZwGFm1pIgSfQDLooWMLOmwDfuXgjcRHBlFGbWCNjk7lvCMicC96Uq0NxmuWw5eItWTEkJrV/J0fJKzp5aXik7snD3AmAQ8AqwGHjG3T8ws6FmVnQZbBdgiZl9BGQDd4bdjwJmm9kCghPf9/jOV1GJiMgelNIzNe7+IvBisW63RJonAZNKGO4d4OhUxiYiIonTHdwiIhJLyUJERGIpWYiISCwlCxERiWXBPRl7PzNbDSzfjVE0BdZUUDgVSXElR3ElR3ElpyrG1dzdY2+TrzLJYneZ2Wx375juOIpTXMlRXMlRXMmpznGpGkpERGIpWYiISCwli/95LN0BlEJxJUdxJUdxJafaxqVzFiIiEktHFiIiEkvJQkREYlXrZGFmzcxsqpktMrMPzGxIumMCMLM6ZjbTzBaEcf0p3TFFmVmGmc0zs3+lO5YiZrbMzN4zs/nhGxYrBTNraGaTzOxDM1tsZpXiudtmdkS4rIo+35nZdZUgrl+H6/z7ZjbBzOqkOyYAMxsSxvRBupeTmY0xs6/N7P1It8Zm9pqZfRx+N6ro6VbrZEHwytbfuntr4HjgGjNrneaYALYA3dz9GKAd0MPMjk9zTFFDCB47X9l0dfd2lew6+DzgZXc/EjiGSrLc3H1JuKzaAR2ATcDz6YzJzA4CBgMd3f3HQAbBe3DSysx+DPwC6ETwG55jZq3SGNKTQI9i3W4Eprj7YcCUsL1CVetk4e5fuPvcsPl7gg35oPRGBeHbDjeErZnhp1JciWBmOcDZwKh0x1LZmVkD4BRgNIC7by16E2QlcxrwibvvzhMQKkpNIMvMagL7AKvSHA8E79eZ4e6bwvf0/Afona5g3P1N4Jtinc8FxobNY4HzKnq61TpZRJlZC4JXt85IbySBsKpnPvA18Jq7V4q4gIeAG4DCdAdSjAOvmtkcMxuY7mBCLYHVwBNhtd0oM6ub7qBK0A+YkO4gwlcp3w+sAL4A1rv7q+mNCoD3gZPNrImZ7QOcxc6vjK4Mst39i7D5S4KXyVUoJQvAzOoBzwLXuft36Y4HwN23h1UEOUCn8FA4rczsHOBrd5+T7lhKcJK7Hwv0JKhOPCXdARH8Sz4WeCR8z/xGUlA9sDvMrBbQC/hHJYilEcE/5JbAgUBdM7skvVGBuy8G7gVeBV4G5gPb0xpUGTy4H6LCayKqfbIws0yCRDHe3Z9LdzzFhdUWU9m1jjIdTgR6mdkyYCLQzcz+lt6QAuG/Utz9a4K6907pjQiAfCA/clQ4iSB5VCY9gbnu/lW6AwG6A5+5+2p33wY8B5yQ5pgAcPfR7t7B3U8BvgU+SndMxXxlZgcAhN9fV/QEqnWyMDMjqE9e7O4PpDueIma2n5k1DJuzgNOBD9MbFbj7Te6e4+4tCKou3nD3tP/zM7O6Zla/qBk4g6DqIK3c/UtgpZkdEXY6Dahs75K/kEpQBRVaARxvZvuE2+ZpVJILAszsR+H3wQTnK55Ob0S7mAz8PGz+OfB/FT2BlL6Dey9wItAfeC88PwDw+/Dd4el0ADDWzDIIEvoz7l5pLlOthLKB54P9CzWBp9395fSGtMO1wPiwuudTYECa49khTKynA1elOxYAd59hZpOAuQRXKs6j8jxe41kzawJsA65J54UKZjYB6AI0NbN84FbgHuAZM7uC4FUNfSt8unrch4iIxKnW1VAiIpIYJQsREYmlZCEiIrGULEREJJaShYiIxFKykL2emW0Iv1uY2UUVPO7fF2t/pyLHX9HM7DIzG57uOKTqUbKQqqQFkFSyCB9YV5adkoW7V4o7ilMlvLdHZBdKFlKV3EPwwLf54XsRMszsz2Y2y8wWmtlVAGbWxczeMrPJhHdUm9kL4UMIPyh6EKGZ3UPwBNT5ZjY+7FZ0FGPhuN8P36Pxs8i4p0XeYTE+vBt5J2GZey14b8lHZnZy2H2nIwMz+5eZdSmadjjND8zsdTPrFI7nUzPrFRl9s7D7x2Z2a2Rcl4TTm29mjxYlhnC8fzGzBUCleN+GVELuro8+e/UH2BB+dwH+Fek+ELg5bK4NzCZ4SF0Xgof6tYyUbRx+ZxE8KqRJdNwlTOsC4DWCdy5kEzyq4oBw3OsJHgBZA5hO8JDD4jFPA/4SNp8FvB42XwYMj5T7F9AlbHagZ9j8PMGD7TIJ3rEwPzL8F0CTyLx0JHjM9j+BzLDcX4FLI+Ptm+7fUZ/K/anuj/uQqu0MoK2Z9QnbGwCHAVuBme7+WaTsYDM7P2xuFpZbW8a4TwImuPt2goe4/Qc4DvguHHc+QPgYmRbAf0sYR9GDK+eEZeJsJXjqKcB7wBZ332Zm7xUb/jV3XxtO/7kw1gKClxzNCg90svjfw+a2EzxMU6RUShZSlRlwrbu/slPHoFpnY7H27kCuu28ys2nA7rzOc0ukeTulb2dbSihTwM7Vw9E4trl70fN5CouGd/fCYudeij/DxwmWxVh3v6mEODaHSU+kVDpnIVXJ90D9SPsrwC/Dx9BjZoeX8vKhBsC3YaI4kuAVu0W2FQ1fzFvAz8LzIvsRvBFvZgXMwzKgnZnVMLNmlO9R66db8E7mLII3pr1N8KrNPpGnpzY2s+YVEK9UEzqykKpkIbA9PFH7JMH7r1sAc8OTzKsp+XWTLwNXm9liYAnwbqTfY8BCM5vr7hdHuj9PcDJ4AcE/9xvc/csw2eyOt4HPCE68LyZ4AmuyZhJUK+UAf3P32QBmdjPB2wRrED49leAJpSKx9NRZERGJpWooERGJpWQhIiKxlCxERCSWkoWIiMRSshARkVhKFiIiEkvJQkREYv1/Fruh+gG6Nk8AAAAASUVORK5CYII=\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np\n",
+    "import sys\n",
+    "import os\n",
+    "from matplotlib import pyplot as plt\n",
+    "\n",
+    "# get accuracy and iteration number\n",
+    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
+    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
+    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
+    "\n",
+    "# get number of points\n",
+    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
+    "num_points = num_points_proxy[0]\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "iters = np.array(iters_proxy).reshape(num_points)\n",
+    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
+    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by iteration - transfer learn')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 40,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get loss\n",
+    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
+    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
+    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation loss by iteration - transfer learn')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Loss')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Train-single-model/.ipynb_checkpoints/MADlib-Keras-transfer-learning-v3-checkpoint.ipynb b/community-artifacts/Deep-learning/Train-single-model/.ipynb_checkpoints/MADlib-Keras-transfer-learning-v3-checkpoint.ipynb
new file mode 100644
index 0000000..6bc6e20
--- /dev/null
+++ b/community-artifacts/Deep-learning/Train-single-model/.ipynb_checkpoints/MADlib-Keras-transfer-learning-v3-checkpoint.ipynb
@@ -0,0 +1,832 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Transfer Learning Using Keras and MADlib\n",
+    "\n",
+    "This is a transfer learning example based on https://keras.io/examples/mnist_transfer_cnn/ \n",
+    "\n",
+    "To load images into tables we use the script called <em>madlib_image_loader.py</em> located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning which uses the Python Imaging Library so supports multiple formats http://www.pythonware.com/products/pil/\n",
+    "\n",
+    "## Table of contents\n",
+    "<a href=\"#import_libraries\">1. Import libraries</a>\n",
+    "\n",
+    "<a href=\"#load_and_prepare_data\">2. Load and prepare data</a>\n",
+    "\n",
+    "<a href=\"#image_preproc\">3. Call image preprocessor</a>\n",
+    "\n",
+    "<a href=\"#define_and_load_model\">4. Define and load model architecture</a>\n",
+    "\n",
+    "<a href=\"#train\">5. Train</a>\n",
+    "\n",
+    "<a href=\"#transfer_learning\">6. Transfer learning</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"import_libraries\"></a>\n",
+    "# 1.  Import libraries\n",
+    "From https://keras.io/examples/mnist_transfer_cnn/ import libraries and define some params"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from __future__ import print_function\n",
+    "\n",
+    "import datetime\n",
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.datasets import mnist\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense, Dropout, Activation, Flatten\n",
+    "from tensorflow.keras.layers import Conv2D, MaxPooling2D\n",
+    "from tensorflow.keras import backend as K\n",
+    "\n",
+    "now = datetime.datetime.now\n",
+    "\n",
+    "batch_size = 128\n",
+    "num_classes = 5\n",
+    "epochs = 5\n",
+    "\n",
+    "# input image dimensions\n",
+    "img_rows, img_cols = 28, 28\n",
+    "# number of convolutional filters to use\n",
+    "filters = 32\n",
+    "# size of pooling area for max pooling\n",
+    "pool_size = 2\n",
+    "# convolution kernel size\n",
+    "kernel_size = 3\n",
+    "\n",
+    "if K.image_data_format() == 'channels_first':\n",
+    "    input_shape = (1, img_rows, img_cols)\n",
+    "else:\n",
+    "    input_shape = (img_rows, img_cols, 1)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Others needed in this workbook"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_and_prepare_data\"></a>\n",
+    "# 2.  Load and prepare data\n",
+    "\n",
+    "First load MNIST data from Keras, consisting of 60,000 28x28 grayscale images of the 10 digits, along with a test set of 10,000 images."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Downloading data from https://storage.googleapis.com/tensorflow/tf-keras-datasets/mnist.npz\n",
+      "11493376/11490434 [==============================] - 2s 0us/step\n",
+      "11501568/11490434 [==============================] - 2s 0us/step\n",
+      "(4861, 28, 28)\n",
+      "(4861, 28, 28, 1)\n"
+     ]
+    }
+   ],
+   "source": [
+    "# the data, split between train and test sets\n",
+    "(x_train, y_train), (x_test, y_test) = mnist.load_data()\n",
+    "\n",
+    "# create two datasets one with digits below 5 and one with 5 and above\n",
+    "x_train_lt5 = x_train[y_train < 5]\n",
+    "y_train_lt5 = y_train[y_train < 5]\n",
+    "x_test_lt5 = x_test[y_test < 5]\n",
+    "y_test_lt5 = y_test[y_test < 5]\n",
+    "\n",
+    "x_train_gte5 = x_train[y_train >= 5]\n",
+    "y_train_gte5 = y_train[y_train >= 5] - 5\n",
+    "x_test_gte5 = x_test[y_test >= 5]\n",
+    "y_test_gte5 = y_test[y_test >= 5] - 5\n",
+    "\n",
+    "# reshape to match model architecture\n",
+    "print(x_test_gte5.shape)\n",
+    "x_train_lt5=x_train_lt5.reshape(len(x_train_lt5), *input_shape)\n",
+    "x_test_lt5 = x_test_lt5.reshape(len(x_test_lt5), *input_shape)\n",
+    "x_train_gte5=x_train_gte5.reshape(len(x_train_gte5), *input_shape)\n",
+    "x_test_gte5 = x_test_gte5.reshape(len(x_test_gte5), *input_shape)\n",
+    "print(x_test_gte5.shape)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load datasets into tables using image loader scripts called <em>madlib_image_loader.py</em> located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# MADlib tools directory\n",
+    "import sys\n",
+    "import os\n",
+    "madlib_site_dir = '/Users/fmcquillan/Documents/Product/MADlib/Demos/data'\n",
+    "sys.path.append(madlib_site_dir)\n",
+    "\n",
+    "# Import image loader module\n",
+    "from madlib_image_loader import ImageLoader, DbCredentials"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Specify database credentials, for connecting to db\n",
+    "#db_creds = DbCredentials(user='gpadmin',\n",
+    "#                         host='35.239.240.26',\n",
+    "#                         port='5432',\n",
+    "#                         password='')\n",
+    "\n",
+    "db_creds = DbCredentials(user='gpadmin',\n",
+    "                         host='localhost',\n",
+    "                         port='8000',\n",
+    "                         password='')"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Initialize ImageLoader (increase num_workers to run faster)\n",
+    "iloader = ImageLoader(num_workers=5, db_creds=db_creds)"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "MainProcess: Connected to madlib db.\n",
+      "Executing: CREATE TABLE train_lt5 (id SERIAL, x REAL[], y TEXT[])\n",
+      "CREATE TABLE\n",
+      "Created table train_lt5 in madlib db\n",
+      "Spawning 5 workers...\n",
+      "Initializing PoolWorker-1 [pid 84210]\n",
+      "PoolWorker-1: Created temporary directory /tmp/madlib_ZWAMbc3vz1\n",
+      "Initializing PoolWorker-2 [pid 84211]\n",
+      "PoolWorker-2: Created temporary directory /tmp/madlib_InjvRoDiNG\n",
+      "Initializing PoolWorker-3 [pid 84212]\n",
+      "PoolWorker-3: Created temporary directory /tmp/madlib_b8amy7agop\n",
+      "PoolWorker-5: Created temporary directory /tmp/madlib_r8iho70gsD\n",
+      "Initializing PoolWorker-4 [pid 84213]\n",
+      "PoolWorker-4: Created temporary directory /tmp/madlib_bXuGj0ZMjg\n",
+      "Initializing PoolWorker-5 [pid 84214]\n",
+      "PoolWorker-1: Connected to madlib db.\n",
+      "PoolWorker-4: Connected to madlib db.\n",
+      "PoolWorker-5: Connected to madlib db.\n",
+      "PoolWorker-2: Connected to madlib db.\n",
+      "PoolWorker-3: Connected to madlib db.\n",
+      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_ZWAMbc3vz1/train_lt50000.tmp\n",
+      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_InjvRoDiNG/train_lt50000.tmp\n",
+      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_r8iho70gsD/train_lt50000.tmp\n",
+      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_bXuGj0ZMjg/train_lt50000.tmp\n",
+      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_b8amy7agop/train_lt50000.tmp\n",
+      "PoolWorker-1: Removed temporary directory /tmp/madlib_ZWAMbc3vz1\n",
+      "\n",
+      "Error in PoolWorker-1 while loading images\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=17722)\n",
+      "CONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "\n",
+      "PoolWorker-1: Can't find temporary directory... exiting.\n",
+      "PoolWorker-2: Removed temporary directory /tmp/madlib_InjvRoDiNG\n",
+      "\n",
+      "Error in PoolWorker-2 while loading images\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=17726)\n",
+      "CONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "\n",
+      "PoolWorker-2: Can't find temporary directory... exiting.\n",
+      "PoolWorker-4: Removed temporary directory /tmp/madlib_bXuGj0ZMjg\n",
+      "PoolWorker-3: Removed temporary directory /tmp/madlib_b8amy7agop\n",
+      "PoolWorker-5: Removed temporary directory /tmp/madlib_r8iho70gsD\n",
+      "\n",
+      "Error in PoolWorker-3 while loading images\n",
+      "Error in PoolWorker-5 while loading images\n",
+      "Error in PoolWorker-4 while loading images\n",
+      "\n",
+      "\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=17738)\n",
+      "CONTEXT:  COPY train_lt5, line 2, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=17730)\n",
+      "CONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=17734)\n",
+      "CONTEXT:  COPY train_lt5, line 2, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "\n",
+      "\n",
+      "\n",
+      "PoolWorker-5: Can't find temporary directory... exiting.\n",
+      "PoolWorker-3: Can't find temporary directory... exiting.\n",
+      "PoolWorker-4: Can't find temporary directory... exiting.\n",
+      "PoolWorker-1: Can't find temporary directory... exiting.\n",
+      "PoolWorker-2: Can't find temporary directory... exiting.\n",
+      "PoolWorker-5: Can't find temporary directory... exiting.\n",
+      "PoolWorker-3: Can't find temporary directory... exiting.\n",
+      "PoolWorker-4: Can't find temporary directory... exiting.\n",
+      "PoolWorker-1: Can't find temporary directory... exiting.\n",
+      "PoolWorker-2: Can't find temporary directory... exiting.\n",
+      "PoolWorker-5: Can't find temporary directory... exiting.\n",
+      "PoolWorker-3: Can't find temporary directory... exiting.\n",
+      "PoolWorker-4: Can't find temporary directory... exiting.\n",
+      "PoolWorker-1: Can't find temporary directory... exiting.\n",
+      "PoolWorker-2: Can't find temporary directory... exiting.\n",
+      "PoolWorker-5: Can't find temporary directory... exiting.\n",
+      "PoolWorker-3: Can't find temporary directory... exiting.\n",
+      "PoolWorker-4: Can't find temporary directory... exiting.\n",
+      "PoolWorker-1: Can't find temporary directory... exiting.\n",
+      "5 workers terminated.\n"
+     ]
+    },
+    {
+     "ename": "BadCopyFileFormat",
+     "evalue": "array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=17722)\nCONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+     "output_type": "error",
+     "traceback": [
+      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
+      "\u001b[0;31mBadCopyFileFormat\u001b[0m                         Traceback (most recent call last)",
+      "\u001b[0;32m<ipython-input-10-3c25ba51b8fc>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m()\u001b[0m\n\u001b[1;32m      3\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      4\u001b[0m \u001b[0;31m# Save images to temporary directories and load into database\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 5\u001b[0;31m \u001b[0miloader\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload_dataset_from_np\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx_train_lt5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_train_lt5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'train_lt5'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mappend\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mFalse\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m      6\u001b[0m \u001b[0miloader\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload_dataset_from_np\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx_test_lt5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_test_lt5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'test_lt5'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mappend\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mFalse\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      7\u001b[0m \u001b[0miloader\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload_dataset_from_np\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx_train_gte5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_train_gte5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'train_gte5'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mappend\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mFalse\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.pyc\u001b[0m in \u001b[0;36mload_dataset_from_np\u001b[0;34m(self, data_x, data_y, table_name, append, label_datatype)\u001b[0m\n\u001b[1;32m    523\u001b[0m         \u001b[0;32mexcept\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mException\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    524\u001b[0m             \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mterminate_workers\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 525\u001b[0;31m             \u001b[0;32mraise\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    526\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    527\u001b[0m         \u001b[0mend_time\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtime\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtime\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;31mBadCopyFileFormat\u001b[0m: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=17722)\nCONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n"
+     ]
+    }
+   ],
+   "source": [
+    "# Drop tables\n",
+    "%sql DROP TABLE IF EXISTS train_lt5, test_lt5, train_gte5, test_gte5\n",
+    "\n",
+    "# Save images to temporary directories and load into database\n",
+    "iloader.load_dataset_from_np(x_train_lt5, y_train_lt5, 'train_lt5', append=False)\n",
+    "iloader.load_dataset_from_np(x_test_lt5, y_test_lt5, 'test_lt5', append=False)\n",
+    "iloader.load_dataset_from_np(x_train_gte5, y_train_gte5, 'train_gte5', append=False)\n",
+    "iloader.load_dataset_from_np(x_test_gte5, y_test_gte5, 'test_gte5', append=False)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"image_preproc\"></a>\n",
+    "# 3. Call image preprocessor\n",
+    "\n",
+    "Transforms from one image per row to multiple images per row for batch optimization.  Also normalizes and one-hot encodes.\n",
+    "\n",
+    "Training dataset < 5"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS train_lt5_packed, train_lt5_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('train_lt5',               -- Source table\n",
+    "                                       'train_lt5_packed',        -- Output table\n",
+    "                                       'y',                       -- Dependent variable\n",
+    "                                       'x',                       -- Independent variable\n",
+    "                                        1000,                     -- Buffer size\n",
+    "                                        255                       -- Normalizing constant\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM train_lt5_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Test dataset < 5"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS test_lt5_packed, test_lt5_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('test_lt5',                -- Source table\n",
+    "                                         'test_lt5_packed',         -- Output table\n",
+    "                                         'y',                       -- Dependent variable\n",
+    "                                         'x',                       -- Independent variable\n",
+    "                                         'train_lt5_packed'         -- Training preproc table\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM test_lt5_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Training dataset >= 5"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS train_gte5_packed, train_gte5_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('train_gte5',              -- Source table\n",
+    "                                       'train_gte5_packed',       -- Output table\n",
+    "                                       'y',                       -- Dependent variable\n",
+    "                                       'x',                       -- Independent variable\n",
+    "                                        1000,                     -- Buffer size\n",
+    "                                        255                       -- Normalizing constant\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM train_gte5_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Test dataset >= 5"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS test_gte5_packed, test_gte5_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('test_gte5',             -- Source table\n",
+    "                                         'test_gte5_packed',      -- Output table\n",
+    "                                         'y',                     -- Dependent variable\n",
+    "                                         'x',                     -- Independent variable\n",
+    "                                         'train_gte5_packed'      -- Training preproc table\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM test_gte5_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"define_and_load_model\"></a>\n",
+    "# 4. Define and load model architecture\n",
+    "\n",
+    "Model with feature and classification layers trainable"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# define two groups of layers: feature (convolutions) and classification (dense)\n",
+    "feature_layers = [\n",
+    "    Conv2D(filters, kernel_size,\n",
+    "           padding='valid',\n",
+    "           input_shape=input_shape),\n",
+    "    Activation('relu'),\n",
+    "    Conv2D(filters, kernel_size),\n",
+    "    Activation('relu'),\n",
+    "    MaxPooling2D(pool_size=pool_size),\n",
+    "    Dropout(0.25),\n",
+    "    Flatten(),\n",
+    "]\n",
+    "\n",
+    "classification_layers = [\n",
+    "    Dense(128),\n",
+    "    Activation('relu'),\n",
+    "    Dropout(0.5),\n",
+    "    Dense(num_classes),\n",
+    "    Activation('softmax')\n",
+    "]\n",
+    "\n",
+    "# create complete model\n",
+    "model = Sequential(feature_layers + classification_layers)\n",
+    "\n",
+    "model.summary()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into model architecture table using psycopg2"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "import psycopg2 as p2\n",
+    "#conn = p2.connect('postgresql://gpadmin@35.239.240.26:5432/madlib')\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
+    "cur = conn.cursor()\n",
+    "\n",
+    "%sql DROP TABLE IF EXISTS model_arch_library;\n",
+    "query = \"SELECT madlib.load_keras_model('model_arch_library', %s, NULL, %s)\"\n",
+    "cur.execute(query,[model.to_json(), \"feature + classification layers trainable\"])\n",
+    "conn.commit()\n",
+    "\n",
+    "# check model loaded OK\n",
+    "%sql SELECT model_id, name FROM model_arch_library;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Model with feature layers frozen"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# freeze feature layers\n",
+    "for l in feature_layers:\n",
+    "    l.trainable = False\n",
+    "\n",
+    "model.summary()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into transfer model architecture table using psycopg2"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "cur.execute(query,[model.to_json(), \"only classification layers trainable\"])\n",
+    "conn.commit()\n",
+    "\n",
+    "# check model loaded OK\n",
+    "%sql SELECT model_id, name FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train\"></a>\n",
+    "# 5.  Train\n",
+    "Train the model for 5-digit classification [0..4]  "
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mnist_model, mnist_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('train_lt5_packed',    -- source table\n",
+    "                               'mnist_model',         -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']$$,  -- compile_params\n",
+    "                                $$ batch_size=128, epochs=1 $$,  -- fit_params\n",
+    "                                5                     -- num_iterations\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM mnist_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Evaluate using test data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mnist_validate;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_evaluate('mnist_model',      -- model\n",
+    "                                   'test_lt5_packed',   -- test table\n",
+    "                                   'mnist_validate'     -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM mnist_validate;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"transfer_learning\"></a>\n",
+    "# 6. Transfer learning"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Use UPDATE to load trained weights from previous run into the model library table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "UPDATE model_arch_library\n",
+    "SET model_weights = mnist_model.model_weights\n",
+    "FROM mnist_model\n",
+    "WHERE model_arch_library.model_id = 2;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Transfer: train dense layers for new classification task [5..9]"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mnist_transfer_model, mnist_transfer_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('train_gte5_packed',   -- source table\n",
+    "                               'mnist_transfer_model',-- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                2,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']$$,  -- compile_params\n",
+    "                                $$ batch_size=128, epochs=1 $$,  -- fit_params\n",
+    "                                5                     -- num_iterations\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM mnist_transfer_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Evaluate using test data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mnist_transfer_validate;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_evaluate('mnist_transfer_model',      -- model\n",
+    "                                   'test_gte5_packed',           -- test table\n",
+    "                                   'mnist_transfer_validate'     -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM mnist_transfer_validate;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-MLP-v2.ipynb b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-MLP-v2.ipynb
new file mode 100644
index 0000000..b16a948
--- /dev/null
+++ b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-MLP-v2.ipynb
@@ -0,0 +1,5025 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Multilayer Perceptron Using Keras and MADlib\n",
+    "\n",
+    "E2E classification example using MADlib calling a Keras MLP.\n",
+    "\n",
+    "Deep learning works best on very large datasets, but that is not convenient for a quick introduction to the syntax.  So in this workbook we use the well known iris data set from https://archive.ics.uci.edu/ml/datasets/iris to help get you started.  It is similar to the example in user docs http://madlib.apache.org/docs/latest/index.html\n",
+    "\n",
+    "For more realistic examples with images please refer to the deep learning notebooks at\n",
+    "https://github.com/apache/madlib-site/tree/asf-site/community-artifacts\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#class\">Classification</a>\n",
+    "\n",
+    "* <a href=\"#create_input_data\">1. Create input data</a>\n",
+    "\n",
+    "* <a href=\"#pp\">2. Call preprocessor for deep learning</a>\n",
+    "\n",
+    "* <a href=\"#load\">3. Define and load model architecture</a>\n",
+    "\n",
+    "* <a href=\"#train\">4. Train</a>\n",
+    "\n",
+    "* <a href=\"#eval\">5. Evaluate</a>\n",
+    "\n",
+    "* <a href=\"#pred\">6. Predict</a>\n",
+    "\n",
+    "* <a href=\"#pred_byom\">7. Predict BYOM</a>\n",
+    "\n",
+    "<a href=\"#class2\">Classification with Other Parameters</a>\n",
+    "\n",
+    "* <a href=\"#val_dataset\">1. Validation dataset</a>\n",
+    "\n",
+    "* <a href=\"#pred_prob\">2. Predict probabilities</a>\n",
+    "\n",
+    "* <a href=\"#warm_start\">3. Warm start</a>\n",
+    "\n",
+    "<a href=\"#transfer_learn\">Transfer learning</a>\n",
+    "\n",
+    "* <a href=\"#load2\">1. Define and load model architecture with some layers frozen</a>\n",
+    "\n",
+    "* <a href=\"#train2\">2. Train transfer model</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {
+    "scrolled": false
+   },
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class\"></a>\n",
+    "# Classification\n",
+    "\n",
+    "<a id=\"create_input_data\"></a>\n",
+    "# 1.  Create input data\n",
+    "\n",
+    "Load iris data set."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "Done.\n",
+      "150 rows affected.\n",
+      "150 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>attributes</th>\n",
+       "        <th>class_text</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>4</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>5</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>6</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>7</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>8</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>10</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>11</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>13</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>14</td>\n",
+       "        <td>[Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>15</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>16</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>17</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>19</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>20</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>21</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>22</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>25</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>27</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>28</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>29</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>30</td>\n",
+       "        <td>[Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>32</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>34</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>35</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>36</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>37</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>39</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>40</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>41</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>42</td>\n",
+       "        <td>[Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>43</td>\n",
+       "        <td>[Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>44</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>45</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>46</td>\n",
+       "        <td>[Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>47</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>48</td>\n",
+       "        <td>[Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>49</td>\n",
+       "        <td>[Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>50</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')]</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>[Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>52</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>53</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>55</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>56</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>57</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>58</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>59</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>60</td>\n",
+       "        <td>[Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>61</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>62</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>63</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>64</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>65</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>67</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>68</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>69</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>70</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>71</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>72</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>75</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>76</td>\n",
+       "        <td>[Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>77</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>79</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>80</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>82</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>84</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>85</td>\n",
+       "        <td>[Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>86</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>87</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>88</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>89</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>91</td>\n",
+       "        <td>[Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>92</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>[Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>95</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>97</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>98</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>[Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>100</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')]</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>101</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>103</td>\n",
+       "        <td>[Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>104</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>105</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>106</td>\n",
+       "        <td>[Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>107</td>\n",
+       "        <td>[Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>108</td>\n",
+       "        <td>[Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>109</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>110</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>112</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>113</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>114</td>\n",
+       "        <td>[Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>115</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>116</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>118</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>119</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>121</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>122</td>\n",
+       "        <td>[Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>124</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>125</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>126</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>129</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>130</td>\n",
+       "        <td>[Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>[Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>132</td>\n",
+       "        <td>[Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>133</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>134</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>[Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>136</td>\n",
+       "        <td>[Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>137</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>138</td>\n",
+       "        <td>[Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>139</td>\n",
+       "        <td>[Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>140</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>141</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>142</td>\n",
+       "        <td>[Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>[Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>144</td>\n",
+       "        <td>[Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>146</td>\n",
+       "        <td>[Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>147</td>\n",
+       "        <td>[Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>148</td>\n",
+       "        <td>[Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>149</td>\n",
+       "        <td>[Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>150</td>\n",
+       "        <td>[Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')]</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (2, [Decimal('4.9'), Decimal('3.0'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (3, [Decimal('4.7'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (4, [Decimal('4.6'), Decimal('3.1'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (5, [Decimal('5.0'), Decimal('3.6'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (6, [Decimal('5.4'), Decimal('3.9'), Decimal('1.7'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (7, [Decimal('4.6'), Decimal('3.4'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (8, [Decimal('5.0'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (9, [Decimal('4.4'), Decimal('2.9'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (10, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (11, [Decimal('5.4'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (12, [Decimal('4.8'), Decimal('3.4'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (13, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (14, [Decimal('4.3'), Decimal('3.0'), Decimal('1.1'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (15, [Decimal('5.8'), Decimal('4.0'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (16, [Decimal('5.7'), Decimal('4.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (17, [Decimal('5.4'), Decimal('3.9'), Decimal('1.3'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (18, [Decimal('5.1'), Decimal('3.5'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (19, [Decimal('5.7'), Decimal('3.8'), Decimal('1.7'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (20, [Decimal('5.1'), Decimal('3.8'), Decimal('1.5'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (21, [Decimal('5.4'), Decimal('3.4'), Decimal('1.7'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (22, [Decimal('5.1'), Decimal('3.7'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (23, [Decimal('4.6'), Decimal('3.6'), Decimal('1.0'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (24, [Decimal('5.1'), Decimal('3.3'), Decimal('1.7'), Decimal('0.5')], u'Iris-setosa'),\n",
+       " (25, [Decimal('4.8'), Decimal('3.4'), Decimal('1.9'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (26, [Decimal('5.0'), Decimal('3.0'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (27, [Decimal('5.0'), Decimal('3.4'), Decimal('1.6'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (28, [Decimal('5.2'), Decimal('3.5'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (29, [Decimal('5.2'), Decimal('3.4'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (30, [Decimal('4.7'), Decimal('3.2'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (31, [Decimal('4.8'), Decimal('3.1'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (32, [Decimal('5.4'), Decimal('3.4'), Decimal('1.5'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (33, [Decimal('5.2'), Decimal('4.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (34, [Decimal('5.5'), Decimal('4.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (35, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (36, [Decimal('5.0'), Decimal('3.2'), Decimal('1.2'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (37, [Decimal('5.5'), Decimal('3.5'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (38, [Decimal('4.9'), Decimal('3.1'), Decimal('1.5'), Decimal('0.1')], u'Iris-setosa'),\n",
+       " (39, [Decimal('4.4'), Decimal('3.0'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (40, [Decimal('5.1'), Decimal('3.4'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (41, [Decimal('5.0'), Decimal('3.5'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (42, [Decimal('4.5'), Decimal('2.3'), Decimal('1.3'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (43, [Decimal('4.4'), Decimal('3.2'), Decimal('1.3'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (44, [Decimal('5.0'), Decimal('3.5'), Decimal('1.6'), Decimal('0.6')], u'Iris-setosa'),\n",
+       " (45, [Decimal('5.1'), Decimal('3.8'), Decimal('1.9'), Decimal('0.4')], u'Iris-setosa'),\n",
+       " (46, [Decimal('4.8'), Decimal('3.0'), Decimal('1.4'), Decimal('0.3')], u'Iris-setosa'),\n",
+       " (47, [Decimal('5.1'), Decimal('3.8'), Decimal('1.6'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (48, [Decimal('4.6'), Decimal('3.2'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (49, [Decimal('5.3'), Decimal('3.7'), Decimal('1.5'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (50, [Decimal('5.0'), Decimal('3.3'), Decimal('1.4'), Decimal('0.2')], u'Iris-setosa'),\n",
+       " (51, [Decimal('7.0'), Decimal('3.2'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (52, [Decimal('6.4'), Decimal('3.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (53, [Decimal('6.9'), Decimal('3.1'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (54, [Decimal('5.5'), Decimal('2.3'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (55, [Decimal('6.5'), Decimal('2.8'), Decimal('4.6'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (56, [Decimal('5.7'), Decimal('2.8'), Decimal('4.5'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (57, [Decimal('6.3'), Decimal('3.3'), Decimal('4.7'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (58, [Decimal('4.9'), Decimal('2.4'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (59, [Decimal('6.6'), Decimal('2.9'), Decimal('4.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (60, [Decimal('5.2'), Decimal('2.7'), Decimal('3.9'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (61, [Decimal('5.0'), Decimal('2.0'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (62, [Decimal('5.9'), Decimal('3.0'), Decimal('4.2'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (63, [Decimal('6.0'), Decimal('2.2'), Decimal('4.0'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (64, [Decimal('6.1'), Decimal('2.9'), Decimal('4.7'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (65, [Decimal('5.6'), Decimal('2.9'), Decimal('3.6'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (66, [Decimal('6.7'), Decimal('3.1'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (67, [Decimal('5.6'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (68, [Decimal('5.8'), Decimal('2.7'), Decimal('4.1'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (69, [Decimal('6.2'), Decimal('2.2'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (70, [Decimal('5.6'), Decimal('2.5'), Decimal('3.9'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (71, [Decimal('5.9'), Decimal('3.2'), Decimal('4.8'), Decimal('1.8')], u'Iris-versicolor'),\n",
+       " (72, [Decimal('6.1'), Decimal('2.8'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (73, [Decimal('6.3'), Decimal('2.5'), Decimal('4.9'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (74, [Decimal('6.1'), Decimal('2.8'), Decimal('4.7'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (75, [Decimal('6.4'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (76, [Decimal('6.6'), Decimal('3.0'), Decimal('4.4'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (77, [Decimal('6.8'), Decimal('2.8'), Decimal('4.8'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (78, [Decimal('6.7'), Decimal('3.0'), Decimal('5.0'), Decimal('1.7')], u'Iris-versicolor'),\n",
+       " (79, [Decimal('6.0'), Decimal('2.9'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (80, [Decimal('5.7'), Decimal('2.6'), Decimal('3.5'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (81, [Decimal('5.5'), Decimal('2.4'), Decimal('3.8'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (82, [Decimal('5.5'), Decimal('2.4'), Decimal('3.7'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (83, [Decimal('5.8'), Decimal('2.7'), Decimal('3.9'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (84, [Decimal('6.0'), Decimal('2.7'), Decimal('5.1'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (85, [Decimal('5.4'), Decimal('3.0'), Decimal('4.5'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (86, [Decimal('6.0'), Decimal('3.4'), Decimal('4.5'), Decimal('1.6')], u'Iris-versicolor'),\n",
+       " (87, [Decimal('6.7'), Decimal('3.1'), Decimal('4.7'), Decimal('1.5')], u'Iris-versicolor'),\n",
+       " (88, [Decimal('6.3'), Decimal('2.3'), Decimal('4.4'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (89, [Decimal('5.6'), Decimal('3.0'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (90, [Decimal('5.5'), Decimal('2.5'), Decimal('4.0'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (91, [Decimal('5.5'), Decimal('2.6'), Decimal('4.4'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (92, [Decimal('6.1'), Decimal('3.0'), Decimal('4.6'), Decimal('1.4')], u'Iris-versicolor'),\n",
+       " (93, [Decimal('5.8'), Decimal('2.6'), Decimal('4.0'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (94, [Decimal('5.0'), Decimal('2.3'), Decimal('3.3'), Decimal('1.0')], u'Iris-versicolor'),\n",
+       " (95, [Decimal('5.6'), Decimal('2.7'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (96, [Decimal('5.7'), Decimal('3.0'), Decimal('4.2'), Decimal('1.2')], u'Iris-versicolor'),\n",
+       " (97, [Decimal('5.7'), Decimal('2.9'), Decimal('4.2'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (98, [Decimal('6.2'), Decimal('2.9'), Decimal('4.3'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (99, [Decimal('5.1'), Decimal('2.5'), Decimal('3.0'), Decimal('1.1')], u'Iris-versicolor'),\n",
+       " (100, [Decimal('5.7'), Decimal('2.8'), Decimal('4.1'), Decimal('1.3')], u'Iris-versicolor'),\n",
+       " (101, [Decimal('6.3'), Decimal('3.3'), Decimal('6.0'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (102, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (103, [Decimal('7.1'), Decimal('3.0'), Decimal('5.9'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (104, [Decimal('6.3'), Decimal('2.9'), Decimal('5.6'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (105, [Decimal('6.5'), Decimal('3.0'), Decimal('5.8'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (106, [Decimal('7.6'), Decimal('3.0'), Decimal('6.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (107, [Decimal('4.9'), Decimal('2.5'), Decimal('4.5'), Decimal('1.7')], u'Iris-virginica'),\n",
+       " (108, [Decimal('7.3'), Decimal('2.9'), Decimal('6.3'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (109, [Decimal('6.7'), Decimal('2.5'), Decimal('5.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (110, [Decimal('7.2'), Decimal('3.6'), Decimal('6.1'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (111, [Decimal('6.5'), Decimal('3.2'), Decimal('5.1'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (112, [Decimal('6.4'), Decimal('2.7'), Decimal('5.3'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (113, [Decimal('6.8'), Decimal('3.0'), Decimal('5.5'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (114, [Decimal('5.7'), Decimal('2.5'), Decimal('5.0'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (115, [Decimal('5.8'), Decimal('2.8'), Decimal('5.1'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (116, [Decimal('6.4'), Decimal('3.2'), Decimal('5.3'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (117, [Decimal('6.5'), Decimal('3.0'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (118, [Decimal('7.7'), Decimal('3.8'), Decimal('6.7'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (119, [Decimal('7.7'), Decimal('2.6'), Decimal('6.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (120, [Decimal('6.0'), Decimal('2.2'), Decimal('5.0'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (121, [Decimal('6.9'), Decimal('3.2'), Decimal('5.7'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (122, [Decimal('5.6'), Decimal('2.8'), Decimal('4.9'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (123, [Decimal('7.7'), Decimal('2.8'), Decimal('6.7'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (124, [Decimal('6.3'), Decimal('2.7'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (125, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (126, [Decimal('7.2'), Decimal('3.2'), Decimal('6.0'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (127, [Decimal('6.2'), Decimal('2.8'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (128, [Decimal('6.1'), Decimal('3.0'), Decimal('4.9'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (129, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (130, [Decimal('7.2'), Decimal('3.0'), Decimal('5.8'), Decimal('1.6')], u'Iris-virginica'),\n",
+       " (131, [Decimal('7.4'), Decimal('2.8'), Decimal('6.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (132, [Decimal('7.9'), Decimal('3.8'), Decimal('6.4'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (133, [Decimal('6.4'), Decimal('2.8'), Decimal('5.6'), Decimal('2.2')], u'Iris-virginica'),\n",
+       " (134, [Decimal('6.3'), Decimal('2.8'), Decimal('5.1'), Decimal('1.5')], u'Iris-virginica'),\n",
+       " (135, [Decimal('6.1'), Decimal('2.6'), Decimal('5.6'), Decimal('1.4')], u'Iris-virginica'),\n",
+       " (136, [Decimal('7.7'), Decimal('3.0'), Decimal('6.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (137, [Decimal('6.3'), Decimal('3.4'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (138, [Decimal('6.4'), Decimal('3.1'), Decimal('5.5'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (139, [Decimal('6.0'), Decimal('3.0'), Decimal('4.8'), Decimal('1.8')], u'Iris-virginica'),\n",
+       " (140, [Decimal('6.9'), Decimal('3.1'), Decimal('5.4'), Decimal('2.1')], u'Iris-virginica'),\n",
+       " (141, [Decimal('6.7'), Decimal('3.1'), Decimal('5.6'), Decimal('2.4')], u'Iris-virginica'),\n",
+       " (142, [Decimal('6.9'), Decimal('3.1'), Decimal('5.1'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (143, [Decimal('5.8'), Decimal('2.7'), Decimal('5.1'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (144, [Decimal('6.8'), Decimal('3.2'), Decimal('5.9'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (145, [Decimal('6.7'), Decimal('3.3'), Decimal('5.7'), Decimal('2.5')], u'Iris-virginica'),\n",
+       " (146, [Decimal('6.7'), Decimal('3.0'), Decimal('5.2'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (147, [Decimal('6.3'), Decimal('2.5'), Decimal('5.0'), Decimal('1.9')], u'Iris-virginica'),\n",
+       " (148, [Decimal('6.5'), Decimal('3.0'), Decimal('5.2'), Decimal('2.0')], u'Iris-virginica'),\n",
+       " (149, [Decimal('6.2'), Decimal('3.4'), Decimal('5.4'), Decimal('2.3')], u'Iris-virginica'),\n",
+       " (150, [Decimal('5.9'), Decimal('3.0'), Decimal('5.1'), Decimal('1.8')], u'Iris-virginica')]"
+      ]
+     },
+     "execution_count": 4,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql \n",
+    "DROP TABLE IF EXISTS iris_data;\n",
+    "\n",
+    "CREATE TABLE iris_data(\n",
+    "    id serial,\n",
+    "    attributes numeric[],\n",
+    "    class_text varchar\n",
+    ");\n",
+    "\n",
+    "INSERT INTO iris_data(id, attributes, class_text) VALUES\n",
+    "(1,ARRAY[5.1,3.5,1.4,0.2],'Iris-setosa'),\n",
+    "(2,ARRAY[4.9,3.0,1.4,0.2],'Iris-setosa'),\n",
+    "(3,ARRAY[4.7,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(4,ARRAY[4.6,3.1,1.5,0.2],'Iris-setosa'),\n",
+    "(5,ARRAY[5.0,3.6,1.4,0.2],'Iris-setosa'),\n",
+    "(6,ARRAY[5.4,3.9,1.7,0.4],'Iris-setosa'),\n",
+    "(7,ARRAY[4.6,3.4,1.4,0.3],'Iris-setosa'),\n",
+    "(8,ARRAY[5.0,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(9,ARRAY[4.4,2.9,1.4,0.2],'Iris-setosa'),\n",
+    "(10,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(11,ARRAY[5.4,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(12,ARRAY[4.8,3.4,1.6,0.2],'Iris-setosa'),\n",
+    "(13,ARRAY[4.8,3.0,1.4,0.1],'Iris-setosa'),\n",
+    "(14,ARRAY[4.3,3.0,1.1,0.1],'Iris-setosa'),\n",
+    "(15,ARRAY[5.8,4.0,1.2,0.2],'Iris-setosa'),\n",
+    "(16,ARRAY[5.7,4.4,1.5,0.4],'Iris-setosa'),\n",
+    "(17,ARRAY[5.4,3.9,1.3,0.4],'Iris-setosa'),\n",
+    "(18,ARRAY[5.1,3.5,1.4,0.3],'Iris-setosa'),\n",
+    "(19,ARRAY[5.7,3.8,1.7,0.3],'Iris-setosa'),\n",
+    "(20,ARRAY[5.1,3.8,1.5,0.3],'Iris-setosa'),\n",
+    "(21,ARRAY[5.4,3.4,1.7,0.2],'Iris-setosa'),\n",
+    "(22,ARRAY[5.1,3.7,1.5,0.4],'Iris-setosa'),\n",
+    "(23,ARRAY[4.6,3.6,1.0,0.2],'Iris-setosa'),\n",
+    "(24,ARRAY[5.1,3.3,1.7,0.5],'Iris-setosa'),\n",
+    "(25,ARRAY[4.8,3.4,1.9,0.2],'Iris-setosa'),\n",
+    "(26,ARRAY[5.0,3.0,1.6,0.2],'Iris-setosa'),\n",
+    "(27,ARRAY[5.0,3.4,1.6,0.4],'Iris-setosa'),\n",
+    "(28,ARRAY[5.2,3.5,1.5,0.2],'Iris-setosa'),\n",
+    "(29,ARRAY[5.2,3.4,1.4,0.2],'Iris-setosa'),\n",
+    "(30,ARRAY[4.7,3.2,1.6,0.2],'Iris-setosa'),\n",
+    "(31,ARRAY[4.8,3.1,1.6,0.2],'Iris-setosa'),\n",
+    "(32,ARRAY[5.4,3.4,1.5,0.4],'Iris-setosa'),\n",
+    "(33,ARRAY[5.2,4.1,1.5,0.1],'Iris-setosa'),\n",
+    "(34,ARRAY[5.5,4.2,1.4,0.2],'Iris-setosa'),\n",
+    "(35,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(36,ARRAY[5.0,3.2,1.2,0.2],'Iris-setosa'),\n",
+    "(37,ARRAY[5.5,3.5,1.3,0.2],'Iris-setosa'),\n",
+    "(38,ARRAY[4.9,3.1,1.5,0.1],'Iris-setosa'),\n",
+    "(39,ARRAY[4.4,3.0,1.3,0.2],'Iris-setosa'),\n",
+    "(40,ARRAY[5.1,3.4,1.5,0.2],'Iris-setosa'),\n",
+    "(41,ARRAY[5.0,3.5,1.3,0.3],'Iris-setosa'),\n",
+    "(42,ARRAY[4.5,2.3,1.3,0.3],'Iris-setosa'),\n",
+    "(43,ARRAY[4.4,3.2,1.3,0.2],'Iris-setosa'),\n",
+    "(44,ARRAY[5.0,3.5,1.6,0.6],'Iris-setosa'),\n",
+    "(45,ARRAY[5.1,3.8,1.9,0.4],'Iris-setosa'),\n",
+    "(46,ARRAY[4.8,3.0,1.4,0.3],'Iris-setosa'),\n",
+    "(47,ARRAY[5.1,3.8,1.6,0.2],'Iris-setosa'),\n",
+    "(48,ARRAY[4.6,3.2,1.4,0.2],'Iris-setosa'),\n",
+    "(49,ARRAY[5.3,3.7,1.5,0.2],'Iris-setosa'),\n",
+    "(50,ARRAY[5.0,3.3,1.4,0.2],'Iris-setosa'),\n",
+    "(51,ARRAY[7.0,3.2,4.7,1.4],'Iris-versicolor'),\n",
+    "(52,ARRAY[6.4,3.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(53,ARRAY[6.9,3.1,4.9,1.5],'Iris-versicolor'),\n",
+    "(54,ARRAY[5.5,2.3,4.0,1.3],'Iris-versicolor'),\n",
+    "(55,ARRAY[6.5,2.8,4.6,1.5],'Iris-versicolor'),\n",
+    "(56,ARRAY[5.7,2.8,4.5,1.3],'Iris-versicolor'),\n",
+    "(57,ARRAY[6.3,3.3,4.7,1.6],'Iris-versicolor'),\n",
+    "(58,ARRAY[4.9,2.4,3.3,1.0],'Iris-versicolor'),\n",
+    "(59,ARRAY[6.6,2.9,4.6,1.3],'Iris-versicolor'),\n",
+    "(60,ARRAY[5.2,2.7,3.9,1.4],'Iris-versicolor'),\n",
+    "(61,ARRAY[5.0,2.0,3.5,1.0],'Iris-versicolor'),\n",
+    "(62,ARRAY[5.9,3.0,4.2,1.5],'Iris-versicolor'),\n",
+    "(63,ARRAY[6.0,2.2,4.0,1.0],'Iris-versicolor'),\n",
+    "(64,ARRAY[6.1,2.9,4.7,1.4],'Iris-versicolor'),\n",
+    "(65,ARRAY[5.6,2.9,3.6,1.3],'Iris-versicolor'),\n",
+    "(66,ARRAY[6.7,3.1,4.4,1.4],'Iris-versicolor'),\n",
+    "(67,ARRAY[5.6,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(68,ARRAY[5.8,2.7,4.1,1.0],'Iris-versicolor'),\n",
+    "(69,ARRAY[6.2,2.2,4.5,1.5],'Iris-versicolor'),\n",
+    "(70,ARRAY[5.6,2.5,3.9,1.1],'Iris-versicolor'),\n",
+    "(71,ARRAY[5.9,3.2,4.8,1.8],'Iris-versicolor'),\n",
+    "(72,ARRAY[6.1,2.8,4.0,1.3],'Iris-versicolor'),\n",
+    "(73,ARRAY[6.3,2.5,4.9,1.5],'Iris-versicolor'),\n",
+    "(74,ARRAY[6.1,2.8,4.7,1.2],'Iris-versicolor'),\n",
+    "(75,ARRAY[6.4,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(76,ARRAY[6.6,3.0,4.4,1.4],'Iris-versicolor'),\n",
+    "(77,ARRAY[6.8,2.8,4.8,1.4],'Iris-versicolor'),\n",
+    "(78,ARRAY[6.7,3.0,5.0,1.7],'Iris-versicolor'),\n",
+    "(79,ARRAY[6.0,2.9,4.5,1.5],'Iris-versicolor'),\n",
+    "(80,ARRAY[5.7,2.6,3.5,1.0],'Iris-versicolor'),\n",
+    "(81,ARRAY[5.5,2.4,3.8,1.1],'Iris-versicolor'),\n",
+    "(82,ARRAY[5.5,2.4,3.7,1.0],'Iris-versicolor'),\n",
+    "(83,ARRAY[5.8,2.7,3.9,1.2],'Iris-versicolor'),\n",
+    "(84,ARRAY[6.0,2.7,5.1,1.6],'Iris-versicolor'),\n",
+    "(85,ARRAY[5.4,3.0,4.5,1.5],'Iris-versicolor'),\n",
+    "(86,ARRAY[6.0,3.4,4.5,1.6],'Iris-versicolor'),\n",
+    "(87,ARRAY[6.7,3.1,4.7,1.5],'Iris-versicolor'),\n",
+    "(88,ARRAY[6.3,2.3,4.4,1.3],'Iris-versicolor'),\n",
+    "(89,ARRAY[5.6,3.0,4.1,1.3],'Iris-versicolor'),\n",
+    "(90,ARRAY[5.5,2.5,4.0,1.3],'Iris-versicolor'),\n",
+    "(91,ARRAY[5.5,2.6,4.4,1.2],'Iris-versicolor'),\n",
+    "(92,ARRAY[6.1,3.0,4.6,1.4],'Iris-versicolor'),\n",
+    "(93,ARRAY[5.8,2.6,4.0,1.2],'Iris-versicolor'),\n",
+    "(94,ARRAY[5.0,2.3,3.3,1.0],'Iris-versicolor'),\n",
+    "(95,ARRAY[5.6,2.7,4.2,1.3],'Iris-versicolor'),\n",
+    "(96,ARRAY[5.7,3.0,4.2,1.2],'Iris-versicolor'),\n",
+    "(97,ARRAY[5.7,2.9,4.2,1.3],'Iris-versicolor'),\n",
+    "(98,ARRAY[6.2,2.9,4.3,1.3],'Iris-versicolor'),\n",
+    "(99,ARRAY[5.1,2.5,3.0,1.1],'Iris-versicolor'),\n",
+    "(100,ARRAY[5.7,2.8,4.1,1.3],'Iris-versicolor'),\n",
+    "(101,ARRAY[6.3,3.3,6.0,2.5],'Iris-virginica'),\n",
+    "(102,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(103,ARRAY[7.1,3.0,5.9,2.1],'Iris-virginica'),\n",
+    "(104,ARRAY[6.3,2.9,5.6,1.8],'Iris-virginica'),\n",
+    "(105,ARRAY[6.5,3.0,5.8,2.2],'Iris-virginica'),\n",
+    "(106,ARRAY[7.6,3.0,6.6,2.1],'Iris-virginica'),\n",
+    "(107,ARRAY[4.9,2.5,4.5,1.7],'Iris-virginica'),\n",
+    "(108,ARRAY[7.3,2.9,6.3,1.8],'Iris-virginica'),\n",
+    "(109,ARRAY[6.7,2.5,5.8,1.8],'Iris-virginica'),\n",
+    "(110,ARRAY[7.2,3.6,6.1,2.5],'Iris-virginica'),\n",
+    "(111,ARRAY[6.5,3.2,5.1,2.0],'Iris-virginica'),\n",
+    "(112,ARRAY[6.4,2.7,5.3,1.9],'Iris-virginica'),\n",
+    "(113,ARRAY[6.8,3.0,5.5,2.1],'Iris-virginica'),\n",
+    "(114,ARRAY[5.7,2.5,5.0,2.0],'Iris-virginica'),\n",
+    "(115,ARRAY[5.8,2.8,5.1,2.4],'Iris-virginica'),\n",
+    "(116,ARRAY[6.4,3.2,5.3,2.3],'Iris-virginica'),\n",
+    "(117,ARRAY[6.5,3.0,5.5,1.8],'Iris-virginica'),\n",
+    "(118,ARRAY[7.7,3.8,6.7,2.2],'Iris-virginica'),\n",
+    "(119,ARRAY[7.7,2.6,6.9,2.3],'Iris-virginica'),\n",
+    "(120,ARRAY[6.0,2.2,5.0,1.5],'Iris-virginica'),\n",
+    "(121,ARRAY[6.9,3.2,5.7,2.3],'Iris-virginica'),\n",
+    "(122,ARRAY[5.6,2.8,4.9,2.0],'Iris-virginica'),\n",
+    "(123,ARRAY[7.7,2.8,6.7,2.0],'Iris-virginica'),\n",
+    "(124,ARRAY[6.3,2.7,4.9,1.8],'Iris-virginica'),\n",
+    "(125,ARRAY[6.7,3.3,5.7,2.1],'Iris-virginica'),\n",
+    "(126,ARRAY[7.2,3.2,6.0,1.8],'Iris-virginica'),\n",
+    "(127,ARRAY[6.2,2.8,4.8,1.8],'Iris-virginica'),\n",
+    "(128,ARRAY[6.1,3.0,4.9,1.8],'Iris-virginica'),\n",
+    "(129,ARRAY[6.4,2.8,5.6,2.1],'Iris-virginica'),\n",
+    "(130,ARRAY[7.2,3.0,5.8,1.6],'Iris-virginica'),\n",
+    "(131,ARRAY[7.4,2.8,6.1,1.9],'Iris-virginica'),\n",
+    "(132,ARRAY[7.9,3.8,6.4,2.0],'Iris-virginica'),\n",
+    "(133,ARRAY[6.4,2.8,5.6,2.2],'Iris-virginica'),\n",
+    "(134,ARRAY[6.3,2.8,5.1,1.5],'Iris-virginica'),\n",
+    "(135,ARRAY[6.1,2.6,5.6,1.4],'Iris-virginica'),\n",
+    "(136,ARRAY[7.7,3.0,6.1,2.3],'Iris-virginica'),\n",
+    "(137,ARRAY[6.3,3.4,5.6,2.4],'Iris-virginica'),\n",
+    "(138,ARRAY[6.4,3.1,5.5,1.8],'Iris-virginica'),\n",
+    "(139,ARRAY[6.0,3.0,4.8,1.8],'Iris-virginica'),\n",
+    "(140,ARRAY[6.9,3.1,5.4,2.1],'Iris-virginica'),\n",
+    "(141,ARRAY[6.7,3.1,5.6,2.4],'Iris-virginica'),\n",
+    "(142,ARRAY[6.9,3.1,5.1,2.3],'Iris-virginica'),\n",
+    "(143,ARRAY[5.8,2.7,5.1,1.9],'Iris-virginica'),\n",
+    "(144,ARRAY[6.8,3.2,5.9,2.3],'Iris-virginica'),\n",
+    "(145,ARRAY[6.7,3.3,5.7,2.5],'Iris-virginica'),\n",
+    "(146,ARRAY[6.7,3.0,5.2,2.3],'Iris-virginica'),\n",
+    "(147,ARRAY[6.3,2.5,5.0,1.9],'Iris-virginica'),\n",
+    "(148,ARRAY[6.5,3.0,5.2,2.0],'Iris-virginica'),\n",
+    "(149,ARRAY[6.2,3.4,5.4,2.3],'Iris-virginica'),\n",
+    "(150,ARRAY[5.9,3.0,5.1,1.8],'Iris-virginica');\n",
+    "\n",
+    "SELECT * FROM iris_data ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Create a test/validation dataset from the training data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>120</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(120L,)]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train, iris_test;\n",
+    "\n",
+    "-- Set seed so results are reproducible\n",
+    "SELECT setseed(0);\n",
+    "\n",
+    "SELECT madlib.train_test_split('iris_data',     -- Source table\n",
+    "                               'iris',          -- Output table root name\n",
+    "                                0.8,            -- Train proportion\n",
+    "                                NULL,           -- Test proportion (0.2)\n",
+    "                                NULL,           -- Strata definition\n",
+    "                                NULL,           -- Output all columns\n",
+    "                                NULL,           -- Sample without replacement\n",
+    "                                TRUE            -- Separate output tables\n",
+    "                              );\n",
+    "\n",
+    "SELECT COUNT(*) FROM iris_train;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pp\"></a>\n",
+    "# 2. Call preprocessor for deep learning\n",
+    "Training dataset (uses training preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train</td>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>60</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train', u'iris_train_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 60, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_train_packed, iris_train_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('iris_train',         -- Source table\n",
+    "                                       'iris_train_packed',  -- Output table\n",
+    "                                       'class_text',         -- Dependent variable\n",
+    "                                       'attributes'          -- Independent variable\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM iris_train_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Validation dataset (uses validation preprocessor):"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>output_table</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "        <th>buffer_size</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>distribution_rules</th>\n",
+       "        <th>__internal_gpu_config__</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_test</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "        <td>15</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>all_segments</td>\n",
+       "        <td>all_segments</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_test', u'iris_test_packed', [u'class_text'], [u'attributes'], [u'character varying'], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'], 15, 1.0, [3], 'all_segments', 'all_segments')]"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_test_packed, iris_test_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('iris_test',          -- Source table\n",
+    "                                         'iris_test_packed',   -- Output table\n",
+    "                                         'class_text',         -- Dependent variable\n",
+    "                                         'attributes',         -- Independent variable\n",
+    "                                         'iris_train_packed'   -- From training preprocessor step\n",
+    "                                          ); \n",
+    "\n",
+    "SELECT * FROM iris_test_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load\"></a>\n",
+    "# 3. Define and load model architecture\n",
+    "Import Keras libraries"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Define model architecture"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "WARNING:tensorflow:From /Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/tensorflow/python/ops/init_ops.py:1251: calling __init__ (from tensorflow.python.ops.init_ops) with dtype is deprecated and will be removed in a future version.\n",
+      "Instructions for updating:\n",
+      "Call initializer instance with the dtype argument instead of passing it to the constructor\n",
+      "Model: \"sequential\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense (Dense)                (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_1 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_2 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 193\n",
+      "Non-trainable params: 0\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model_simple = Sequential()\n",
+    "model_simple.add(Dense(10, activation='relu', input_shape=(4,)))\n",
+    "model_simple.add(Dense(10, activation='relu'))\n",
+    "model_simple.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model_simple.summary();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model_simple.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Sophie', u'A simple model')]"
+      ]
+     },
+     "execution_count": 11,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS model_arch_library;\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,\n",
+    "                               \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Sophie',              -- Name\n",
+    "                               'A simple model'       -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT model_id, model_arch, name, description FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train\"></a>\n",
+    "# 4.  Train\n",
+    "Train the model:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 12,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 12,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10                    -- num_iterations\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 13,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>1</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>10</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-06 00:29:48.575453</td>\n",
+       "        <td>2021-03-06 00:29:51.861215</td>\n",
+       "        <td>[3.28567409515381]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.958333313465</td>\n",
+       "        <td>0.560008466244</td>\n",
+       "        <td>[0.958333313465118]</td>\n",
+       "        <td>[0.560008466243744]</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>[10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, None, None, 10, None, None, u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 6, 0, 29, 48, 575453), datetime.datetime(2021, 3, 6, 0, 29, 51, 861215), [3.28567409515381], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.958333313465118, 0.560008466243744, [0.958333313465118], [0.560008466243744], None, None, None, None, [10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 13,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"eval\"></a>\n",
+    "# 5. Evaluate\n",
+    "\n",
+    "Now run evaluate using model we built above:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 14,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>loss</th>\n",
+       "        <th>metric</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>0.590213775635</td>\n",
+       "        <td>0.899999976158</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(0.590213775634766, 0.899999976158142, [u'accuracy'], u'categorical_crossentropy')]"
+      ]
+     },
+     "execution_count": 14,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_validate;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_evaluate('iris_model',       -- model\n",
+    "                                   'iris_test_packed',  -- test table\n",
+    "                                   'iris_validate'      -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_validate;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred\"></a>\n",
+    "# 6. Predict\n",
+    "\n",
+    "Now predict using model we built.  We will use the validation data set for prediction as well, which is not usual but serves to show the syntax. The prediction is in the estimated_class_text column:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 15,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "90 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8026635</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.13821265</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.059123855</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8471821</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.110732675</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.042085275</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8697099</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.09588991</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.03440017</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9113638</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.06569701</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.022939174</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8007704</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.14301367</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.056215953</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.7946505</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.14609303</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.05925647</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8087025</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.1362152</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.055082306</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9220808</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.059425894</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.018493252</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.82773703</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.12367095</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.04859213</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.52441037</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.2698906</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.205699</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4541727</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3792952</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.16653205</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5121435</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.2632511</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.22460538</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.44443503</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4180967</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.13746823</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.46657953</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3879333</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.14548717</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.47301942</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3099994</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.21698117</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.48130804</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.2771792</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.2415128</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.48122597</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.30558127</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.2131927</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.45175043</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.2748035</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.2734461</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.45799258</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.29800493</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.24400246</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.41659498</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.34956554</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.23383953</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.46772137</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3688568</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.16342185</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4250459</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.41558483</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.15936929</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.46659094</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3897162</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.1436929</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5056077</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.37151548</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.12287672</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.42669904</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.418276</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.15502496</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.41957054</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.41565675</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.16477272</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.4755917</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.40127525</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.12313308</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.50083333</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.34366286</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.15550385</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.46772137</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.3688568</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.16342185</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.47896492</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.36823604</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.15279905</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(9, u'class_text', u'Iris-setosa', 0.8026635, 1),\n",
+       " (9, u'class_text', u'Iris-versicolor', 0.13821265, 2),\n",
+       " (9, u'class_text', u'Iris-virginica', 0.059123855, 3),\n",
+       " (12, u'class_text', u'Iris-setosa', 0.8471821, 1),\n",
+       " (12, u'class_text', u'Iris-versicolor', 0.110732675, 2),\n",
+       " (12, u'class_text', u'Iris-virginica', 0.042085275, 3),\n",
+       " (18, u'class_text', u'Iris-setosa', 0.8697099, 1),\n",
+       " (18, u'class_text', u'Iris-versicolor', 0.09588991, 2),\n",
+       " (18, u'class_text', u'Iris-virginica', 0.03440017, 3),\n",
+       " (23, u'class_text', u'Iris-setosa', 0.9113638, 1),\n",
+       " (23, u'class_text', u'Iris-versicolor', 0.06569701, 2),\n",
+       " (23, u'class_text', u'Iris-virginica', 0.022939174, 3),\n",
+       " (24, u'class_text', u'Iris-setosa', 0.8007704, 1),\n",
+       " (24, u'class_text', u'Iris-versicolor', 0.14301367, 2),\n",
+       " (24, u'class_text', u'Iris-virginica', 0.056215953, 3),\n",
+       " (26, u'class_text', u'Iris-setosa', 0.7946505, 1),\n",
+       " (26, u'class_text', u'Iris-versicolor', 0.14609303, 2),\n",
+       " (26, u'class_text', u'Iris-virginica', 0.05925647, 3),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.8087025, 1),\n",
+       " (31, u'class_text', u'Iris-versicolor', 0.1362152, 2),\n",
+       " (31, u'class_text', u'Iris-virginica', 0.055082306, 3),\n",
+       " (33, u'class_text', u'Iris-setosa', 0.9220808, 1),\n",
+       " (33, u'class_text', u'Iris-versicolor', 0.059425894, 2),\n",
+       " (33, u'class_text', u'Iris-virginica', 0.018493252, 3),\n",
+       " (38, u'class_text', u'Iris-setosa', 0.82773703, 1),\n",
+       " (38, u'class_text', u'Iris-versicolor', 0.12367095, 2),\n",
+       " (38, u'class_text', u'Iris-virginica', 0.04859213, 3),\n",
+       " (51, u'class_text', u'Iris-versicolor', 0.52441037, 1),\n",
+       " (51, u'class_text', u'Iris-virginica', 0.2698906, 2),\n",
+       " (51, u'class_text', u'Iris-setosa', 0.205699, 3),\n",
+       " (54, u'class_text', u'Iris-versicolor', 0.4541727, 1),\n",
+       " (54, u'class_text', u'Iris-virginica', 0.3792952, 2),\n",
+       " (54, u'class_text', u'Iris-setosa', 0.16653205, 3),\n",
+       " (66, u'class_text', u'Iris-versicolor', 0.5121435, 1),\n",
+       " (66, u'class_text', u'Iris-virginica', 0.2632511, 2),\n",
+       " (66, u'class_text', u'Iris-setosa', 0.22460538, 3),\n",
+       " (73, u'class_text', u'Iris-virginica', 0.44443503, 1),\n",
+       " (73, u'class_text', u'Iris-versicolor', 0.4180967, 2),\n",
+       " (73, u'class_text', u'Iris-setosa', 0.13746823, 3),\n",
+       " (78, u'class_text', u'Iris-versicolor', 0.46657953, 1),\n",
+       " (78, u'class_text', u'Iris-virginica', 0.3879333, 2),\n",
+       " (78, u'class_text', u'Iris-setosa', 0.14548717, 3),\n",
+       " (81, u'class_text', u'Iris-versicolor', 0.47301942, 1),\n",
+       " (81, u'class_text', u'Iris-virginica', 0.3099994, 2),\n",
+       " (81, u'class_text', u'Iris-setosa', 0.21698117, 3),\n",
+       " (83, u'class_text', u'Iris-versicolor', 0.48130804, 1),\n",
+       " (83, u'class_text', u'Iris-virginica', 0.2771792, 2),\n",
+       " (83, u'class_text', u'Iris-setosa', 0.2415128, 3),\n",
+       " (93, u'class_text', u'Iris-versicolor', 0.48122597, 1),\n",
+       " (93, u'class_text', u'Iris-virginica', 0.30558127, 2),\n",
+       " (93, u'class_text', u'Iris-setosa', 0.2131927, 3),\n",
+       " (94, u'class_text', u'Iris-versicolor', 0.45175043, 1),\n",
+       " (94, u'class_text', u'Iris-setosa', 0.2748035, 2),\n",
+       " (94, u'class_text', u'Iris-virginica', 0.2734461, 3),\n",
+       " (96, u'class_text', u'Iris-versicolor', 0.45799258, 1),\n",
+       " (96, u'class_text', u'Iris-virginica', 0.29800493, 2),\n",
+       " (96, u'class_text', u'Iris-setosa', 0.24400246, 3),\n",
+       " (99, u'class_text', u'Iris-versicolor', 0.41659498, 1),\n",
+       " (99, u'class_text', u'Iris-setosa', 0.34956554, 2),\n",
+       " (99, u'class_text', u'Iris-virginica', 0.23383953, 3),\n",
+       " (102, u'class_text', u'Iris-virginica', 0.46772137, 1),\n",
+       " (102, u'class_text', u'Iris-versicolor', 0.3688568, 2),\n",
+       " (102, u'class_text', u'Iris-setosa', 0.16342185, 3),\n",
+       " (111, u'class_text', u'Iris-versicolor', 0.4250459, 1),\n",
+       " (111, u'class_text', u'Iris-virginica', 0.41558483, 2),\n",
+       " (111, u'class_text', u'Iris-setosa', 0.15936929, 3),\n",
+       " (117, u'class_text', u'Iris-virginica', 0.46659094, 1),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.3897162, 2),\n",
+       " (117, u'class_text', u'Iris-setosa', 0.1436929, 3),\n",
+       " (123, u'class_text', u'Iris-virginica', 0.5056077, 1),\n",
+       " (123, u'class_text', u'Iris-versicolor', 0.37151548, 2),\n",
+       " (123, u'class_text', u'Iris-setosa', 0.12287672, 3),\n",
+       " (127, u'class_text', u'Iris-versicolor', 0.42669904, 1),\n",
+       " (127, u'class_text', u'Iris-virginica', 0.418276, 2),\n",
+       " (127, u'class_text', u'Iris-setosa', 0.15502496, 3),\n",
+       " (128, u'class_text', u'Iris-virginica', 0.41957054, 1),\n",
+       " (128, u'class_text', u'Iris-versicolor', 0.41565675, 2),\n",
+       " (128, u'class_text', u'Iris-setosa', 0.16477272, 3),\n",
+       " (131, u'class_text', u'Iris-virginica', 0.4755917, 1),\n",
+       " (131, u'class_text', u'Iris-versicolor', 0.40127525, 2),\n",
+       " (131, u'class_text', u'Iris-setosa', 0.12313308, 3),\n",
+       " (135, u'class_text', u'Iris-virginica', 0.50083333, 1),\n",
+       " (135, u'class_text', u'Iris-versicolor', 0.34366286, 2),\n",
+       " (135, u'class_text', u'Iris-setosa', 0.15550385, 3),\n",
+       " (143, u'class_text', u'Iris-virginica', 0.46772137, 1),\n",
+       " (143, u'class_text', u'Iris-versicolor', 0.3688568, 2),\n",
+       " (143, u'class_text', u'Iris-setosa', 0.16342185, 3),\n",
+       " (145, u'class_text', u'Iris-virginica', 0.47896492, 1),\n",
+       " (145, u'class_text', u'Iris-versicolor', 0.36823604, 2),\n",
+       " (145, u'class_text', u'Iris-setosa', 0.15279905, 3)]"
+      ]
+     },
+     "execution_count": 15,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_model', -- model\n",
+    "                                   'iris_test',  -- test_table\n",
+    "                                   'id',  -- id column\n",
+    "                                   'attributes', -- independent var\n",
+    "                                   'iris_predict'  -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id, rank;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 16,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3L,)]"
+      ]
+     },
+     "execution_count": 16,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict JOIN iris_test USING (id)\n",
+    "WHERE iris_predict.class_value != iris_test.class_text AND iris_predict.rank = 1;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 17,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90.00</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('90.00'),)]"
+      ]
+     },
+     "execution_count": 17,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict.class_value as estimated\n",
+    "     from iris_predict inner join iris_test\n",
+    "     on iris_test.id=iris_predict.id where iris_predict.rank = 1) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred_byom\"></a>\n",
+    "# 7. Predict BYOM\n",
+    "The predict BYOM function allows you to do inference on models that have not been trained on MADlib, but rather imported from elsewhere.  \n",
+    "\n",
+    "We will use the validation dataset for prediction as well, which is not usual but serves to show the syntax.\n",
+    "\n",
+    "See load_keras_model()\n",
+    "http://madlib.apache.org/docs/latest/group__grp__keras__model__arch.html\n",
+    "for details on how to load the model architecture and weights.  In this example we will use weights we already have:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 18,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 18,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "UPDATE model_arch_library \n",
+    "SET model_weights = iris_model.model_weights \n",
+    "FROM iris_model \n",
+    "WHERE model_arch_library.model_id = 1;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Now train using a model from the model architecture table directly without referencing the model table from the MADlib training.  \n",
+    "\n",
+    "Note that if you specify the class values parameter as we do below, it must reflect how the dependent variable was 1-hot encoded for training.  In this example the 'training_preprocessor_dl()' in Step 2 above encoded in the order {'Iris-setosa', 'Iris-versicolor', 'Iris-virginica'} so this is the order we pass in the parameter.  If we accidently picked another order that did not match the 1-hot encoding, the predictions would be wrong."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 19,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "30 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8026635</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8471821</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8697099</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9113638</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8007704</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.7946505</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.8087025</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9220808</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.82773703</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.52441037</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4541727</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.5121435</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.44443503</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.46657953</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.47301942</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.48130804</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.48122597</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.45175043</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.45799258</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.41659498</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.46772137</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.4250459</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.46659094</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5056077</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.42669904</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.41957054</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.4755917</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.50083333</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.46772137</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>dependent_var</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.47896492</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(9, u'dependent_var', u'Iris-setosa', 0.8026635),\n",
+       " (12, u'dependent_var', u'Iris-setosa', 0.8471821),\n",
+       " (18, u'dependent_var', u'Iris-setosa', 0.8697099),\n",
+       " (23, u'dependent_var', u'Iris-setosa', 0.9113638),\n",
+       " (24, u'dependent_var', u'Iris-setosa', 0.8007704),\n",
+       " (26, u'dependent_var', u'Iris-setosa', 0.7946505),\n",
+       " (31, u'dependent_var', u'Iris-setosa', 0.8087025),\n",
+       " (33, u'dependent_var', u'Iris-setosa', 0.9220808),\n",
+       " (38, u'dependent_var', u'Iris-setosa', 0.82773703),\n",
+       " (51, u'dependent_var', u'Iris-versicolor', 0.52441037),\n",
+       " (54, u'dependent_var', u'Iris-versicolor', 0.4541727),\n",
+       " (66, u'dependent_var', u'Iris-versicolor', 0.5121435),\n",
+       " (73, u'dependent_var', u'Iris-virginica', 0.44443503),\n",
+       " (78, u'dependent_var', u'Iris-versicolor', 0.46657953),\n",
+       " (81, u'dependent_var', u'Iris-versicolor', 0.47301942),\n",
+       " (83, u'dependent_var', u'Iris-versicolor', 0.48130804),\n",
+       " (93, u'dependent_var', u'Iris-versicolor', 0.48122597),\n",
+       " (94, u'dependent_var', u'Iris-versicolor', 0.45175043),\n",
+       " (96, u'dependent_var', u'Iris-versicolor', 0.45799258),\n",
+       " (99, u'dependent_var', u'Iris-versicolor', 0.41659498),\n",
+       " (102, u'dependent_var', u'Iris-virginica', 0.46772137),\n",
+       " (111, u'dependent_var', u'Iris-versicolor', 0.4250459),\n",
+       " (117, u'dependent_var', u'Iris-virginica', 0.46659094),\n",
+       " (123, u'dependent_var', u'Iris-virginica', 0.5056077),\n",
+       " (127, u'dependent_var', u'Iris-versicolor', 0.42669904),\n",
+       " (128, u'dependent_var', u'Iris-virginica', 0.41957054),\n",
+       " (131, u'dependent_var', u'Iris-virginica', 0.4755917),\n",
+       " (135, u'dependent_var', u'Iris-virginica', 0.50083333),\n",
+       " (143, u'dependent_var', u'Iris-virginica', 0.46772137),\n",
+       " (145, u'dependent_var', u'Iris-virginica', 0.47896492)]"
+      ]
+     },
+     "execution_count": 19,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict_byom;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict_byom('model_arch_library',  -- model arch table\n",
+    "                                         1,                    -- model arch id\n",
+    "                                        'iris_test',           -- test_table\n",
+    "                                        'id',                  -- id column\n",
+    "                                        'attributes',          -- independent var\n",
+    "                                        'iris_predict_byom',   -- output table\n",
+    "                                        'response',            -- prediction type\n",
+    "                                         FALSE,                -- use GPUs\n",
+    "                                         ARRAY[ARRAY['Iris-setosa', 'Iris-versicolor', 'Iris-virginica']], -- class values\n",
+    "                                         1.0                   -- normalizing const\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict_byom ORDER BY id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Count missclassifications:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 20,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(3L,)]"
+      ]
+     },
+     "execution_count": 20,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM iris_predict_byom JOIN iris_test USING (id)\n",
+    "WHERE iris_predict_byom.class_value != iris_test.class_text;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Percent missclassifications:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 21,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>90.00</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('90.00'),)]"
+      ]
+     },
+     "execution_count": 21,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100/(150*0.2),2) as test_accuracy_percent from\n",
+    "    (select iris_test.class_text as actual, iris_predict_byom.class_value as estimated\n",
+    "     from iris_predict_byom inner join iris_test\n",
+    "     on iris_test.id=iris_predict_byom.id) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"class2\"></a>\n",
+    "# Classification with Other Parameters\n",
+    "\n",
+    "<a id=\"val_dataset\"></a>\n",
+    "# 1.  Validation dataset\n",
+    "Now use a validation dataset and compute metrics every 2nd iteration using the 'metrics_compute_frequency' parameter.  This can help reduce run time if you do not need metrics computed at every iteration."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 22,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 22,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10,                   -- num_iterations\n",
+    "                                FALSE,                -- use GPUs\n",
+    "                                'iris_test_packed',   -- validation dataset\n",
+    "                                2,                    -- metrics compute frequency\n",
+    "                                FALSE,                -- warm start\n",
+    "                               'Sophie L.',           -- name\n",
+    "                               'Simple MLP for iris dataset'  -- description\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 23,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>1</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Simple MLP for iris dataset</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-06 00:30:02.409489</td>\n",
+       "        <td>2021-03-06 00:30:03.741511</td>\n",
+       "        <td>[0.776393890380859, 0.917426109313965, 1.05355596542358, 1.18816304206848, 1.33194589614868]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.983333349228</td>\n",
+       "        <td>0.251336187124</td>\n",
+       "        <td>[0.975000023841858, 0.983333349227905, 0.975000023841858, 0.991666674613953, 0.983333349227905]</td>\n",
+       "        <td>[0.474240601062775, 0.406035482883453, 0.347332179546356, 0.295203357934952, 0.251336187124252]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.272865414619</td>\n",
+       "        <td>[0.966666638851166, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.50140792131424, 0.429758101701736, 0.369670689105988, 0.317984014749527, 0.272865414619446]</td>\n",
+       "        <td>[2, 4, 6, 8, 10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', None, 2, u'Sophie L.', u'Simple MLP for iris dataset', u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 6, 0, 30, 2, 409489), datetime.datetime(2021, 3, 6, 0, 30, 3, 741511), [0.776393890380859, 0.917426109313965, 1.05355596542358, 1.18816304206848, 1.33194589614868], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.983333349227905, 0.251336187124252, [0.975000023841858, 0.983333349227905, 0.975000023841858, 0.991666674613953, 0.983333349227905], [0.474240601062775, 0.406035482883453, 0.347332179546356, 0.295203357934952, 0.251336187124252], 0.966666638851166, 0.272865414619446, [0.966666638851166, 0.899999976158142, 0.966666638851166, 0.966666638851166, 0.966666638851166], [0.50140792131424, 0.429758101701736, 0.369670689105988, 0.317984014749527, 0.272865414619446], [2, 4, 6, 8, 10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 23,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Accuracy by iteration"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 24,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np\n",
+    "import sys\n",
+    "import os\n",
+    "from matplotlib import pyplot as plt\n",
+    "\n",
+    "# get accuracy and iteration number\n",
+    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
+    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
+    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
+    "\n",
+    "# get number of points\n",
+    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
+    "num_points = num_points_proxy[0]\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "iters = np.array(iters_proxy).reshape(num_points)\n",
+    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
+    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by iteration')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Loss by iteration"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 25,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get loss\n",
+    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
+    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
+    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation loss by iteration')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Loss')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Accuracy by time"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 26,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get time\n",
+    "time_proxy = %sql SELECT metrics_elapsed_time FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "time = np.array(time_proxy).reshape(num_points)/60.0\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by time')\n",
+    "plt.xlabel('Time (min)')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(time, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(time, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Time to achieve a given accuracy"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 27,
+   "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": [
+    "#plot\n",
+    "plt.title('Iris time by validation accuracy')\n",
+    "plt.xlabel('Accuracy')\n",
+    "plt.ylabel('Time (min)')\n",
+    "plt.grid(True)\n",
+    "plt.plot(train_accuracy, time, 'g.-', label='Train')\n",
+    "plt.plot(test_accuracy, time, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"pred_prob\"></a>\n",
+    "# 2. Predict probabilities\n",
+    "Predict with probabilities for each class:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 28,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n",
+      "90 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>id</th>\n",
+       "        <th>class_name</th>\n",
+       "        <th>class_value</th>\n",
+       "        <th>prob</th>\n",
+       "        <th>rank</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9294206</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.06840064</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>9</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.002178827</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.95446134</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.044602826</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>12</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0009358918</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9657251</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.033696607</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>18</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.00057823793</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9846532</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.015122785</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>23</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.00022406144</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9215944</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.07646777</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>24</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0019379102</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9169111</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.08092835</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>26</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0021605128</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.92948824</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.06870209</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>31</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0018096273</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.9864705</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.013397024</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>33</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0001324016</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.94184655</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.056837272</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>38</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0013162153</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8235848</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.10498193</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>51</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.07143335</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.60921746</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.33687788</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>54</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.05390459</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.8232913</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.09234422</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>66</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.084364414</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5667156</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.41627046</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>73</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.017013948</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.584179</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.3888702</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>78</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.026950791</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7245893</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.17719878</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>81</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.09821194</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7691566</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.12076647</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>83</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.11007695</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.7436626</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.16565561</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>93</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.09068175</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.72709775</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.1566202</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>94</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.11628201</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.727868</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.15816915</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>96</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.11396286</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.66456306</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.2604218</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>99</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.0750151</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7876302</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.20347802</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>102</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.008891817</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5265808</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.45525447</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>111</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.018164707</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7262987</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.2655178</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>117</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.008183556</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8544553</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.1435273</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>123</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0020173935</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.5349145</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.44564962</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>127</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.019435901</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.545205</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.43319198</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>128</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.021603057</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8042598</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.19291137</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>131</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.0028288176</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.8229959</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.17136817</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>135</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.005635898</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7876302</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.20347802</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>143</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.008891817</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-virginica</td>\n",
+       "        <td>0.7896347</td>\n",
+       "        <td>1</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-versicolor</td>\n",
+       "        <td>0.20416242</td>\n",
+       "        <td>2</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>145</td>\n",
+       "        <td>class_text</td>\n",
+       "        <td>Iris-setosa</td>\n",
+       "        <td>0.006202857</td>\n",
+       "        <td>3</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(9, u'class_text', u'Iris-setosa', 0.9294206, 1),\n",
+       " (9, u'class_text', u'Iris-versicolor', 0.06840064, 2),\n",
+       " (9, u'class_text', u'Iris-virginica', 0.002178827, 3),\n",
+       " (12, u'class_text', u'Iris-setosa', 0.95446134, 1),\n",
+       " (12, u'class_text', u'Iris-versicolor', 0.044602826, 2),\n",
+       " (12, u'class_text', u'Iris-virginica', 0.0009358918, 3),\n",
+       " (18, u'class_text', u'Iris-setosa', 0.9657251, 1),\n",
+       " (18, u'class_text', u'Iris-versicolor', 0.033696607, 2),\n",
+       " (18, u'class_text', u'Iris-virginica', 0.00057823793, 3),\n",
+       " (23, u'class_text', u'Iris-setosa', 0.9846532, 1),\n",
+       " (23, u'class_text', u'Iris-versicolor', 0.015122785, 2),\n",
+       " (23, u'class_text', u'Iris-virginica', 0.00022406144, 3),\n",
+       " (24, u'class_text', u'Iris-setosa', 0.9215944, 1),\n",
+       " (24, u'class_text', u'Iris-versicolor', 0.07646777, 2),\n",
+       " (24, u'class_text', u'Iris-virginica', 0.0019379102, 3),\n",
+       " (26, u'class_text', u'Iris-setosa', 0.9169111, 1),\n",
+       " (26, u'class_text', u'Iris-versicolor', 0.08092835, 2),\n",
+       " (26, u'class_text', u'Iris-virginica', 0.0021605128, 3),\n",
+       " (31, u'class_text', u'Iris-setosa', 0.92948824, 1),\n",
+       " (31, u'class_text', u'Iris-versicolor', 0.06870209, 2),\n",
+       " (31, u'class_text', u'Iris-virginica', 0.0018096273, 3),\n",
+       " (33, u'class_text', u'Iris-setosa', 0.9864705, 1),\n",
+       " (33, u'class_text', u'Iris-versicolor', 0.013397024, 2),\n",
+       " (33, u'class_text', u'Iris-virginica', 0.0001324016, 3),\n",
+       " (38, u'class_text', u'Iris-setosa', 0.94184655, 1),\n",
+       " (38, u'class_text', u'Iris-versicolor', 0.056837272, 2),\n",
+       " (38, u'class_text', u'Iris-virginica', 0.0013162153, 3),\n",
+       " (51, u'class_text', u'Iris-versicolor', 0.8235848, 1),\n",
+       " (51, u'class_text', u'Iris-virginica', 0.10498193, 2),\n",
+       " (51, u'class_text', u'Iris-setosa', 0.07143335, 3),\n",
+       " (54, u'class_text', u'Iris-versicolor', 0.60921746, 1),\n",
+       " (54, u'class_text', u'Iris-virginica', 0.33687788, 2),\n",
+       " (54, u'class_text', u'Iris-setosa', 0.05390459, 3),\n",
+       " (66, u'class_text', u'Iris-versicolor', 0.8232913, 1),\n",
+       " (66, u'class_text', u'Iris-virginica', 0.09234422, 2),\n",
+       " (66, u'class_text', u'Iris-setosa', 0.084364414, 3),\n",
+       " (73, u'class_text', u'Iris-virginica', 0.5667156, 1),\n",
+       " (73, u'class_text', u'Iris-versicolor', 0.41627046, 2),\n",
+       " (73, u'class_text', u'Iris-setosa', 0.017013948, 3),\n",
+       " (78, u'class_text', u'Iris-versicolor', 0.584179, 1),\n",
+       " (78, u'class_text', u'Iris-virginica', 0.3888702, 2),\n",
+       " (78, u'class_text', u'Iris-setosa', 0.026950791, 3),\n",
+       " (81, u'class_text', u'Iris-versicolor', 0.7245893, 1),\n",
+       " (81, u'class_text', u'Iris-virginica', 0.17719878, 2),\n",
+       " (81, u'class_text', u'Iris-setosa', 0.09821194, 3),\n",
+       " (83, u'class_text', u'Iris-versicolor', 0.7691566, 1),\n",
+       " (83, u'class_text', u'Iris-virginica', 0.12076647, 2),\n",
+       " (83, u'class_text', u'Iris-setosa', 0.11007695, 3),\n",
+       " (93, u'class_text', u'Iris-versicolor', 0.7436626, 1),\n",
+       " (93, u'class_text', u'Iris-virginica', 0.16565561, 2),\n",
+       " (93, u'class_text', u'Iris-setosa', 0.09068175, 3),\n",
+       " (94, u'class_text', u'Iris-versicolor', 0.72709775, 1),\n",
+       " (94, u'class_text', u'Iris-setosa', 0.1566202, 2),\n",
+       " (94, u'class_text', u'Iris-virginica', 0.11628201, 3),\n",
+       " (96, u'class_text', u'Iris-versicolor', 0.727868, 1),\n",
+       " (96, u'class_text', u'Iris-virginica', 0.15816915, 2),\n",
+       " (96, u'class_text', u'Iris-setosa', 0.11396286, 3),\n",
+       " (99, u'class_text', u'Iris-versicolor', 0.66456306, 1),\n",
+       " (99, u'class_text', u'Iris-setosa', 0.2604218, 2),\n",
+       " (99, u'class_text', u'Iris-virginica', 0.0750151, 3),\n",
+       " (102, u'class_text', u'Iris-virginica', 0.7876302, 1),\n",
+       " (102, u'class_text', u'Iris-versicolor', 0.20347802, 2),\n",
+       " (102, u'class_text', u'Iris-setosa', 0.008891817, 3),\n",
+       " (111, u'class_text', u'Iris-virginica', 0.5265808, 1),\n",
+       " (111, u'class_text', u'Iris-versicolor', 0.45525447, 2),\n",
+       " (111, u'class_text', u'Iris-setosa', 0.018164707, 3),\n",
+       " (117, u'class_text', u'Iris-virginica', 0.7262987, 1),\n",
+       " (117, u'class_text', u'Iris-versicolor', 0.2655178, 2),\n",
+       " (117, u'class_text', u'Iris-setosa', 0.008183556, 3),\n",
+       " (123, u'class_text', u'Iris-virginica', 0.8544553, 1),\n",
+       " (123, u'class_text', u'Iris-versicolor', 0.1435273, 2),\n",
+       " (123, u'class_text', u'Iris-setosa', 0.0020173935, 3),\n",
+       " (127, u'class_text', u'Iris-virginica', 0.5349145, 1),\n",
+       " (127, u'class_text', u'Iris-versicolor', 0.44564962, 2),\n",
+       " (127, u'class_text', u'Iris-setosa', 0.019435901, 3),\n",
+       " (128, u'class_text', u'Iris-virginica', 0.545205, 1),\n",
+       " (128, u'class_text', u'Iris-versicolor', 0.43319198, 2),\n",
+       " (128, u'class_text', u'Iris-setosa', 0.021603057, 3),\n",
+       " (131, u'class_text', u'Iris-virginica', 0.8042598, 1),\n",
+       " (131, u'class_text', u'Iris-versicolor', 0.19291137, 2),\n",
+       " (131, u'class_text', u'Iris-setosa', 0.0028288176, 3),\n",
+       " (135, u'class_text', u'Iris-virginica', 0.8229959, 1),\n",
+       " (135, u'class_text', u'Iris-versicolor', 0.17136817, 2),\n",
+       " (135, u'class_text', u'Iris-setosa', 0.005635898, 3),\n",
+       " (143, u'class_text', u'Iris-virginica', 0.7876302, 1),\n",
+       " (143, u'class_text', u'Iris-versicolor', 0.20347802, 2),\n",
+       " (143, u'class_text', u'Iris-setosa', 0.008891817, 3),\n",
+       " (145, u'class_text', u'Iris-virginica', 0.7896347, 1),\n",
+       " (145, u'class_text', u'Iris-versicolor', 0.20416242, 2),\n",
+       " (145, u'class_text', u'Iris-setosa', 0.006202857, 3)]"
+      ]
+     },
+     "execution_count": 28,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_predict;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict('iris_model',      -- model\n",
+    "                                   'iris_test',       -- test_table\n",
+    "                                   'id',              -- id column\n",
+    "                                   'attributes',      -- independent var\n",
+    "                                   'iris_predict',    -- output table\n",
+    "                                   'prob'             -- response type\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM iris_predict ORDER BY id, rank;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"warm_start\"></a>\n",
+    "# 3. Warm start\n",
+    "Next, use the warm_start parameter to continue learning, using the coefficients from the run above. Note that we don't drop the model table or model summary table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 29,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 29,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10,                   -- num_iterations\n",
+    "                                FALSE,                -- use GPUs\n",
+    "                                'iris_test_packed',   -- validation dataset\n",
+    "                                2,                    -- metrics compute frequency\n",
+    "                                TRUE,                 -- warm start\n",
+    "                               'Sophie L.',           -- name \n",
+    "                               'Simple MLP for iris dataset'  -- description\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "In the summary table and plots below note that the loss and accuracy values pick up from where the previous run left off:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 30,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>1</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2</td>\n",
+       "        <td>Sophie L.</td>\n",
+       "        <td>Simple MLP for iris dataset</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-06 00:30:10.030573</td>\n",
+       "        <td>2021-03-06 00:30:11.397745</td>\n",
+       "        <td>[0.810183048248291, 0.952910184860229, 1.08659505844116, 1.22299003601074, 1.36708807945251]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.991666674614</td>\n",
+       "        <td>0.139637470245</td>\n",
+       "        <td>[0.975000023841858, 0.975000023841858, 0.991666674613953, 0.991666674613953, 0.991666674613953]</td>\n",
+       "        <td>[0.21851558983326, 0.192899897694588, 0.170841887593269, 0.153602108359337, 0.139637470245361]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.162758678198</td>\n",
+       "        <td>[0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.239112228155136, 0.212523519992828, 0.192814856767654, 0.176179185509682, 0.162758678197861]</td>\n",
+       "        <td>[2, 4, 6, 8, 10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 1, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', None, 2, u'Sophie L.', u'Simple MLP for iris dataset', u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 6, 0, 30, 10, 30573), datetime.datetime(2021, 3, 6, 0, 30, 11, 397745), [0.810183048248291, 0.952910184860229, 1.08659505844116, 1.22299003601074, 1.36708807945251], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.991666674613953, 0.139637470245361, [0.975000023841858, 0.975000023841858, 0.991666674613953, 0.991666674613953, 0.991666674613953], [0.21851558983326, 0.192899897694588, 0.170841887593269, 0.153602108359337, 0.139637470245361], 0.966666638851166, 0.162758678197861, [0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166], [0.239112228155136, 0.212523519992828, 0.192814856767654, 0.176179185509682, 0.162758678197861], [2, 4, 6, 8, 10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 30,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 31,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np\n",
+    "import sys\n",
+    "import os\n",
+    "from matplotlib import pyplot as plt\n",
+    "\n",
+    "# get accuracy and iteration number\n",
+    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
+    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
+    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
+    "\n",
+    "# get number of points\n",
+    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
+    "num_points = num_points_proxy[0]\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "iters = np.array(iters_proxy).reshape(num_points)\n",
+    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
+    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by iteration - warm start')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 32,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get loss\n",
+    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
+    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
+    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation loss by iteration - warm start')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Loss')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"transfer_learn\"></a>\n",
+    "# Transfer learning\n",
+    "\n",
+    "<a id=\"load2\"></a>\n",
+    "# 1. Define and load model architecture with some layers frozen\n",
+    "Here we want to start with initial weights from a pre-trained model rather than training from scratch.  We also want to use a model architecture with the earlier feature layer(s) frozen to save on training time.  The example below is somewhat contrived but gives you the idea of the steps.\n",
+    "\n",
+    "First define a model architecture with the 1st hidden layer frozen:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 33,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Model: \"sequential_1\"\n",
+      "_________________________________________________________________\n",
+      "Layer (type)                 Output Shape              Param #   \n",
+      "=================================================================\n",
+      "dense_3 (Dense)              (None, 10)                50        \n",
+      "_________________________________________________________________\n",
+      "dense_4 (Dense)              (None, 10)                110       \n",
+      "_________________________________________________________________\n",
+      "dense_5 (Dense)              (None, 3)                 33        \n",
+      "=================================================================\n",
+      "Total params: 193\n",
+      "Trainable params: 143\n",
+      "Non-trainable params: 50\n",
+      "_________________________________________________________________\n"
+     ]
+    }
+   ],
+   "source": [
+    "model_transfer = Sequential()\n",
+    "model_transfer.add(Dense(10, activation='relu', input_shape=(4,), trainable=False))\n",
+    "model_transfer.add(Dense(10, activation='relu'))\n",
+    "model_transfer.add(Dense(3, activation='softmax'))\n",
+    "    \n",
+    "model_transfer.summary();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 34,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": false, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"name\": \"sequential_1\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 34,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model_transfer.to_json()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load transfer model into model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 35,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "2 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>model_arch</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_1', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>Sophie</td>\n",
+       "        <td>A simple model</td>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2</td>\n",
+       "        <td>{u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_2', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'dtype': u'float32', u'activation': u'relu', u'trainable': False, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'batch_input_shape': [None, 4], u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_3', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'relu', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 10, u'use_bias': True, u'activity_regularizer': None}}, {u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u'VarianceScaling', u'config': {u'distribution': u'uniform', u'scale': 1.0, u'seed': None, u'mode': u'fan_avg'}}, u'name': u'dense_4', u'kernel_constraint': None, u'bias_regularizer': None, u'bias_constraint': None, u'activation': u'softmax', u'trainable': True, u'kernel_regularizer': None, u'bias_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}</td>\n",
+       "        <td>Maria</td>\n",
+       "        <td>A transfer model</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1340 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Sophie', u'A simple model'),\n",
+       " (2, {u'class_name': u'Sequential', u'keras_version': u'2.1.6', u'config': [{u'class_name': u'Dense', u'config': {u'kernel_initializer': {u'class_name': u' ... (1341 characters truncated) ... s_initializer': {u'class_name': u'Zeros', u'config': {}}, u'units': 3, u'use_bias': True, u'activity_regularizer': None}}], u'backend': u'tensorflow'}, u'Maria', u'A transfer model')]"
+      ]
+     },
+     "execution_count": 35,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT madlib.load_keras_model('model_arch_library',  -- Output table,                      \n",
+    "$$\n",
+    "{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": false, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"batch_input_shape\": [null, 4], \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 3, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}\n",
+    "$$\n",
+    "::json,         -- JSON blob\n",
+    "                               NULL,                  -- Weights\n",
+    "                               'Maria',               -- Name\n",
+    "                               'A transfer model'     -- Descr\n",
+    ");\n",
+    "\n",
+    "SELECT model_id, model_arch, name, description FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train2\"></a>\n",
+    "# 2. Train transfer model\n",
+    "\n",
+    "Fetch the weights from a previous MADlib run.  (Normally these would be downloaded from a source that trained the same model architecture on a related dataset.)"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 36,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 36,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "UPDATE model_arch_library \n",
+    "SET model_weights = iris_model.model_weights \n",
+    "FROM iris_model \n",
+    "WHERE model_arch_library.model_id = 2;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Now train the model using the transfer model and the pre-trained weights:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 37,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>madlib_keras_fit</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td></td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[('',)]"
+      ]
+     },
+     "execution_count": 37,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS iris_model, iris_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('iris_train_packed',   -- source table\n",
+    "                               'iris_model',          -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                2,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] $$,  -- compile_params\n",
+    "                                $$ batch_size=5, epochs=3 $$,  -- fit_params\n",
+    "                                10,                   -- num_iterations\n",
+    "                                FALSE,                -- use GPUs\n",
+    "                                'iris_test_packed',   -- validation dataset\n",
+    "                                2                     -- metrics compute frequency\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 38,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>source_table</th>\n",
+       "        <th>model</th>\n",
+       "        <th>dependent_varname</th>\n",
+       "        <th>independent_varname</th>\n",
+       "        <th>model_arch_table</th>\n",
+       "        <th>model_id</th>\n",
+       "        <th>compile_params</th>\n",
+       "        <th>fit_params</th>\n",
+       "        <th>num_iterations</th>\n",
+       "        <th>validation_table</th>\n",
+       "        <th>object_table</th>\n",
+       "        <th>metrics_compute_frequency</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "        <th>model_type</th>\n",
+       "        <th>model_size</th>\n",
+       "        <th>start_training_time</th>\n",
+       "        <th>end_training_time</th>\n",
+       "        <th>metrics_elapsed_time</th>\n",
+       "        <th>madlib_version</th>\n",
+       "        <th>num_classes</th>\n",
+       "        <th>dependent_vartype</th>\n",
+       "        <th>normalizing_const</th>\n",
+       "        <th>metrics_type</th>\n",
+       "        <th>loss_type</th>\n",
+       "        <th>training_metrics_final</th>\n",
+       "        <th>training_loss_final</th>\n",
+       "        <th>training_metrics</th>\n",
+       "        <th>training_loss</th>\n",
+       "        <th>validation_metrics_final</th>\n",
+       "        <th>validation_loss_final</th>\n",
+       "        <th>validation_metrics</th>\n",
+       "        <th>validation_loss</th>\n",
+       "        <th>metrics_iters</th>\n",
+       "        <th>class_text_class_values</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>iris_train_packed</td>\n",
+       "        <td>iris_model</td>\n",
+       "        <td>[u'class_text']</td>\n",
+       "        <td>[u'attributes']</td>\n",
+       "        <td>model_arch_library</td>\n",
+       "        <td>2</td>\n",
+       "        <td> loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] </td>\n",
+       "        <td> batch_size=5, epochs=3 </td>\n",
+       "        <td>10</td>\n",
+       "        <td>iris_test_packed</td>\n",
+       "        <td>None</td>\n",
+       "        <td>2</td>\n",
+       "        <td>None</td>\n",
+       "        <td>None</td>\n",
+       "        <td>madlib_keras</td>\n",
+       "        <td>0.7900390625</td>\n",
+       "        <td>2021-03-06 00:30:15.327042</td>\n",
+       "        <td>2021-03-06 00:30:16.549371</td>\n",
+       "        <td>[0.64433479309082, 0.790453910827637, 0.93260383605957, 1.07773494720459, 1.22224497795105]</td>\n",
+       "        <td>1.18.0-dev</td>\n",
+       "        <td>[3]</td>\n",
+       "        <td>[u'character varying']</td>\n",
+       "        <td>1.0</td>\n",
+       "        <td>[u'accuracy']</td>\n",
+       "        <td>categorical_crossentropy</td>\n",
+       "        <td>0.991666674614</td>\n",
+       "        <td>0.101017765701</td>\n",
+       "        <td>[0.991666674613953, 0.991666674613953, 0.991666674613953, 0.991666674613953, 0.991666674613953]</td>\n",
+       "        <td>[0.127938449382782, 0.11921951174736, 0.112009204924107, 0.106061458587646, 0.101017765700817]</td>\n",
+       "        <td>0.966666638851</td>\n",
+       "        <td>0.124947711825</td>\n",
+       "        <td>[0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166]</td>\n",
+       "        <td>[0.151599556207657, 0.142985984683037, 0.135312005877495, 0.129271760582924, 0.124947711825371]</td>\n",
+       "        <td>[2, 4, 6, 8, 10]</td>\n",
+       "        <td>[u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica']</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'iris_train_packed', u'iris_model', [u'class_text'], [u'attributes'], u'model_arch_library', 2, u\" loss='categorical_crossentropy', optimizer='adam', metrics=['accuracy'] \", u' batch_size=5, epochs=3 ', 10, u'iris_test_packed', None, 2, None, None, u'madlib_keras', 0.7900390625, datetime.datetime(2021, 3, 6, 0, 30, 15, 327042), datetime.datetime(2021, 3, 6, 0, 30, 16, 549371), [0.64433479309082, 0.790453910827637, 0.93260383605957, 1.07773494720459, 1.22224497795105], u'1.18.0-dev', [3], [u'character varying'], 1.0, [u'accuracy'], u'categorical_crossentropy', 0.991666674613953, 0.101017765700817, [0.991666674613953, 0.991666674613953, 0.991666674613953, 0.991666674613953, 0.991666674613953], [0.127938449382782, 0.11921951174736, 0.112009204924107, 0.106061458587646, 0.101017765700817], 0.966666638851166, 0.124947711825371, [0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166, 0.966666638851166], [0.151599556207657, 0.142985984683037, 0.135312005877495, 0.129271760582924, 0.124947711825371], [2, 4, 6, 8, 10], [u'Iris-setosa', u'Iris-versicolor', u'Iris-virginica'])]"
+      ]
+     },
+     "execution_count": 38,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM iris_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Note loss picks up from where the last training left off:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 39,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "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\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np\n",
+    "import sys\n",
+    "import os\n",
+    "from matplotlib import pyplot as plt\n",
+    "\n",
+    "# get accuracy and iteration number\n",
+    "iters_proxy = %sql SELECT metrics_iters FROM iris_model_summary;\n",
+    "train_accuracy_proxy = %sql SELECT training_metrics FROM iris_model_summary;\n",
+    "test_accuracy_proxy = %sql SELECT validation_metrics FROM iris_model_summary;\n",
+    "\n",
+    "# get number of points\n",
+    "num_points_proxy = %sql SELECT array_length(metrics_iters,1) FROM iris_model_summary;\n",
+    "num_points = num_points_proxy[0]\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "iters = np.array(iters_proxy).reshape(num_points)\n",
+    "train_accuracy = np.array(train_accuracy_proxy).reshape(num_points)\n",
+    "test_accuracy = np.array(test_accuracy_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation accuracy by iteration - transfer learn')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Accuracy')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_accuracy, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_accuracy, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 40,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n",
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYsAAAEWCAYAAACXGLsWAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADl0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uIDIuMi41LCBodHRwOi8vbWF0cGxvdGxpYi5vcmcvSM8oowAAIABJREFUeJzs3XmczlX7wPHPNYuZsW9ZZxjrCMnOIGsLCZWKSJZ6FKmI1qfkp9XTU1Iplceard2QUgYtCCVEDEKMRJStbGOu3x/nO9MYgxnmnnuW6/163a+57+963es155zvOUdUFWOMMeZcAvwdgDHGmOzPkoUxxpjzsmRhjDHmvCxZGGOMOS9LFsYYY87LkoUxxpjzsmThYyLSU0Q+z8LzRYqIikiQ9/hTEemdnm0v4FyPicj4i4n3LMftIyLfZPZxz3Kui3oNUh3LJ69HBmM46/udm4nIDSKyU0SOiEi9TDjeJBF5OjNiyy0u+guS14nIduBOVV2Q1npVnQZMy9KgTj9/h8w4joi0Bt5R1fAUx342M46dW6R8PUQkEtgGBKtqgi/OJyIjgKqqeluKGDLl/b6AWPrgvgct/HF+4L/AIFWd7afz53pWsvChzPhv1eRNufGzIyKBPjx8RWD9heyYla91Tn5fLVlkIq/6ZImIjBaR/cCIlFUq4owWkb0ickhEfhSR2mkcp5uIfJdq2RARifHudxSRH7xj7PT+wzxbTItF5E7vfqCI/FdE9onIVqBjqm37isgGETksIltF5C5veQHgU6CcV8w/IiLlRGSEiLyTYv/OIrJeRA545700xbrtIjJMRNaKyEERmSUioel8XZuJyEpvv5Ui0izVa77Vi3mbiPT0llcVkS+9ffaJyKzznKafiPwqIrtFZJh3jDIi8reIlEhxvvoi8ruIBKcRZ8rX4yvv7wHv9Yr2tunnvcZ/ish8EamYYn8VkXtEZDOw2Vs2xnuPD4nI9yJyhbe8PfAY0M07/hpvecr3O0BEHheRX7zP3BQRKeKtS6p+6y0iO7zX6N/peT/SeN6XAuOAaC+WA97ySSLyhojME5G/gDbn+uyeLyYRaSwi33n77hGRl0QkRESOAIHAGhH52du2nIh84L1X20TkvlTv0/si8o6IHAL6pOM5Xiciq73P9lIRqZNi3SMi8rP3GfxJRG5Ise6svwnivot/evH5pUSYIapqt4u4AduBK737fYAE4F5cFV+Yt+wbb/01wPdAUUCAS4GyaRwzP3AYqJZi2Uqgu3e/NXAZLtnXAfYA13vrIgEFgrzHi3HVAwB3AxuBCKA4sCjVth2BKl5srYC/gfopzhmfKs4RuKopgOrAX8BVQDDwELAFyJfidVoBlPPOvQG4+yyvacrXrDjwJ9DLe01v9R6XAAoAh4Aob9uyQC3v/gzg395rFAq0OMu5kl6vGd7xLgN+T/GezgMGpNh+NPDqWY6V8vU47X3wlnXxXpNLvefyOLA0xXoFvvCec5i37DbvuQYBQ4HfgNDU50txjJTvdz/vfJWBgsCHwNRU8b2N+5xeDhwHLr3A70Hye5Zi2STgINA8xfvQmvN/dtOMCVgG9PLuFwSapnrtqnr3A3Dfs+FAPu/5bwWuSfG6nQSu97YNS+P5TAKe9u7XA/YCTXBJqTfu8xzirb8Z97kOALrhvgdlz/ObcBL4l3e8AcCvgPj79+xcNytZZL5fVfVVVU1Q1aOp1p0ECgE1cB+MDaq6O/UBVPVvYDbuhxERqebtE+OtX6yqP6pqoqquxf3QtUpHbLcAL6vqTlX9A3gu1Xk/UdWf1fkS+By4Ip3Puxvwiap+oaoncXXIYUCzFNu8oqq/eueeA9RNx3E7AptVdar3ms7AJbxO3vpEoLaIhKnqblVNqoo4iauaKKeqx1T1fA3m/6eqf6nqj8BEvNcemIz7wU6qRrkVmJqOuNNyN/Cc974nAM8CdVOWLrz1fyR9dlT1HVXd7z33F4EQICqd5+sJvKSqW1X1CPAo0F1Orwr5P1U9qqprgDW4H+jMNFtVl3if1WPp/OyeLaaTQFURKamqR1T127OcsxFwiaqOVNUTqroVl4C6p9hmmap+7MWR+nuaWn/gTVVdrqqnVHUyLok1BVDV97zPdaKqzsKVChun2D+t34RfVPVtVT2F+4yVBUqfJw6/smSR+XaebYWqLgReA8YCe0XkLREpfJbNp/PPD1YP4GMviSAiTURkkVfEPoj7ESqZjtjKpYrvl5QrRaSDiHwrIn94VQnXpvO4ScdOPp6qJnrnKp9im99S3P8b999hho6bIu7yqvoXLkndDewWkU9EpIa3zUO4EtIKcVVj/c5zntSvSznv/mygpohUwpWaDqrqinTEnZaKwBivKuMA8IcXY8rX6LTPj7iquw1eddoBoAgX+J5494M4/UfpvO+JiFSQf6ofj6Tz3ElSP5/0fHbPFtMduBLsRnHVkded5ZwVcVWmB1K81o9x+vM+6/f0LMcbmup4EXifERG5PUUV1QGgdqrnlNa5kp9j0vea9H0f/MaSReY75zC+qvqKqjYAauI++A+eZdMvgEtEpC4uaUxPsW46rpQRoapFcPXFko7YduM+5EkqJN0RkRDgA1yJoLSqFsVVwSQd93zDE/+K+1IlHU+8c+1KR1zpPq6nQtJxVXW+ql6F+89sI+4/SFT1N1X9l6qWA+4CXheRquc4T+rX5VfvOMeAd3Gli16kv1SR1uu1E7hLVYumuIWp6tK09vPaJx7ClQiLee/JQS7wPfGeVwKu6ifdVHWHqhZMup1ts3Quv9DPLqq6WVVvBUoBo4D3xbWnpbYT2JbqdS6kqtemI9607ASeSXW8/Ko6wysVvg0MAkp479G6VM8pVwztbckiC4lII+8/q2BcveYxXDXKGbyqnPeAF3B12F+kWF0I+ENVj4lIY1zJIz3eBe4TkXARKQY8kmJdPlwVx+9AgtfgdnWK9XuAEkkNpGc5dkcRaec9v6G4ovrSs2yfXvOA6iLSQ0SCRKQbLtHOFZHSItLF+8E4DhzBez1F5GYRSbrM90/cFzbN19rzhIjkF5FaQF8gZYP4FFw9c2fSnyx+985XOcWyccCj3jkQkSIicvM5jlEI9+P+OxAkIsOBlCXRPUCkiJztezwDGCIilUSkIK7aa5b65lLePUC4iOQ7z3YX+tlFRG4TkUu8UusBb3Fa7+kK4LCIPCwiYeIu7KgtIo3Se65U3gbu9r67IiIFxDXUF8K1cynuPUJE+uJKFrmOJYusVRj3wfsTVyWwH5cMzmY6cCXwXqov+EBgpIgcxjXivZvO878NzMfVA6/CNXgCoKqHgfu8Y/2J+xLHpFi/Effjs9UrbpdLcVxUNQ733/erwD5cm0InVT2RztjSpKr7getwyWc/7j/t61R1H+7z+wDuP+g/cHXfA7xdGwHLvWqTGOB+r+76bL7ENQbHAv9V1eSOlKq6BPejtEpVU1eJnS3uv4FngCXe69VUVT/C/Uc807sKZx1wrqtg5gOfAZtwn5djnF6l8Z73d7+IrEpj/wm45PYVrs/HMVxDqy8sxF26+puI7DvHdhf62QVoD6z33tMxuAs+zmhv8NoBrsO1iW3DfR7H46rwMkxVv8M1Rr+G+25swbuCSlV/Al7ENb7vwTXeL7mQ82R3oporSkjG+JSILASmq6pfe2gb4y+WLIw5D6/64gtcPfthf8djjD9YNZQx5yAik4EFwGBLFCYvs5KFMcaY87KShTHGmPPKsYNapVayZEmNjIy84P3/+usvChRI65Jt/7K4MsbiyhiLK2NyY1zff//9PlW95LwbajYYcyQzbg0aNNCLsWjRoova31csroyxuDLG4sqY3BgX8J3a2FDGGGMygyULY4wx52XJwhhjzHnlmgZuY4zJiJMnTxIfH8+xY8fSvU+RIkXYsGGDD6O6MOmJKzQ0lPDwcIKDz5i3K10sWRhj8qT4+HgKFSpEZGQkbpDk8zt8+DCFChXycWQZd764VJX9+/cTHx9PpUqVLugcVg1ljMmTjh07RokSJdKdKHIyEaFEiRIZKkWlZskCYNkyKkybBsuW+TsSY0wWyguJIsnFPlerhlq6FFq1otKpUzBtGsTGQnS0v6MyxphsxUoWH34ICQmIKhw9Cq+9Bgm+mBvGGGP+sX//furWrUvdunUpU6YM5cuXT3584kT6poHp27cvcXFxPo7UsWTRtSuEhaEiIALTp0O1ajB6NBw65O/ojDG5VIkSJVi9ejWrV6/m7rvvZsiQIcmP8+VzEw6qKomJZ5/gceLEiURFRWVJvJYsoqMhNpZtd9wBX33lShoREfDAAxAe7v5u3+7vKI0x2cDyX5fz3NfPsWyn79o3t2zZQs2aNenZsye1atVi9+7d9O/fn4YNG1KrVi1GjhyZvG2LFi1YvXo1CQkJFC1alEceeYTLL7+c6Oho9u7dm6lx+bTNQkTa46Y/DATGq+rzqda3BF4G6uCmSHw/xbpTwI/ewx2q2tlngUZHs+P4cSq3aOEe33ADfPedK128+iqMGeOWDRkCzZq5EogxJtcY/NlgVv+2+pzbHDx+kLV71pKoiQRIAHVK16FIyNlnaq1bpi4vt3/5guLZuHEjU6ZMoWHDhgA8//zzFC9enISEBNq0acNNN91EzZo1T4/v4EFatWrF888/zwMPPMCECRN45JFHLuj8afFZyUJEAoGxuDmGawK3ikjNVJvtwM1lOz2NQxxV1brezXeJ4mwaNnQN3tu2wUMPwcKF0KIFNG0KM2fCyZNZHpIxxn8OHjtIoroqoURN5OCxgz47V5UqVZITBcCMGTOoX78+9evXZ8OGDfz0009n7BMWFkaHDm5K9wYNGrA9k2tEfFmyaAxsUdWtACIyE+gCJD9LVd3urTt7pZy/hYfDc8/B44/D5Mnw8stw661u+b33wr/+BcWK+TtKY8xFSE8JYNnOZbSb0o4Tp06QLzAf026cRnSEb66cTDnc+ObNmxkzZgwrVqygaNGi3HbbbWn2l0hq5wAIDAwkIZMv1PFlsigP7EzxOB5okoH9Q0XkOyABeF5VP069gYj0B/oDlC5dmsWLF19wsEeOHDn//jVrwrhxlFi+nPD33qPYww9z6skn2d2hA7u6duVo+fIXfP6LissPLK6MsbgyJiviKlKkCIcPp3+m3NpFa/PxjR+z9NeltAhvQe2itTO0/7kcP36c4OBgDh8+zJEjR0hMTEw+9u7duylQoAAiwubNm/nss89o1aoVhw8f5tSpU/z111+cOnUKIHmfo0ePcvLkyTPiO3bs2AW/rtm5n0VFVd0lIpWBhSLyo6r+nHIDVX0LeAugYcOG2rp16ws+2eLFi0n3/m3bwqOPwurVBL78MuHTpxP+8cfQqZNrEG/ZMtPaNTIUVxayuDLG4sqYrIhrw4YNGR66Izo8mqsvvTrTYwkJCSEkJIRChQpRsGBBAgICkmO74oorqF27No0aNaJixYq0aNGCsLAwChUqRGBgIAUKFCAwMBAgeZ+wsDCCg4PPeH6hoaHUq1fvgmL0ZbLYBUSkeBzuLUsXVd3l/d0qIouBesDP59wpq9WtC5MmwfPPw+uvwxtvQEwM1KvnGsO7dYMURUNjjEnLiBEjku9XrVqV1av/aWwXEaZOnZrmft988w3gShQHDhxIXt69e3e6d++eqTH68tLZlUA1EakkIvmA7kBMenYUkWIiEuLdLwk0J0VbR7ZTpgyMHAk7dsBbb8GxY3D77RAZCc8+C/v3+ztCY4y5KD5LFqqaAAwC5gMbgHdVdb2IjBSRzgAi0khE4oGbgTdFZL23+6XAdyKyBliEa7PIvskiSViYa/Bevx4+/RQuuwz+/W/Xb+Puu2HjRn9HaIwxF8SnbRaqOg+Yl2rZ8BT3V+Kqp1LvtxS4zJex+ZQItG/vbuvXuyuoJk2CN9+Ea691VVTt2ll/DWNMjmE9uH2tVi14+21XRfV//+c6+111FVx+OUyc6KqsjDEmm7NkkVVKlYLhw13SmDjRlSr69YOKFV0SyeSu+cYYk5ksWWS1kBDo0wdWr4YFC6BRIxgxAipUgDvugHXr/B2hMcacwZKFv4i4dou5c2HDBujbF2bMcI3iV1/tGsjPMdqkMSZny4whygEmTJjAnj17fBipY8kiO6hRw/XR2LnTXWq7fr1rCK9dm7Jz5rh5NowxuUp6hihPD0sWeVGJEq5n+LZt8M47EBZG1EsvuUtvH38cdu/2d4TG5GkBy5e7seJ8PAXz5MmTady4MXXr1mXgwIEkJiaSkJBAr169uOyyy6hduzavvPIKs2bNYvXq1fTp0yfDJZKMys7DfeRd+fJBz57Qowc/vPIK9RYvdiWO//zHDWI4ZIjrPW6MyRyDB7t2xHM5eJD8a9e66uGAAKhTB4qcfYhy6tZ1l81n0Lp16/joo49YunQpQUFB9O/fn5kzZ1KlShX27dvHjz+6mRsOHDhA0aJFefXVVxk1ahTNmzfP8LkywkoW2ZkIBy+/HD76CDZtch37PvjADSfSpg3MmWPtGsZklYMH//m+JSa6xz6wYMECVq5cScOGDalbty5ffvklP//8M1WrViUuLo777ruP+fPnU+RcicoHrGSRU1StCq+84oYVGT/e3e/c2U0Be//97gqrFMMaG2MyID0lgGXL3EUpJ0640v+0aW6mzUymqvTr14+nnnrqjHVr167l008/ZezYsXzwwQe89dZbmX7+s7GSRU5TtCgMGwY//+wmYSpeHAYNcvNrPPIIxMf7O0JjcqfoaP6OiYGnnoLYWJ8kCoArr7ySd999l3379gHuqqkdO3bw+++/o6rcfPPNjBw5klWrVgFupNkjR474JJaUrGSRUwUHu1Ftu3Vz//G89BK88AK8+CLcfLNr12jUyN9RGpOrJDZpAlde6dNzXHbZZTz55JNceeWVJCYmEhwczLhx4wgMDOSOO+5AVRERRo0aBUDfvn0ZNGgQBQoUYMWKFRm6kiojLFnkBtHR8N57sH27mzN8/HjXZ6NFC5c0unQBb7x7Y0z2k3KIcoAePXrQo0ePM7b74Ycfzlh2yy230KFDhwzPzZFRVg2Vm0RGupLFzp0wejTs2gVdu7p2jTFjIJNm9TLG5D2WLHKjwoXdpYCbN7urp8qXd4/Dw2HoUFcCMcaYDLBkkZsFBsKNN8LXX8OKFdCxoythVKkCt9zi845FxmR3qurvELLMxT5XSxZ5RaNGMH266x0+bBh88QU0awZNm8KsWZCQ4O8IjclSoaGh7N+/P08kDFVl//79hIaGXvAxrIE7r4mIgFGj4IknYPJkd3159+5u1Nt774U773SX5xqTy4WHhxMfH8/vv/+e7n2OHTt2UT+4vpKeuEJDQwkPP2OuuXSzZJFXFSwI99wDAwa4kW9Hj4YHH3TDpffr5zr6Vani7yiN8Zng4GAqVaqUoX0WL15MvXr1fBTRhcuKuKwaKq8LCHA9wRctglWr3NVT48a5K6huuAG++gryQDHdGHNulizMP+rVc1VTv/wCjz3mGsZbtYKGDd0ouD4c0dIYk71ZsjBnKlsWnn7aTQH75pvw99/QqxdUquSGZ54/nwrTptnVVMbkIZYszNnlzw/9+7vJmObNg1q1XImjfXsqjR/vRr5dutTfURpjsoAlC3N+AQHQoQN8/rlr+BZBAI4fd/04Xn0V/vzT31EaY3zIkoXJmG7dIDSUxIAAN5hhsWJw332u6uq22+DLL61B3JhcyJKFyZjoaIiNZXu/fi4xbNgAP/wAd9zhLsFt3drNKf7CC7B3r7+jNcZkEksWJuOio9nRs+c/4/nXrQtjx8Kvv7qrqUqVgocecmNR3Xyzq76yGf2MydEsWZjMkz8/3H67u+T2p59cj/BFi+Caa1wHv6efdiPhGmNyHEsWxjcuvdQNl75rl5vRr0oVN8RIhQrQqRPExNh4VMbkIJYsjG+FhLhG8QULYMsWePhh+O47NyFTxYrw+ONucENjTLZmycJknSpV4NlnXWe/jz92Pcafe84tv/pqN9uf9RI3JlvyabIQkfYiEiciW0TkkTTWtxSRVSKSICI3pbG+sIjEi8hrvozTZLHgYFeymDvXTcQ0YgTExbk5NsqXd0Oox8X5O0pjTAo+SxYiEgiMBToANYFbRaRmqs12AH2A6Wc5zFPAV76K0WQDEREwfDhs3QqffgotW7oJmmrUcPenToWjR/0dpTF5ni9LFo2BLaq6VVVPADOBLik3UNXtqroWOOO6ShFpAJQGPvdhjCa7CAyE9u3dNLDx8W7Ojd273dVV5cq5K6vWrvV3lMbkWb5MFuWBnSkex3vLzktEAoAXgWE+iMtkd6VLu34amza5S2+vvRbefhsuvxyaNIHx4+HIEX9HaUyeIr6aUtBrg2ivqnd6j3sBTVR1UBrbTgLmqur73uNBQH5V/Y+I9AEanmW//kB/gNKlSzeYOXPmBcd75MgRChYseMH7+4rF5QQdPEjpL76g3CefUGD7dhLCwtjbti27O3bkcI0aIOKXuNLL4soYiytjLiauNm3afK+qDc+7oar65AZEA/NTPH4UePQs204CbkrxeBquPWM7sA84BDx/rvM1aNBAL8aiRYsuan9fsbhSSUxUXbpUtW9f1fz5VUG1Th3VV19V/fNPe70yyOLKmNwYF/CdpuM33ZfVUCuBaiJSSUTyAd2BmPTsqKo9VbWCqkbiqqKmqOoZV1OZPEjEDTMyYYJr0xg3zl1dde+9ULYsNZ591vUgt8EMjclUPksWqpoADALmAxuAd1V1vYiMFJHOACLSSETigZuBN0Vkva/iMblQ4cJw112uk9+qVdC3LyWXLnVXUdWs6XqQ//67v6M0JlfwaT8LVZ2nqtVVtYqqPuMtG66qMd79laoarqoFVLWEqtZK4xiTNI32CmNOU68evP46S997DyZOhOLFXX+N8uVdD/IvvrDBDI25CNaD2+QqiWFh0KcPLFkC69bBoEEQG+t6iFetCs8840bHNcZkiCULk3vVqgUvveQGM5wxw80h/vjjbjDDpB7kNpihMeliycLkfiEh0L27K2Fs3gwPPgjLl7vRbyMjXQ/y7dv9HaUx2ZolC5O3VK3qBi/cuRM+/BDq1HHzbFSu7ObdeP99G8zQmDRYsjB5U3Aw3HADzJvnShXDh7spYm++2Y1XldSD3BgDWLIwxrVhjBjh5tWYNw+aN4fRoyEqys0pPm0aHDvm7yiN8StLFsYkCQyEDh1c9dTOna66Kj4ebrvNDWZ4//3w44/+jtIYv7BkYUxaypSBRx5xVVGxsa49Y9w418bRtCn87382mKHJUyxZGHMuAQHQtq279PbXX1311OHDcOedrrSR1IPchhcxuZwlC2PSq0QJGDzYdfZbsgS6dnWTMzVqBPXrw+uvw4ED/o7SGJ+wZGFMRolAs2ZuWJHdu12SEIF77nGljd694ZtvrLRhchVLFsZcjCJFYMAAN5Dh99+7RPHRR3DFFf/0IP/0UypMmwbLlvk7WmMumCULYzJL/frwxhuutDFhAhQtCkOHwrXXUmn8eGjVyl2aa0wOZMnCmMxWoAD07QtLl7rLbUUQgJMn4brr3BDqL74IW7b4O1Jj0s2ShTG+1K0bhIaSGBDgxqjq2xcOHXLDp1erBrVrw7//DStW2BDqJluzZGGML0VHQ2ws2/v1g0WLXP+M1atdb/ExY6BUKRg1Cpo0ccOMDBgA8+fb+FQm27FkYYyvRUezo2dPlziSREbCfffBwoWwdy9MmeLWT50K7dtDyZJupNwZM+DgQb+FbkwSSxbG+Fvx4tCrlxvxdt8+N89G9+6weDH06AGXXOImbxo71g0/YowfWLIwJjsJDYWOHeGtt1yP8aVLYcgQ2LHDzfoXEQENG7ph1X/80fpymCxjycKY7CogwFVNjRoFGze6IdSffx7y5XNDqtepA1WqwAMPwJdf2qx/xqcsWRiTU9SoAQ8/7Eobv/7qSh+XXup6kLdu7QY/7NPHdQr86y9/R2tyGUsWxuREZcrAv/4Fn3wCv/8O773nhlePiYEbb3QN5J07u86Be/f6O1qTCwT5OwBjzEUqVAhuusndTp6Er7+G2bPh449hzpx/xrK6/nro0sX17zAmg6xkYUxuEhzshlQfM8ZNF/vDD/Dkk65a6sEHoXp1N2bVY49ZR0CTIZYsjMmtRKBuXZcsfvjBJY8xY1wV1n/+A02aEH3LLXD33fDZZ3D8uL8jNtmYJQtj8oqKFV1HwNhY144xdSoHa9eGd95x7R2XXOKGJ5k+3eblMGewNgtj8qLixeG22/gpPJxSTZu6nuQff+wayN99F4KC3BVWXbq4W0SEvyM2fmYlC2PyutBQuPba0zsCDh3qeovfey9UqAANGsDIkbB2rXUEzKMsWRhj/pHUEfD5510nwI0bXafAkBAYMQIuvxwqV3a9yhcvto6AeYglC2PM2UVFwUMPnd4RsFYtN8lTmzZQurSbHfDDD60jYC7n02QhIu1FJE5EtojII2msbykiq0QkQURuSrG8ord8tYisF5G7fRmnMSYdkjoCzp3rBjx8/303jtWcOdC1q+sI2KmTG4bdOgLmOj5r4BaRQGAscBUQD6wUkRhV/SnFZjuAPsCwVLvvBqJV9biIFATWefv+6qt4jTEZULCgSxBdu7qOgN98809HwLlz3WW70dH/dASsXt3fEZuL5MuSRWNgi6puVdUTwEygS8oNVHW7qq4FElMtP6GqSRd9h/g4TmPMxQgOdlVSL7/sJnVavdr17Th61FVhRUVBzZrw6KPw7bfWETCH8uWPcHlgZ4rH8d6ydBGRCBFZ6x1jlJUqjMkBRFwj+JNPwqpVriPgK69AuXLwwguutFG+PNx1F3z6qXUEzEFEfXQZnNcG0V5V7/Qe9wKaqOqgNLadBMxV1ffTWFcO+BjopKp7Uq3rD/QHKF26dIOZM2decLxHjhyhYMGCF7y/r1hcGWNxZUxWxhV0+DDFv/2WkkuWUHzFCoKOHiUhLIw/GjdmX/Pm/NG0KQmFCmV5XBmRG+Nq06bN96ra8LwbqqpPbkA0MD/F40eBR8+y7STgpnMca8K51qsqDRo00IuxaNGii9rfVyyujLG4MsZvcR09qvrJJ6r9+6uWLq0KqkFBqu3aqQ4Zojtuvll1yRL/xHahwsfUAAAgAElEQVQOufF9BL7TdPym+7IaaiVQTUQqiUg+oDsQk54dRSRcRMK8+8WAFkCczyI1xmStpI6Ab77pLsldtgyGDYMtW2D0aCLeew+aN3e3Z55xPcwPH/Z31Hmaz66GUtUEERkEzAcCgQmqul5ERuIyWYyINAI+AooBnUTk/1S1FnAp8KKIKCDAf1X1R1/Faozxo4AAaNrU3QoXhscfd43gIrB5s3uctF3t2q7dI+lWrZrbzvicT8eGUtV5wLxUy4anuL8SCE9jvy+AOr6MzRiTDbVuDSEhJB4/TkBIiLsct0YNWL7clT6WLYMZM1yJBNwYV02b/pM8Gjd283uYTGcDCRpjso/oaIiNZfuECVTu1889Bmjf3t3AlTo2bPgneSxbBvO8/0mt9OEz6UoWIlIFiFfXSa417r/+KaqaK8Yxnr1xNnN3zCVkZwjREdH+DseYvC06mh3Hj1M5+izfxYAAN+RIrVpw551u2Z9/WunDx9JbsvgAaCgiVYG3gNnAdOBaXwWWVeZumsv1s64HYMqOKSy8fSHNKzT3c1TGmAwpVizjpY+UCaR6dSt9nEd6k0Wi12B9A/Cqqr4qIj/4MrCssvLXlQiCopw4dYIbZt3A022fpledXoQFh/k7PGPMhThb6WPFin+Sx6xZbmBEOL300bSpK30ULuy/+LOh9CaLkyJyK9Ab6OQtC/ZNSFmrfZX2vLDkBY4nHCcoMIhiocW4a+5d/HvhvxnYcCD3NL6HUgVK+TtMY8zFKlYMrrnG3eD00se3355e+hA5s+0jj49vld5k0Re4G3hGVbeJSCVgqu/CyjrREdHE3h7LhEUT6NemH03Dm/LlL1/y4rIXGfnVSEYtGUWvOr0YEj2EmpfU9He4xpjMklbp48CB09s+UpU+LqtWzY20m9T2kYdKH+lKFupGir0PkjvJFVLVUb4MLCtFR0RzvMLx5Mbt1pGtaR3Zmrh9cYz+djST10xm/A/j6VC1A0Ojh9K2UlvE6jeNyX2KFj2z9LFxY3LyCF2wAIZ7V/+nLn00beoGTcylvw3p6sEtIotFpLCIFAdWAW+LyEu+Dc3/okpGMe66cewYvIORrUfy/e7vuXLqldR7sx5T1kzhxKkT/g7RGONLAQFuxNw77oDx41k5aZJr+/jsM5c0ypVzpY++feHSS6FECdcz/amnYMECOHTI388g06R3uI8iqnoIuBF3yWwT4ErfhZW9XFLgEp5o9QS/DP6F8Z3GczLxJL0/7k3ky5E89/Vz/HH0D3+HaIzJKkmljxEjXNL44w9Yvx7Gj4cbb4RffnGJ5Kqr3LZ16rhRdidOdKWUHDpEe3qTRZCIlAVuAeb6MJ5sLTQolDvq38G6Aev4tOen1C5Vm8cWPkbE6AgGzRvElj+2+DtEY0xWS1X6YP16V/qYP98N1Z5U+ujXz5U+SpbMkaWP9DZwj8SN8bREVVeKSGVgs+/Cyt5EhPZV29O+anvW7lnLS8te4q3v3+L1la9zfY3reSD6AZpHNLd2DWPyqqJF4eqr3Q1Ob/tIuvLqs89A1bVx1Kp15pVXAdlrzrf0NnC/B7yX4vFWoKuvgspJ6pSuw6TrJ/Fcu+d4bcVrvPHdG3y08SMal2/M0Oih3HjpjQQF2KgqxuRpSaWPpBIIwMGDp1959e678Pbbbl2xYv8MrhgdDU2a+P3Kq/QO9xEOvAokdW3+GrhfVeN9FVhOU7ZQWZ5p9wyPXfEYk9dMZvS3o+n2fjcqFqnI/U3u5476d1A4JO9cZmeMOY8iRc4sfcTFnd7rPD2lj2XLqDBtGoSE/DOWlg+kt5wzETcXRTnvNsdbZlIpkK8AAxsNZOM9G/m428dUKFKBBz5/gIjREQz7fBg7Du7wd4jGmOwoIMC1afTr50oY69b90/YxYgSEh8N7753e9tG0KbRsSaX//Q/atXMJxlfhpXO7S1R1oqomeLdJwCU+iyoXCAwIpEuNLnzV9ytW3LmCDlU78PK3L1N5TGV6fNCD7379zt8hGmOyu6TSx/Dhbs7y/fvhp5/gf/+Drl1h2zZISEBU4cQJWLzYZ6GkN1nsF5HbRCTQu90G7PdZVLlMo/KNmHnTTH6+72fub3I/czfNpdHbjWg1qRWzN84mUXPmpXTGmCyWuvTx8ccQFoYGBEC+fG4+EF+dOp3b9cNdNvsbsBu4Cejjo5hyrYpFK/LiNS8S/0A8L179ItsPbOf6WddT47UavLHyDf4++be/QzTG5CTe/B/b+vWD2Fj/t1mo6i+q2llVL1HVUqp6PXY11AUrHFKYB6If4Of7fmZm15kUDS3KwHkDiRgdweMLH+e3I7/5O0RjTE4RHc2Onj19migg/SWLtDyQaVHkUUEBQXSr3Y3ldy7n675f07JiS579+lkqvlyRvrP78uMem3bcGJM9XEyysB5nmUREaFGhBR91+4i4QXH8q/6/eHf9u9QZV4cH1z7I/C3zUVV/h2mMycMuJlnYr5cPVCtRjdeufY2dQ3byTNtn2PrXVtpPa0+dcXWY+MNEjicc93eIxpg86JzJQkQOi8ihNG6Hcf0tjI8UDyvOY1c8xowmM5jUZRKC0C+mHxVfrsjTXz3Nvr/3+TtEY0wecs5koaqFVLVwGrdCqmpjWGSBfAH56F23N2vuXsMXvb6gXtl6PLHoCSqMrsCAuQPYtH+Tv0M0xuQB2WukKnNWIsKVla/k056fsm7AOnpc1oMJqydQ47UadJ7RmS+3f2ntGsYYn7FkkQPVKlWL8Z3Hs2PwDp5o+QTL4pfRenJrGr3diOk/TufkqZP+DtEYk8tYssjBShcszf+1+T92DN7BuI7jOHLiCD0/7EnlVyrzwpIXOHjsoL9DNMbkEpYscoGw4DDuangXP93zE3NunUO14tV4aMFDhI8OZ8hnQ9h+YLu/QzTG5HCWLHKRAAnguurXsbD3Qr7v/z1dorrw2srXqPJKFW557xaWxy/3d4jGmBzKkkUuVb9sfd658R223b+NYdHD+Pznz2n6v6Y0n9CcDzd8yKnEU/4O0RiTg1iyyOXCC4cz6qpR7ByykzHtx7D78G66vtuV6q9V59Xlr3LkxBF/h2iMyQEsWeQRhUIKcV+T+9h872bev/l9ShcozX2f3UfE6AgeXfAouw7t8neIxphszKfJQkTai0iciGwRkUfSWN9SRFaJSIKI3JRieV0RWSYi60VkrYh082WceUlgQCBda3Zl6R1LWdpvKe0qteM/S/9D5JhIbv/odlb/ttrfIRpjsiGfJQsRCQTGAh2AmsCtIlIz1WY7cPNiTE+1/G/gdlWtBbQHXhaRor6KNa+Kjojm/VveZ/O9mxnYcCAfbviQem/Wo92UdszbPM8mZTLGJPNlyaIxsEVVt6rqCWAm0CXlBqq6XVXXAomplm9S1c3e/V+Bvdg0rj5TuVhlxnQYQ/wD8Yy6chRx++LoOL0jtV+vzdvfv83Rk0f9HaIxxs/EV0NEeNVK7VX1Tu9xL6CJqg5KY9tJwFxVfT+NdY2ByUAt1dP/1RWR/kB/gNKlSzeYOXPmBcd75MgRChYseMH7+4o/4jqZeJLFvy/mvfj32HxkM0WDi9KlXBe6lOtCsXzF/BZXelhcGWNxZUxujKtNmzbfq2rD826oqj654aZeHZ/icS/gtbNsOwm4KY3lZYE4oOn5ztegQQO9GIsWLbqo/X3Fn3ElJibqwq0L9brp1ykj0JCnQvRfMf/S6Wun652T79SlO5b6LbazsfcxYyyujMmNcQHfaTp+0305cuwuICLF43BvWbqISGHgE+DfqvptJsdm0kFEaFOpDW0qtWHjvo28/O3LTFw9kbdXvQ3AlF+mMK/nPNpVbufnSI0xvubLNouVQDURqSQi+YDuQEx6dvS2/wiYomlUTZmsV6NkDcZdN44Hmz2IeJMknkg8QYdpHbj9o9uJ3RprDeLG5GI+SxaqmgAMAuYDG4B3VXW9iIwUkc4AItJIROKBm4E3RWS9t/stQEugj4is9m51fRWrSb+O1ToSGhRKAAGEBIbQoWoHZsfN5sqpVxL5ciSPxT5G3L44f4dpjMlkPp3ASFXnAfNSLRue4v5KXPVU6v3eAd7xZWzmwkRHRBN7eywTFk2gX5t+REdEc/TkUWLiYpi8ZjKjloziuW+eo0n5JvS+vDfdanejeFhxf4dtjLlI1oPbZFh0RDQ9K/QkOiIacKPedqvdjXk95xE/JJ4XrnqBv07+xcB5Ayn7Ylluevcm5sTNsXk2jMnBLFmYTFW2UFmGNRvG2rvXsqr/KgY0HMBXv3xF55mdk4dMt17ixuQ8liyMT4gI9crW4+X2L7PrgV3M7j6bFhVaMHblWOq9WY/Lx13Oi0tf5Lcjv/k7VGNMOliyMD4XHBhM56jOfHDLB+weupux144lNCiUYV8MI/ylcDpO78isdbM4lnDM36EaY87CkoXJUiXyl2Bgo4Esv3M5G+7ZwEPNH2LtnrV0/6A7Zf5bhrvm3MXSnUuTOmUaY7IJSxbGb2qUrMGz7Z5l+/3b+aLXF3SK6sQ7P75D8wnNiXotiqe/eppfDvzi7zCNMViyMNlAYEAgV1a+kqk3TOW3ob8xsctEyhcuzxOLniByTCRtJrdh0upJHD5+2N+hGpNnWbIw2UqhkEL0qduHRb0Xse3+bYxsPZL4Q/H0nd2XMi+W4faPbmfB1gU2LawxWcyShcm2IotG8kSrJ9g0aBNL+i3htstuIyYuhqumXkXkGOstbkxWsmRhsj0RoVlEM97s9Ca7h+5mZteZ1Cldh1FLRlFjbA2ajG/C6ytf54+jf/g7VGNyLUsWJkdJ6i3+SY9P2PXALv571X85evIo98y7h7IvluXJ9U9ab3FjfMCShcmxyhQsw9BmQ1lz9xpW9V/FwIYDWXtwLZ1ndqb8S+UZ/Nlgftj9g12Ga0wm8OlAgsZkhaTe4vXK1qNjvo4cLX+UyWsm88Z3bzBm+RguK3UZvS/vTY/LelC2UFl/h2tMjmQlC5OrBAUE0SmqE+/f8n5yb/H8wfldb/HR4Vw77VrrLW7MBbBkYXKt4mHFGdhoIN/e+S0b7tnAw80f5se9P57WW3zJjiVWTWVMOliyMHlCyt7iC3otoHNUZ9758R1aTGxB9deq89SXT7H9wHZ/h2lMtmXJwuQpgQGBtKvcjik3TEnuLR5eOJzhi4dTaUwl2kxuw8QfJlpvcWNSsWRh8qyUvcW337+dp9o8RfyhePrF9KPMi2Xo9VEv6y1ujMeShTFAxaIVebzl46f1Fp8TNye5t/ijCx5l476N/g7TGL+xZGFMCil7i/827Ddm3TSLOqXr8MLSF7h07KXWW9zkWZYsjDmL0KBQbql1C5/0+IT4B+LP6C3e9d2uxMTFWG9xkydYsjAmHZJ6i68dsJYf7vqBgQ0H8vUvX9NlZhfrLW7yBEsWxmRQ3TJ1Gd1+NLse2EVM9xhaRbbije/eoP5b9akzrg7/Xfpfdh/e7e8wjclUNtyHMRcoODCYTlGd6BTViT+O/sGsdbOYvGYyD37xIA8veJhrqlxD78t7U7pAaabtmEbIzhCiI6L9HbYxF8SShTGZoHhYcQY0GsCARgOI2xfHlDVTmLp2Kt0/6J68zZQdU5jXYx7tKrfzY6TGXBirhjImk0WVjOKZds+wffB2+tXtl7z8xKkTXPPONVw77VrGfTeOXYd2+TFKYzLGkoUxPhIgAdxZ/07CgsIIIICQwBBuqnkTcfvjGPDJAMJHh9Po7UY89eVTrPltjTWOm2zNqqGM8aHoiGhib49lwqIJ9GvTj+iIaFSVDfs2EBMXQ0xcDE8ufpLhi4dTsUhFOkd1pnNUZ1pWbEm+wHz+Dt+YZJYsjPGx6Ihojlc4nty4LSLUvKQmNS+pySMtHuG3I7/xyaZPiNkUw/hV43l1xasUDilMh6od6BLVhQ7VOlA0tKifn4XJ6yxZGONnZQqW4Y76d3BH/Tv4++TfLNi6gJi4GOZsmsOs9bMICgiiZcWWdK7uSh2VilXyd8gmD/Jpm4WItBeROBHZIiKPpLG+pYisEpEEEbkp1brPROSAiMz1ZYzGZCf5g/PTOaoz4zuPZ/fQ3Sy7YxkPNnuQ3478xuD5g6n8SmXqvFGHxxc+zopdK0jURH+HbPIInyULEQkExgIdgJrArSJSM9VmO4A+wPQ0DvEC0MtX8RmT3QVIAE3Dm/Jsu2dZP3A9m+/dzEtXv0SJ/CV4/pvnaTK+CeVfKk//Of35ZNMnHD151N8hm1zMl9VQjYEtqroVQERmAl2An5I2UNXt3roz/j1S1VgRae3D+IzJUaoWr8qQ6CEMiR7CH0f/YN7mecTExTBz3UzeXvU2+YPzc3WVq+lcvTMdq3ekVIFS/g7Z5CLiq8v1vGql9qp6p/e4F9BEVQelse0kYK6qvp9qeWtgmKped5Zz9Af6A5QuXbrBzJkzLzjeI0eOULBgwQve31csrozJi3GdSDzBmgNrWLp/KUv3L2Xv8b0IQq3CtWhWohnNSzYnIiwCEcnSuC6GxZUxFxNXmzZtvlfVhufdUFV9cgNuAsaneNwLeO0s204CbkpjeWtcEjnv+Ro0aKAXY9GiRRe1v69YXBmT1+NKTEzUVb+u0hGLRmj9N+srI1BGoNVeqaZD5w/Vr7Z/pSdPnczyuDLK4sqYi4kL+E7T8Rvry2qoXUBEisfh3jJjjI+ICPXK1qNe2Xo82fpJdh7cyZxNc4iJi+GV5a/w4rIXKRFWgo7VO9K5emfCEsL8HbLJIXyZLFYC1USkEi5JdAd6+PB8xphUIopEMLDRQAY2Gsih44f4/OfPiYmLYe6muUxZM4VgCebKPVfSOaoznap3onzh8v4O2WRTPrsaSlUTgEHAfGAD8K6qrheRkSLSGUBEGolIPHAz8KaIrE/aX0S+Bt4D2olIvIhc46tYjckLCocU5qaaNzHlhinsGbaHxb0Xc33569m0f1Py8CMN32rIyC9H2vAj5gw+7ZSnqvOAeamWDU9xfyWueiqtfa/wZWzG5GVBAUG0imyFVlFatWrFxn0biYmLYXbcbEYsHsGTi5+kQpEKyR0BW0W2suFH8jjrwW1MHiciXHrJpVx6yaU83OJh9hzZwyebPyEmLob//fA/Xlv5WvLwI52jOtOhageKhRXzd9gmi1myMMacpnTB0vSr149+9fpx9OTRM4YfCZRAWlZsSZeoLnSK6kTlYpX9HbLJApYsjDFnFRYcljwbYKImsnLXyuTqqsHzBzN4/mBql6qdXF3VqHwjAsRmPsiNLFkYY9IlQAJoEt6EJuFNeKbdM/z8x8/Jl+WOWjKKZ795ljIFy9Cpeic6R3WmXaV2hAXbpbm5hSULY8wFqVK8CoObDmZw08H8efRPN/zIpn+GHwkLCuPqKlfTJaqLDT+SC1iyMMZctGJhxehZpyc96/TkxKkTfLn9y+TqqtlxsxGE6Ijo5OqqGiVrpDn8iMm+rHLRGJOp8gXm46oqV/Hqta/yy+Bf+OGuHxjRegTHE47zSOwj1Hy9JlGvRTHs82F89ctXJCQm+Dtkkw5WsjDG+IyIULdMXeqWqcvwVsOJPxTPnLg5xGyK4dUVr/LishcpHlacjtU60iWqC1dXuZp1e9cxbcc0QnaGJM8uaPzPkoUxJsuEFw5nQKMBDGg0gMPHD7vhRza54Uemrp1KUEAQiZqIqvLOzneI7RVLswrN/B22waqhjDF+UiikEF1rdmXy9ZPZM2wPX/b5kqbhTV2yQDmWcIxr3rmGHh/0YPyq8Wz7c5u/Q87TrGRhjPG7pHnG/3Plf2g3pR3HE44TGBBIs4hmLN6+mBnrZgAQWTSStpFtaVe5HW0i21C2UFk/R553WLIwxmQb0RHRxN4ey4RFE+jXph/REdGoKhv3bWThtoUs3L6QjzZ+xITVEwC4tOSltK3UlnaV2tEqshXFw4r7+RnkXpYsjDHZSnRENMcrHE9u3E45dtU9je/hVOIp1uxZw8JtC4ndFsuk1ZMYu3IsgpvLo12ldrSt1JYWFVpQMF/2m9Uup7JkYYzJUQIDAqlftj71y9ZnWLNhnDh1gpW7VhK7LZaF2xYyZvkYXlj6AkEBQTQNb0rbyLa0rdSWpuFNCQkK8Xf4OZYlC2NMjpYvMB/NKzSneYXmDG81nL9P/s2SHUuSq62e/vppRn41krCgMFpUaJFcbVW/bH0CAwL9HX6OYcnCGJOr5A/Oz1VVruKqKlcBcODYAb765avkaqtHYx8FoEhIEVpFtkpuMK91SS3rVX4OliyMMbla0dCidI5yw4wA7Dmyh8XbFydXW8XExQBQqkAp2kS2SW7zqFyssiWPFCxZGGPylNIFS9Otdje61e4GwC8HfkmusordGsus9bMAqFCkQnLiaFupLeUKlfNn2H5nycIYk6dVLFqRvvX60rdeX1SVTfs3JZc6ZsfNZuLqiQDUKFmDqHxR7C+1n9aRrSmRv4SfI89aliyMMcYjIkSVjCKqZBQDGw0kURNZ89ua5JLH51s/Z/Z7bhTdumXqJjeWX1Hxilx/ma4lC2OMOYsACaBe2XrUK1uPoc2GsmDhAvJXy5/cWJ40GGJQQBCNyzdOrrZqGt6U0KBQf4efqSxZGGNMOgUFBNEsohnNIprxeMvHOXryKEt3Lk2utnrm62d46qunCA0KdZfpen08GpRrQFBAzv65zdnRG2OMH4UFh9GucjvaVW4HwMFjB5Mv0124fSGPLXwMgMIhhWlVsVVyY3ntUrVz3FzlliyMMSaTFAktQqeoTnSK6gTA3r/2snj74uRqqzmb5gBwSf5LaFOpTXIfjyrFqmT7y3QtWRhjjI+UKlCKW2rdwi21bgFgx8EdLNq2iNhtscRui+Xd9e8CEFE4IrmxvG2ltpQvXN6fYafJkoUxxmSRCkUq0Ltub3rX7Y2qsvmPzcRujWXh9oXM3TSXyWsmA1C9RPXkUkfryNaUzF/Sz5FbsjDGGL8QEaqXqE71EtUZ0GgAiZrI2j1rXXvHtoW88+M7jPt+HACXl748udRxRcUrKBxSOMvjtWRhjDHZQIAEJM9X/kD0A5w8dZLvfv0uub1j7MqxvPTtSwRKII3LN05uLA+QgCyZs9yShTHGZEPBgcFER0QTHRHNv1v+m6Mnj7IsfllytdXz3zzPM18/k7z9tJ3TiL091mcJw5KFMcbkAGHBYcmlCYBDxw9x77x7mbp2Kopy4tQJFm9f7LNk4dMLfUWkvYjEicgWEXkkjfUtRWSViCSIyE2p1vUWkc3erbcv4zTGmJymcEhh7m54N6FBoQQQQL7AfLSObO2z8/ksWYhIIDAW6ADUBG4VkZqpNtsB9AGmp9q3OPAk0ARoDDwpIsV8FasxxuRESXOW96vUz6dVUODbaqjGwBZV3QogIjOBLsBPSRuo6nZvXWKqfa8BvlDVP7z1XwDtgRk+jNcYY3Kc1HOW+4ovk0V5YGeKx/G4ksKF7ntGLxUR6Q/0ByhdujSLFy++oEABjhw5clH7+4rFlTEWV8ZYXBmTl+PK0Q3cqvoW8BZAw4YNtXXr1hd8rMWLF3Mx+/uKxZUxFlfGWFwZk5fj8mUD9y4gIsXjcG+Zr/c1xhiTyXyZLFYC1USkkojkA7oDMencdz5wtYgU8xq2r/aWGWOM8QOfJQtVTQAG4X7kNwDvqup6ERkpIp0BRKSRiMQDNwNvish6b98/gKdwCWclMDKpsdsYY0zW82mbharOA+alWjY8xf2VuCqmtPadAEzwZXzGGGPSR1TV3zFkChH5HfjlIg5REtiXSeFkJosrYyyujLG4MiY3xlVRVS8530a5JllcLBH5TlUb+juO1CyujLG4Msbiypi8HFfOmtfPGGOMX1iyMMYYc16WLP7xlr8DOAuLK2MsroyxuDImz8ZlbRbGGGPOy0oWxhhjzsuShTHGmPPK08lCRCJEZJGI/CQi60Xkfn/HBCAioSKyQkTWeHH9n79jSklEAkXkBxGZ6+9YkojIdhH5UURWi8h3/o4niYgUFZH3RWSjiGwQEd+OI51OIhLlvVZJt0MiMjgbxDXE+8yvE5EZIhLq75gAROR+L6b1/n6dRGSCiOwVkXUplhUXkS+8yeK+8MX8P3k6WQAJwFBVrQk0Be5JY4ImfzgOtFXVy4G6QHsRaernmFK6HzeES3bTRlXrZrPr4McAn6lqDeByssnrpqpx3mtVF2gA/A185M+YRKQ8cB/QUFVrA4G4MeX8SkRqA//CzdFzOXCdiFT1Y0iTcPP7pPQIEKuq1YBY73GmytPJQlV3q+oq7/5h3Bf5jHkzspo6R7yHwd4tW1yJICLhQEdgvL9jye5EpAjQEvgfgKqeUNUD/o0qTe2An1X1YkZAyCxBQJiIBAH5gV/9HA/ApcByVf3bG/PuS+BGfwWjql8BqcfK6wJM9u5PBq7P7PPm6WSRkohEAvWA5f6NxPGqelYDe3GzBmaLuICXgYeA1LMb+psCn4vI996kWNlBJeB3YKJXbTdeRAr4O6g0dCcbzEKpqruA/+KmW94NHFTVz/0bFQDrgCtEpISI5Aeu5fQpFLKD0qq627v/G1A6s09gyQIQkYLAB8BgVT3k73gAVPWUV0UQDjT2isJ+JSLXAXtV9Xt/x5KGFqpaHzfn+z0i0tLfAeH+S64PvKGq9YC/8EH1wMXwpg/oDLyXDWIphvsPuRJQDiggIrf5NypQ1Q3AKOBz4DNgNXDKr0Gdg7r+EJleE5Hnk4WIBOMSxTRV/dDf8aTmVVss4sw6Sn9oDnQWke3ATKCtiLzj35Ac779SVHUvru69sX8jAtx0wPEpSoXv45JHdtIBWKWqe/wdCHAlsE1Vf1fVk8CHQDM/xwSAqv5PVRuoakvgT2CTv2NKZY+IlAXw/u7N7BPk6WQhIoKrT96gqi/5O54kInKJiBT17ocBVwEb/RsVqOqjqhquqpG4qouFqur3//xEpICIFEq6j5ssa9259/I9Vf0N2CkiUd6idsBPfgwpLbeSDZ9zbmkAAAQtSURBVKqgPDuApiKS3/tutiObXBAgIqW8vxVw7RXT/RvRGWKA3t793sDszD5Bjp6DOxM0B3oBP3rtAwCPefNw+FNZYLKIBOIS+ruqmm0uU82GSgMfud8XgoDpqvqZf0NKdi8wzavu2Qr09XM8ybzEehVwl79jAVDV5SLyPrAKd6XiD2Sf4TU+EJESwEngHn9eqCAiM4DWQElv8rgngeeBd0XkDtxUDbdk+nltuA9jjDHnk6eroYwxxqSPJQtjjDHnZcnCGGPMeVmyMMYYc16WLIwxxpyXJQuT44nIEe9vpIj0yORjP5bq8dLMPH5mE5E+IvKav+MwuY8lC5ObRAIZShbegHX/3975hFhZhWH894xIzCIERUJwSBeFq2hRgWBwFynUQhAlF0W4SlvUMggEt4q4C0FXBoq7jJyFoosRk2I0GUdFLEgXgUJI5J9wHMfXxXk/Ovd2xw//ROP1+cHhnu/cc95zvnu5vPf7Dt/zPIquZBERc+KJ4v+KfLbHmH/hZGEGie0UwbeJ9EWYJ2mnpNOSJiVtBpDUkXRS0vfkE9WSvksRwouNEKGk7RQF1AlJB7KtuYpRxr6QPhobq9hjlYfFgXwauYvss0PFt+QXSe9me9eVgaRRSZ1m7pzzoqTjkt7JOL9JWluFH8n2XyVtq2J9nPNNSNrTJIaMu0vSOWBO+G2YOUhEuLg81wW4na8dYLRq/xTYmvWXgDMUkboORdRvedV3Yb4OU6RCFtWx+8y1HjhG8Vx4hSJVsSRj/0URgBwCfqSIHPaueQzYlfUPgONZ3wR8XfUbBTpZD+D9rB+iCNvNp3gsTFTjrwGLqnN5iyKzfRiYn/12A59UcT/8v79Hl7ldXnS5DzPYrAHekLQhjxcArwH3gPGIuFL1/ULSuqyPZL8bj4i9CjgYETMUEbcTwNvAzYz9O0DKyCwDfugToxGu/Dn7tHGPonoKcB6YiohpSed7xh+LiBs5/7e51vsUk6PTeaEzzD9iczMUMU1jZsXJwgwyAj6PiKNdjeW2zp2e4/eAlRHxt6Qx4GnsPKeq+gyz/86m+vS5T/ft4Xod0xHR6PM8aMZHxIOevZdeDZ+gfBbfRMRXfdZxN5OeMbPiPQszSNwCXq6OjwKfpQw9kl6fxXxoAfBnJooVFIvdhulmfA8ngY25L7KY4og3/gzO4SrwpqQhSSM8mdT6ahVP5mGKY9opitXmhko9daGkV5/Bes0Lgq8szCAxCczkRu0+iv/1MuBsbjL/QX+7ySPAFkmXgMvAT9V7e4FJSWcj4qOq/RBlM/gc5Z/7lxFxPZPN03AKuELZeL9EUWB9XMYpt5WWAvsj4gyApK0UN8EhUj2VolBqTCtWnTXGGNOKb0MZY4xpxcnCGGNMK04WxhhjWnGyMMYY04qThTHGmFacLIwxxrTiZGGMMaaVhwphJaN/JOSoAAAAAElFTkSuQmCC\n",
+      "text/plain": [
+       "<Figure size 432x288 with 1 Axes>"
+      ]
+     },
+     "metadata": {
+      "needs_background": "light"
+     },
+     "output_type": "display_data"
+    }
+   ],
+   "source": [
+    "# get loss\n",
+    "train_loss_proxy = %sql SELECT training_loss FROM iris_model_summary;\n",
+    "test_loss_proxy = %sql SELECT validation_loss FROM iris_model_summary;\n",
+    "\n",
+    "# reshape to np arrays\n",
+    "train_loss = np.array(train_loss_proxy).reshape(num_points)\n",
+    "test_loss = np.array(test_loss_proxy).reshape(num_points)\n",
+    "\n",
+    "#plot\n",
+    "plt.title('Iris validation loss by iteration - transfer learn')\n",
+    "plt.xlabel('Iteration number')\n",
+    "plt.ylabel('Loss')\n",
+    "plt.grid(True)\n",
+    "plt.plot(iters, train_loss, 'g.-', label='Train')\n",
+    "plt.plot(iters, test_loss, 'r.-', label='Test')\n",
+    "plt.legend();"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/MADlib-Keras-cifar10-cnn-v3.ipynb b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-cifar10-cnn-v3.ipynb
similarity index 99%
rename from community-artifacts/Deep-learning/MADlib-Keras-cifar10-cnn-v3.ipynb
rename to community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-cifar10-cnn-v3.ipynb
index 987ff4b..f7053ef 100644
--- a/community-artifacts/Deep-learning/MADlib-Keras-cifar10-cnn-v3.ipynb
+++ b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-cifar10-cnn-v3.ipynb
@@ -59,46 +59,21 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 2,
+   "execution_count": 1,
    "metadata": {
     "scrolled": true
    },
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
     "%load_ext sql"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 4,
+   "execution_count": 2,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: gpadmin@madlib'"
-      ]
-     },
-     "execution_count": 4,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP (PM demo machine) - direct external IP access\n",
-    "#%sql postgresql://gpadmin@34.67.65.96:5432/madlib\n",
-    "\n",
     "# Greenplum Database 5.x on GCP - via tunnel\n",
     "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
@@ -108,7 +83,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 5,
+   "execution_count": 3,
    "metadata": {},
    "outputs": [
     {
@@ -126,15 +101,15 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.17-dev, git revision: rel/v1.16-54-gec5614f, cmake configuration time: Wed Dec 18 17:08:05 UTC 2019, build type: release, build system: Linux-3.10.0-1062.4.3.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
-     "execution_count": 5,
+     "execution_count": 3,
      "metadata": {},
      "output_type": "execute_result"
     }
@@ -155,32 +130,17 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 6,
+   "execution_count": 5,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "Using TensorFlow backend.\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Couldn't import dot_parser, loading of dot files will not be possible.\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
     "from __future__ import print_function\n",
-    "import keras\n",
-    "from keras.datasets import cifar10\n",
-    "from keras.preprocessing.image import ImageDataGenerator\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense, Dropout, Activation, Flatten\n",
-    "from keras.layers import Conv2D, MaxPooling2D\n",
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.datasets import cifar10\n",
+    "from tensorflow.keras.preprocessing.image import ImageDataGenerator\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense, Dropout, Activation, Flatten\n",
+    "from tensorflow.keras.layers import Conv2D, MaxPooling2D\n",
     "import os\n",
     "\n",
     "batch_size = 32\n",
@@ -197,7 +157,7 @@
   },
   {
    "cell_type": "code",
-   "execution_count": 7,
+   "execution_count": 6,
    "metadata": {},
    "outputs": [],
    "source": [
diff --git a/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-cifar10-inference-v1.ipynb b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-cifar10-inference-v1.ipynb
new file mode 100644
index 0000000..eb3daa2
--- /dev/null
+++ b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-cifar10-inference-v1.ipynb
@@ -0,0 +1,718 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Inference for CIFAR-10 dataset using predict BYOM\n",
+    "The predict BYOM function allows you to do inference using models that have not been trained with MADlib, but rather imported or created elsewhere. It was added in MADlib 1.17.\n",
+    "\n",
+    "In this workbook we train a model in Python using\n",
+    "https://keras.io/examples/cifar10_cnn/\n",
+    "and run inference on the validation set.\n",
+    "\n",
+    "## Table of contents\n",
+    "\n",
+    "<a href=\"#setup\">1. Setup</a>\n",
+    "\n",
+    "<a href=\"#train_model\">2. Train model in Python</a>\n",
+    "\n",
+    "<a href=\"#load_model\">3. Load model into table</a>\n",
+    "\n",
+    "<a href=\"#load_images\">4. Get validation data set and load into table</a>\n",
+    "\n",
+    "<a href=\"#inference\">5. Inference</a>"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"setup\"></a>\n",
+    "# 1. Setup"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g9d9f756, cmake configuration time: Thu Mar  4 23:11:53 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train_model\"></a>\n",
+    "# 2. Train model in Python\n",
+    "\n",
+    "Train a model in Python using https://keras.io/examples/cifar10_cnn/\n",
+    "\n",
+    "Define model"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "x_train shape: (50000, 32, 32, 3)\n",
+      "50000 train samples\n",
+      "10000 test samples\n",
+      "WARNING:tensorflow:From /Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/tensorflow/python/ops/init_ops.py:1251: calling __init__ (from tensorflow.python.ops.init_ops) with dtype is deprecated and will be removed in a future version.\n",
+      "Instructions for updating:\n",
+      "Call initializer instance with the dtype argument instead of passing it to the constructor\n"
+     ]
+    }
+   ],
+   "source": [
+    "from __future__ import print_function\n",
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.datasets import cifar10\n",
+    "from tensorflow.keras.preprocessing.image import ImageDataGenerator\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense, Dropout, Activation, Flatten\n",
+    "from tensorflow.keras.layers import Conv2D, MaxPooling2D\n",
+    "import os\n",
+    "\n",
+    "batch_size = 32\n",
+    "num_classes = 10\n",
+    "epochs = 2\n",
+    "data_augmentation = True\n",
+    "num_predictions = 20\n",
+    "#save_dir = os.path.join(os.getcwd(), 'saved_models')\n",
+    "#model_name = 'keras_cifar10_trained_model.h5'\n",
+    "\n",
+    "# The data, split between train and test sets:\n",
+    "(x_train, y_train), (x_test, y_test) = cifar10.load_data()\n",
+    "print('x_train shape:', x_train.shape)\n",
+    "print(x_train.shape[0], 'train samples')\n",
+    "print(x_test.shape[0], 'test samples')\n",
+    "\n",
+    "# Convert class vectors to binary class matrices.\n",
+    "y_train = keras.utils.to_categorical(y_train, num_classes)\n",
+    "y_test = keras.utils.to_categorical(y_test, num_classes)\n",
+    "\n",
+    "model = Sequential()\n",
+    "model.add(Conv2D(32, (3, 3), padding='same',\n",
+    "                 input_shape=x_train.shape[1:]))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(Conv2D(32, (3, 3)))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(MaxPooling2D(pool_size=(2, 2)))\n",
+    "model.add(Dropout(0.25))\n",
+    "\n",
+    "model.add(Conv2D(64, (3, 3), padding='same'))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(Conv2D(64, (3, 3)))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(MaxPooling2D(pool_size=(2, 2)))\n",
+    "model.add(Dropout(0.25))\n",
+    "\n",
+    "model.add(Flatten())\n",
+    "model.add(Dense(512))\n",
+    "model.add(Activation('relu'))\n",
+    "model.add(Dropout(0.5))\n",
+    "model.add(Dense(num_classes))\n",
+    "model.add(Activation('softmax'))\n",
+    "\n",
+    "# initiate RMSprop optimizer\n",
+    "opt = keras.optimizers.RMSprop(lr=0.0001, decay=1e-6)\n",
+    "\n",
+    "# Let's train the model using RMSprop\n",
+    "model.compile(loss='categorical_crossentropy',\n",
+    "              optimizer=opt,\n",
+    "              metrics=['accuracy']);"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "data": {
+      "text/plain": [
+       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.2.4-tf\", \"config\": {\"layers\": [{\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"conv2d\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 32, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"batch_input_shape\": [null, 32, 32, 3], \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation\"}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"conv2d_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"valid\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 32, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_1\"}}, {\"class_name\": \"MaxPooling2D\", \"config\": {\"name\": \"max_pooling2d\", \"dtype\": \"float32\", \"trainable\": true, \"data_format\": \"channels_last\", \"pool_size\": [2, 2], \"padding\": \"valid\", \"strides\": [2, 2]}}, {\"class_name\": \"Dropout\", \"config\": {\"name\": \"dropout\", \"dtype\": \"float32\", \"trainable\": true, \"rate\": 0.25, \"seed\": null, \"noise_shape\": null}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"conv2d_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 64, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_2\"}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"conv2d_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"valid\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 64, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_3\"}}, {\"class_name\": \"MaxPooling2D\", \"config\": {\"name\": \"max_pooling2d_1\", \"dtype\": \"float32\", \"trainable\": true, \"data_format\": \"channels_last\", \"pool_size\": [2, 2], \"padding\": \"valid\", \"strides\": [2, 2]}}, {\"class_name\": \"Dropout\", \"config\": {\"name\": \"dropout_1\", \"dtype\": \"float32\", \"trainable\": true, \"rate\": 0.25, \"seed\": null, \"noise_shape\": null}}, {\"class_name\": \"Flatten\", \"config\": {\"dtype\": \"float32\", \"trainable\": true, \"name\": \"flatten\", \"data_format\": \"channels_last\"}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 512, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_4\"}}, {\"class_name\": \"Dropout\", \"config\": {\"name\": \"dropout_2\", \"dtype\": \"float32\", \"trainable\": true, \"rate\": 0.5, \"seed\": null, \"noise_shape\": null}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"GlorotUniform\", \"config\": {\"dtype\": \"float32\", \"seed\": null}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {\"dtype\": \"float32\"}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Activation\", \"config\": {\"dtype\": \"float32\", \"activation\": \"softmax\", \"trainable\": true, \"name\": \"activation_5\"}}], \"name\": \"sequential\"}, \"backend\": \"tensorflow\"}'"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "model.to_json()"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Using real-time data augmentation.\n",
+      "Epoch 1/2\n",
+      "1563/1563 [==============================] - 107s 69ms/step - loss: 1.8637 - acc: 0.3142 - val_loss: 1.6037 - val_acc: 0.4154\n",
+      "Epoch 2/2\n",
+      "1563/1563 [==============================] - 116s 74ms/step - loss: 1.5880 - acc: 0.4174 - val_loss: 1.4362 - val_acc: 0.4754\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "<tensorflow.python.keras.callbacks.History at 0x14dfc98d0>"
+      ]
+     },
+     "execution_count": 7,
+     "metadata": {},
+     "output_type": "execute_result"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "10000/10000 [==============================] - 7s 698us/sample - loss: 1.4365 - acc: 0.4754\n",
+      "Test loss: 1.4364811393737793\n",
+      "Test accuracy: 0.4754\n"
+     ]
+    }
+   ],
+   "source": [
+    "x_train = x_train.astype('float32')\n",
+    "x_test = x_test.astype('float32')\n",
+    "x_train /= 255\n",
+    "x_test /= 255\n",
+    "\n",
+    "if not data_augmentation:\n",
+    "    print('Not using data augmentation.')\n",
+    "    model.fit(x_train, y_train,\n",
+    "              batch_size=batch_size,\n",
+    "              epochs=epochs,\n",
+    "              validation_data=(x_test, y_test),\n",
+    "              shuffle=True)\n",
+    "else:\n",
+    "    print('Using real-time data augmentation.')\n",
+    "    # This will do preprocessing and realtime data augmentation:\n",
+    "    datagen = ImageDataGenerator(\n",
+    "        featurewise_center=False,  # set input mean to 0 over the dataset\n",
+    "        samplewise_center=False,  # set each sample mean to 0\n",
+    "        featurewise_std_normalization=False,  # divide inputs by std of the dataset\n",
+    "        samplewise_std_normalization=False,  # divide each input by its std\n",
+    "        zca_whitening=False,  # apply ZCA whitening\n",
+    "        zca_epsilon=1e-06,  # epsilon for ZCA whitening\n",
+    "        rotation_range=0,  # randomly rotate images in the range (degrees, 0 to 180)\n",
+    "        # randomly shift images horizontally (fraction of total width)\n",
+    "        width_shift_range=0.1,\n",
+    "        # randomly shift images vertically (fraction of total height)\n",
+    "        height_shift_range=0.1,\n",
+    "        shear_range=0.,  # set range for random shear\n",
+    "        zoom_range=0.,  # set range for random zoom\n",
+    "        channel_shift_range=0.,  # set range for random channel shifts\n",
+    "        # set mode for filling points outside the input boundaries\n",
+    "        fill_mode='nearest',\n",
+    "        cval=0.,  # value used for fill_mode = \"constant\"\n",
+    "        horizontal_flip=True,  # randomly flip images\n",
+    "        vertical_flip=False,  # randomly flip images\n",
+    "        # set rescaling factor (applied before any other transformation)\n",
+    "        rescale=None,\n",
+    "        # set function that will be applied on each input\n",
+    "        preprocessing_function=None,\n",
+    "        # image data format, either \"channels_first\" or \"channels_last\"\n",
+    "        data_format=None,\n",
+    "        # fraction of images reserved for validation (strictly between 0 and 1)\n",
+    "        validation_split=0.0)\n",
+    "\n",
+    "    # Compute quantities required for feature-wise normalization\n",
+    "    # (std, mean, and principal components if ZCA whitening is applied).\n",
+    "    datagen.fit(x_train)\n",
+    "\n",
+    "    # Fit the model on the batches generated by datagen.flow().\n",
+    "    model.fit_generator(datagen.flow(x_train, y_train,\n",
+    "                                     batch_size=batch_size),\n",
+    "                        epochs=epochs,\n",
+    "                        validation_data=(x_test, y_test),\n",
+    "                        workers=1)\n",
+    "\n",
+    "# Save model and weights\n",
+    "#if not os.path.isdir(save_dir):\n",
+    "#    os.makedirs(save_dir)\n",
+    "#model_path = os.path.join(save_dir, model_name)\n",
+    "#model.save(model_path)\n",
+    "#print('Saved trained model at %s ' % model_path)\n",
+    "\n",
+    "# Score trained model.\n",
+    "scores = model.evaluate(x_test, y_test, verbose=1)\n",
+    "print('Test loss:', scores[0])\n",
+    "print('Test accuracy:', scores[1])"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_model\"></a>\n",
+    "# 3.  Load model into table\n",
+    "\n",
+    "Load the model architecture and weights into the model architecture table"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>model_id</th>\n",
+       "        <th>name</th>\n",
+       "        <th>description</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>1</td>\n",
+       "        <td>CIFAR10 model</td>\n",
+       "        <td>CNN model with weights trained on CIFAR10.</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(1, u'CIFAR10 model', u'CNN model with weights trained on CIFAR10.')]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "import psycopg2 as p2\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
+    "#conn = p2.connect('postgresql://fmcquillan@localhost:5432/madlib')\n",
+    "cur = conn.cursor()\n",
+    "\n",
+    "from keras.layers import *\n",
+    "from keras import Sequential\n",
+    "import numpy as np\n",
+    "\n",
+    "# get weights, flatten and serialize\n",
+    "weights = model.get_weights()\n",
+    "weights_flat = [w.flatten() for w in weights]\n",
+    "weights1d =  np.concatenate(weights_flat).ravel()\n",
+    "weights_bytea = p2.Binary(weights1d.tostring())\n",
+    "\n",
+    "%sql DROP TABLE IF EXISTS model_arch_library_cifar10;\n",
+    "query = \"SELECT madlib.load_keras_model('model_arch_library_cifar10', %s,%s,%s,%s)\"\n",
+    "cur.execute(query,[model.to_json(), weights_bytea, \"CIFAR10 model\", \"CNN model with weights trained on CIFAR10.\"])\n",
+    "conn.commit()\n",
+    "\n",
+    "# check weights loaded OK\n",
+    "%sql SELECT model_id, name, description FROM model_arch_library_cifar10;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_images\"></a>\n",
+    "# 4. Get validation data set and load into table\n",
+    "\n",
+    "First set up image loader using the script called <em>madlib_image_loader.py</em> located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "import sys\n",
+    "import os\n",
+    "madlib_site_dir = '/Users/fmcquillan/Documents/Product/MADlib/Demos/data'\n",
+    "sys.path.append(madlib_site_dir)\n",
+    "\n",
+    "# Import image loader module\n",
+    "from madlib_image_loader import ImageLoader, DbCredentials\n",
+    "\n",
+    "# Specify database credentials, for connecting to db\n",
+    "#db_creds = DbCredentials(user='fmcquillan',\n",
+    "#                         host='localhost',\n",
+    "#                         port='5432',\n",
+    "#                         password='')\n",
+    "\n",
+    "# Specify database credentials, for connecting to db\n",
+    "db_creds = DbCredentials(user='gpadmin', \n",
+    "                         db_name='madlib',\n",
+    "                         host='localhost',\n",
+    "                         port='8000',\n",
+    "                         password='')\n",
+    "\n",
+    "# Initialize ImageLoader (increase num_workers to run faster)\n",
+    "iloader = ImageLoader(num_workers=5, db_creds=db_creds)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Next load CIFAR-10 data from Keras consisting of 50,000 32x32 color training images, labeled over 10 categories, and 10,000 test images."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "MainProcess: Connected to madlib db.\n",
+      "Executing: CREATE TABLE cifar_10_test_data (id SERIAL, x REAL[], y TEXT)\n",
+      "CREATE TABLE\n",
+      "Created table cifar_10_test_data in madlib db\n",
+      "Spawning 5 workers...\n",
+      "Initializing PoolWorker-1 [pid 95042]\n",
+      "PoolWorker-1: Created temporary directory /tmp/madlib_dTZhEGBDFE\n",
+      "Initializing PoolWorker-2 [pid 95043]\n",
+      "PoolWorker-2: Created temporary directory /tmp/madlib_ctWjbhcjwz\n",
+      "Initializing PoolWorker-3 [pid 95044]\n",
+      "PoolWorker-3: Created temporary directory /tmp/madlib_nx9VuMScrX\n",
+      "Initializing PoolWorker-4 [pid 95045]\n",
+      "PoolWorker-4: Created temporary directory /tmp/madlib_thkphNCw4r\n",
+      "Initializing PoolWorker-5 [pid 95046]\n",
+      "PoolWorker-5: Created temporary directory /tmp/madlib_037luEXgEL\n",
+      "PoolWorker-2: Connected to madlib db.\n",
+      "PoolWorker-3: Connected to madlib db.\n",
+      "PoolWorker-1: Connected to madlib db.\n",
+      "PoolWorker-5: Connected to madlib db.\n",
+      "PoolWorker-4: Connected to madlib db.\n",
+      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_nx9VuMScrX/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_dTZhEGBDFE/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_ctWjbhcjwz/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_037luEXgEL/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_thkphNCw4r/cifar_10_test_data0000.tmp\n",
+      "PoolWorker-3: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_nx9VuMScrX/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-1: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_dTZhEGBDFE/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-2: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-4: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-5: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_ctWjbhcjwz/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_thkphNCw4r/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_037luEXgEL/cifar_10_test_data0001.tmp\n",
+      "PoolWorker-3: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-1: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-2: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-4: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-5: Loaded 1000 images into cifar_10_test_data\n",
+      "PoolWorker-1: Removed temporary directory /tmp/madlib_dTZhEGBDFE\n",
+      "PoolWorker-2: Removed temporary directory /tmp/madlib_ctWjbhcjwz\n",
+      "PoolWorker-3: Removed temporary directory /tmp/madlib_nx9VuMScrX\n",
+      "PoolWorker-5: Removed temporary directory /tmp/madlib_037luEXgEL\n",
+      "PoolWorker-4: Removed temporary directory /tmp/madlib_thkphNCw4r\n",
+      "Done!  Loaded 10000 images in 108.267487049s\n",
+      "5 workers terminated.\n"
+     ]
+    }
+   ],
+   "source": [
+    "from keras.datasets import cifar10\n",
+    "\n",
+    "# Load dataset into np array\n",
+    "(x_train, y_train), (x_test, y_test) = cifar10.load_data()\n",
+    "\n",
+    "%sql DROP TABLE IF EXISTS cifar_10_test_data;\n",
+    "\n",
+    "# Save images to temporary directories and load into database\n",
+    "#iloader.load_dataset_from_np(x_train, y_train, 'cifar_10_train_data', append=False)\n",
+    "iloader.load_dataset_from_np(x_test, y_test, 'cifar_10_test_data', append=False)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"inference\"></a>\n",
+    "# 5. Inference"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 11,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n"
+     ]
+    },
+    {
+     "ename": "InternalError",
+     "evalue": "(psycopg2.errors.InternalError_) plpy.Error: Unable to get number of classes from model architecture. (plpython.c:5038)\nCONTEXT:  Traceback (most recent call last):\n  PL/Python function \"madlib_keras_predict_byom\", line 23, in <module>\n    madlib_keras_predict.PredictBYOM(**globals())\n  PL/Python function \"madlib_keras_predict_byom\", line 42, in wrapper\n  PL/Python function \"madlib_keras_predict_byom\", line 314, in __init__\n  PL/Python function \"madlib_keras_predict_byom\", line 326, in validate_and_set_defaults\n  PL/Python function \"madlib_keras_predict_byom\", line 207, in set_default_class_values\n  PL/Python function \"madlib_keras_predict_byom\", line 75, in get_num_classes\nPL/Python function \"madlib_keras_predict_byom\"\n\n[SQL: SELECT madlib.madlib_keras_predict_byom('model_arch_library_cifar10',  -- model arch table\n                                         1,                            -- model arch id\n                                        'cifar_10_test_data',          -- test_table\n                                        'id',                          -- id column\n                                        'x',                           -- independent var\n                                        'cifar10_predict_byom',        -- output table\n                                        'response',                    -- prediction type\n                                         FALSE,                        -- use gpus\n                                         NULL,                         -- class values\n                                         255.0                         -- normalizing const\n                                   );]\n(Background on this error at: http://sqlalche.me/e/2j85)",
+     "output_type": "error",
+     "traceback": [
+      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
+      "\u001b[0;31mInternalError\u001b[0m                             Traceback (most recent call last)",
+      "\u001b[0;32m<ipython-input-11-d7da0ccca3f1>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m()\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0mget_ipython\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrun_cell_magic\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34mu'sql'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34mu''\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34mu\"DROP TABLE IF EXISTS cifar10_predict_byom;\\n\\nSELECT madlib.madlib_keras_predict_byom('model_arch_library_cifar10',  -- model arch table\\n                                         1,                            -- model arch id\\n                                        'cifar_10_test_data',          -- test_table\\n                                        'id',                          -- id column\\n                                        'x',                           -- independent var\\n                                        'cifar10_predict_byom',        -- output table\\n                                        'response',                    -- prediction type\\n                                         FALSE,                        -- use gpus\\n                                         NULL,                         -- class values\\n                                         255.0                         -- normalizing const\\n                                   );\\nSELECT * FROM cifar10_predict_byom ORDER BY id LIMIT 10;\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/IPython/core/interactiveshell.pyc\u001b[0m in \u001b[0;36mrun_cell_magic\u001b[0;34m(self, magic_name, line, cell)\u001b[0m\n\u001b[1;32m   2115\u001b[0m             \u001b[0mmagic_arg_s\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mvar_expand\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mline\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstack_depth\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2116\u001b[0m             \u001b[0;32mwith\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbuiltin_trap\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2117\u001b[0;31m                 \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mfn\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mmagic_arg_s\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcell\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2118\u001b[0m             \u001b[0;32mreturn\u001b[0m \u001b[0mresult\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2119\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m</Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/decorator.pyc:decorator-gen-124>\u001b[0m in \u001b[0;36mexecute\u001b[0;34m(self, line, cell, local_ns)\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/IPython/core/magic.pyc\u001b[0m in \u001b[0;36m<lambda>\u001b[0;34m(f, *a, **k)\u001b[0m\n\u001b[1;32m    186\u001b[0m     \u001b[0;31m# but it's overkill for just that one bit of state.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    187\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mmagic_deco\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0marg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 188\u001b[0;31m         \u001b[0mcall\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mlambda\u001b[0m \u001b[0mf\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mf\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    189\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    190\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mcallable\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0marg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m</Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/decorator.pyc:decorator-gen-123>\u001b[0m in \u001b[0;36mexecute\u001b[0;34m(self, line, cell, local_ns)\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/IPython/core/magic.pyc\u001b[0m in \u001b[0;36m<lambda>\u001b[0;34m(f, *a, **k)\u001b[0m\n\u001b[1;32m    186\u001b[0m     \u001b[0;31m# but it's overkill for just that one bit of state.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    187\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mmagic_deco\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0marg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 188\u001b[0;31m         \u001b[0mcall\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mlambda\u001b[0m \u001b[0mf\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mf\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    189\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    190\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mcallable\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0marg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sql/magic.pyc\u001b[0m in \u001b[0;36mexecute\u001b[0;34m(self, line, cell, local_ns)\u001b[0m\n\u001b[1;32m    135\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    136\u001b[0m         \u001b[0;32mtry\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 137\u001b[0;31m             \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0msql\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrun\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrun\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mconn\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparsed\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;34m'sql'\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0muser_ns\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    138\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    139\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0mresult\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mNone\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0misinstance\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mresult\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstr\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcolumn_local_vars\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sql/run.pyc\u001b[0m in \u001b[0;36mrun\u001b[0;34m(conn, sql, config, user_namespace)\u001b[0m\n\u001b[1;32m    361\u001b[0m             \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    362\u001b[0m                 \u001b[0mtxt\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0msqlalchemy\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msql\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtext\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mstatement\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 363\u001b[0;31m                 \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mconn\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msession\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mexecute\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtxt\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0muser_namespace\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    364\u001b[0m             \u001b[0m_commit\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mconn\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mconn\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mconfig\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mconfig\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    365\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0mresult\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0mconfig\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfeedback\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36mexecute\u001b[0;34m(self, object_, *multiparams, **params)\u001b[0m\n\u001b[1;32m    980\u001b[0m             \u001b[0;32mraise\u001b[0m \u001b[0mexc\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mObjectNotExecutableError\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mobject_\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    981\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 982\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mmeth\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmultiparams\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparams\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    983\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    984\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0m_execute_function\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mfunc\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmultiparams\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparams\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/sql/elements.pyc\u001b[0m in \u001b[0;36m_execute_on_connection\u001b[0;34m(self, connection, multiparams, params)\u001b[0m\n\u001b[1;32m    285\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0m_execute_on_connection\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mconnection\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmultiparams\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparams\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    286\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msupports_execution\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 287\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mconnection\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_execute_clauseelement\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmultiparams\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparams\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    288\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    289\u001b[0m             \u001b[0;32mraise\u001b[0m \u001b[0mexc\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mObjectNotExecutableError\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36m_execute_clauseelement\u001b[0;34m(self, elem, multiparams, params)\u001b[0m\n\u001b[1;32m   1099\u001b[0m             \u001b[0mdistilled_params\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1100\u001b[0m             \u001b[0mcompiled_sql\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1101\u001b[0;31m             \u001b[0mdistilled_params\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1102\u001b[0m         )\n\u001b[1;32m   1103\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_has_events\u001b[0m \u001b[0;32mor\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mengine\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_has_events\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36m_execute_context\u001b[0;34m(self, dialect, constructor, statement, parameters, *args)\u001b[0m\n\u001b[1;32m   1248\u001b[0m         \u001b[0;32mexcept\u001b[0m \u001b[0mBaseException\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1249\u001b[0m             self._handle_dbapi_exception(\n\u001b[0;32m-> 1250\u001b[0;31m                 \u001b[0me\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparameters\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcursor\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcontext\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1251\u001b[0m             )\n\u001b[1;32m   1252\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36m_handle_dbapi_exception\u001b[0;34m(self, e, statement, parameters, cursor, context)\u001b[0m\n\u001b[1;32m   1474\u001b[0m                 \u001b[0mutil\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mraise_from_cause\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mnewraise\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexc_info\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1475\u001b[0m             \u001b[0;32melif\u001b[0m \u001b[0mshould_wrap\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1476\u001b[0;31m                 \u001b[0mutil\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mraise_from_cause\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0msqlalchemy_exception\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexc_info\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1477\u001b[0m             \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1478\u001b[0m                 \u001b[0mutil\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mreraise\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0mexc_info\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/util/compat.pyc\u001b[0m in \u001b[0;36mraise_from_cause\u001b[0;34m(exception, exc_info)\u001b[0m\n\u001b[1;32m    396\u001b[0m     \u001b[0mexc_type\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexc_value\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexc_tb\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mexc_info\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    397\u001b[0m     \u001b[0mcause\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mexc_value\u001b[0m \u001b[0;32mif\u001b[0m \u001b[0mexc_value\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mexception\u001b[0m \u001b[0;32melse\u001b[0m \u001b[0mNone\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 398\u001b[0;31m     \u001b[0mreraise\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtype\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mexception\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexception\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtb\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mexc_tb\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcause\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mcause\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    399\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    400\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/base.pyc\u001b[0m in \u001b[0;36m_execute_context\u001b[0;34m(self, dialect, constructor, statement, parameters, *args)\u001b[0m\n\u001b[1;32m   1244\u001b[0m                 \u001b[0;32mif\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mevt_handled\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1245\u001b[0m                     self.dialect.do_execute(\n\u001b[0;32m-> 1246\u001b[0;31m                         \u001b[0mcursor\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparameters\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcontext\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1247\u001b[0m                     )\n\u001b[1;32m   1248\u001b[0m         \u001b[0;32mexcept\u001b[0m \u001b[0mBaseException\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Library/Python/2.7/lib/python/site-packages/sqlalchemy/engine/default.pyc\u001b[0m in \u001b[0;36mdo_execute\u001b[0;34m(self, cursor, statement, parameters, context)\u001b[0m\n\u001b[1;32m    579\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    580\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mdo_execute\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcursor\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparameters\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcontext\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mNone\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 581\u001b[0;31m         \u001b[0mcursor\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mexecute\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mparameters\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    582\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    583\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mdo_execute_no_params\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcursor\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mstatement\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcontext\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mNone\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;31mInternalError\u001b[0m: (psycopg2.errors.InternalError_) plpy.Error: Unable to get number of classes from model architecture. (plpython.c:5038)\nCONTEXT:  Traceback (most recent call last):\n  PL/Python function \"madlib_keras_predict_byom\", line 23, in <module>\n    madlib_keras_predict.PredictBYOM(**globals())\n  PL/Python function \"madlib_keras_predict_byom\", line 42, in wrapper\n  PL/Python function \"madlib_keras_predict_byom\", line 314, in __init__\n  PL/Python function \"madlib_keras_predict_byom\", line 326, in validate_and_set_defaults\n  PL/Python function \"madlib_keras_predict_byom\", line 207, in set_default_class_values\n  PL/Python function \"madlib_keras_predict_byom\", line 75, in get_num_classes\nPL/Python function \"madlib_keras_predict_byom\"\n\n[SQL: SELECT madlib.madlib_keras_predict_byom('model_arch_library_cifar10',  -- model arch table\n                                         1,                            -- model arch id\n                                        'cifar_10_test_data',          -- test_table\n                                        'id',                          -- id column\n                                        'x',                           -- independent var\n                                        'cifar10_predict_byom',        -- output table\n                                        'response',                    -- prediction type\n                                         FALSE,                        -- use gpus\n                                         NULL,                         -- class values\n                                         255.0                         -- normalizing const\n                                   );]\n(Background on this error at: http://sqlalche.me/e/2j85)"
+     ]
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS cifar10_predict_byom;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_predict_byom('model_arch_library_cifar10',  -- model arch table\n",
+    "                                         1,                            -- model arch id\n",
+    "                                        'cifar_10_test_data',          -- test_table\n",
+    "                                        'id',                          -- id column\n",
+    "                                        'x',                           -- independent var\n",
+    "                                        'cifar10_predict_byom',        -- output table\n",
+    "                                        'response',                    -- prediction type\n",
+    "                                         FALSE,                        -- use gpus\n",
+    "                                         NULL,                         -- class values\n",
+    "                                         255.0                         -- normalizing const\n",
+    "                                   );\n",
+    "SELECT * FROM cifar10_predict_byom ORDER BY id LIMIT 10;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Number of missclassifications:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>count</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>2551</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(2551L,)]"
+      ]
+     },
+     "execution_count": 5,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT COUNT(*) FROM cifar10_predict_byom JOIN cifar_10_test_data USING (id)\n",
+    "WHERE cifar10_predict_byom.estimated_dependent_var != cifar_10_test_data.y;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Predict accuracy. From https://keras.io/examples/cifar10_cnn/ accuracy claim is 75% on validation set after 25 epochs.  From run above test accuracy: 0.7449.  MADlib predict BYOM accuracy matches:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>test_accuracy_percent</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>74.49</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(Decimal('74.49'),)]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%%sql\n",
+    "SELECT round(count(*)*100.0/10000.0, 2) as test_accuracy_percent from\n",
+    "    (select cifar_10_test_data.y as actual, cifar10_predict_byom.estimated_dependent_var as estimated\n",
+    "     from cifar10_predict_byom inner join cifar_10_test_data\n",
+    "     on cifar_10_test_data.id=cifar10_predict_byom.id) q\n",
+    "WHERE q.actual=q.estimated;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 2
+}
diff --git a/community-artifacts/Deep-learning/MADlib-Keras-imagenet-inference-v1.ipynb b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-imagenet-inference-v1.ipynb
old mode 100644
new mode 100755
similarity index 99%
rename from community-artifacts/Deep-learning/MADlib-Keras-imagenet-inference-v1.ipynb
rename to community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-imagenet-inference-v1.ipynb
index 968ea88..733e442
--- a/community-artifacts/Deep-learning/MADlib-Keras-imagenet-inference-v1.ipynb
+++ b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-imagenet-inference-v1.ipynb
@@ -5179,7 +5179,7 @@
    "name": "python",
    "nbconvert_exporter": "python",
    "pygments_lexer": "ipython2",
-   "version": "2.7.10"
+   "version": "2.7.16"
   }
  },
  "nbformat": 4,
diff --git a/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-transfer-learning-v3.ipynb b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-transfer-learning-v3.ipynb
new file mode 100644
index 0000000..3d17972
--- /dev/null
+++ b/community-artifacts/Deep-learning/Train-single-model/MADlib-Keras-transfer-learning-v3.ipynb
@@ -0,0 +1,829 @@
+{
+ "cells": [
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "# Transfer Learning Using Keras and MADlib\n",
+    "\n",
+    "This is a transfer learning example based on https://keras.io/examples/mnist_transfer_cnn/ \n",
+    "\n",
+    "To load images into tables we use the script called <em>madlib_image_loader.py</em> located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning which uses the Python Imaging Library so supports multiple formats http://www.pythonware.com/products/pil/\n",
+    "\n",
+    "## Table of contents\n",
+    "<a href=\"#import_libraries\">1. Import libraries</a>\n",
+    "\n",
+    "<a href=\"#load_and_prepare_data\">2. Load and prepare data</a>\n",
+    "\n",
+    "<a href=\"#image_preproc\">3. Call image preprocessor</a>\n",
+    "\n",
+    "<a href=\"#define_and_load_model\">4. Define and load model architecture</a>\n",
+    "\n",
+    "<a href=\"#train\">5. Train</a>\n",
+    "\n",
+    "<a href=\"#transfer_learning\">6. Transfer learning</a>"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 1,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [],
+   "source": [
+    "%load_ext sql"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 2,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
+    "        \n",
+    "# PostgreSQL local\n",
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 3,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "1 rows affected.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/html": [
+       "<table>\n",
+       "    <tr>\n",
+       "        <th>version</th>\n",
+       "    </tr>\n",
+       "    <tr>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
+       "    </tr>\n",
+       "</table>"
+      ],
+      "text/plain": [
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
+      ]
+     },
+     "execution_count": 3,
+     "metadata": {},
+     "output_type": "execute_result"
+    }
+   ],
+   "source": [
+    "%sql select madlib.version();\n",
+    "#%sql select version();"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"import_libraries\"></a>\n",
+    "# 1.  Import libraries\n",
+    "From https://keras.io/examples/mnist_transfer_cnn/ import libraries and define some params"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 4,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "from __future__ import print_function\n",
+    "\n",
+    "import datetime\n",
+    "from tensorflow import keras\n",
+    "from tensorflow.keras.datasets import mnist\n",
+    "from tensorflow.keras.models import Sequential\n",
+    "from tensorflow.keras.layers import Dense, Dropout, Activation, Flatten\n",
+    "from tensorflow.keras.layers import Conv2D, MaxPooling2D\n",
+    "from tensorflow.keras import backend as K\n",
+    "\n",
+    "now = datetime.datetime.now\n",
+    "\n",
+    "batch_size = 128\n",
+    "num_classes = 5\n",
+    "epochs = 5\n",
+    "\n",
+    "# input image dimensions\n",
+    "img_rows, img_cols = 28, 28\n",
+    "# number of convolutional filters to use\n",
+    "filters = 32\n",
+    "# size of pooling area for max pooling\n",
+    "pool_size = 2\n",
+    "# convolution kernel size\n",
+    "kernel_size = 3\n",
+    "\n",
+    "if K.image_data_format() == 'channels_first':\n",
+    "    input_shape = (1, img_rows, img_cols)\n",
+    "else:\n",
+    "    input_shape = (img_rows, img_cols, 1)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Others needed in this workbook"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 5,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "import pandas as pd\n",
+    "import numpy as np"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"load_and_prepare_data\"></a>\n",
+    "# 2.  Load and prepare data\n",
+    "\n",
+    "First load MNIST data from Keras, consisting of 60,000 28x28 grayscale images of the 10 digits, along with a test set of 10,000 images."
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 6,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "(4861, 28, 28)\n",
+      "(4861, 28, 28, 1)\n"
+     ]
+    }
+   ],
+   "source": [
+    "# the data, split between train and test sets\n",
+    "(x_train, y_train), (x_test, y_test) = mnist.load_data()\n",
+    "\n",
+    "# create two datasets one with digits below 5 and one with 5 and above\n",
+    "x_train_lt5 = x_train[y_train < 5]\n",
+    "y_train_lt5 = y_train[y_train < 5]\n",
+    "x_test_lt5 = x_test[y_test < 5]\n",
+    "y_test_lt5 = y_test[y_test < 5]\n",
+    "\n",
+    "x_train_gte5 = x_train[y_train >= 5]\n",
+    "y_train_gte5 = y_train[y_train >= 5] - 5\n",
+    "x_test_gte5 = x_test[y_test >= 5]\n",
+    "y_test_gte5 = y_test[y_test >= 5] - 5\n",
+    "\n",
+    "# reshape to match model architecture\n",
+    "print(x_test_gte5.shape)\n",
+    "x_train_lt5=x_train_lt5.reshape(len(x_train_lt5), *input_shape)\n",
+    "x_test_lt5 = x_test_lt5.reshape(len(x_test_lt5), *input_shape)\n",
+    "x_train_gte5=x_train_gte5.reshape(len(x_train_gte5), *input_shape)\n",
+    "x_test_gte5 = x_test_gte5.reshape(len(x_test_gte5), *input_shape)\n",
+    "print(x_test_gte5.shape)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load datasets into tables using image loader scripts called <em>madlib_image_loader.py</em> located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 7,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# MADlib tools directory\n",
+    "import sys\n",
+    "import os\n",
+    "madlib_site_dir = '/Users/fmcquillan/Documents/Product/MADlib/Demos/data'\n",
+    "sys.path.append(madlib_site_dir)\n",
+    "\n",
+    "# Import image loader module\n",
+    "from madlib_image_loader import ImageLoader, DbCredentials"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 8,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Specify database credentials, for connecting to db\n",
+    "#db_creds = DbCredentials(user='gpadmin',\n",
+    "#                         host='35.239.240.26',\n",
+    "#                         port='5432',\n",
+    "#                         password='')\n",
+    "\n",
+    "db_creds = DbCredentials(user='gpadmin',\n",
+    "                         host='localhost',\n",
+    "                         port='8000',\n",
+    "                         password='')"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 9,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# Initialize ImageLoader (increase num_workers to run faster)\n",
+    "iloader = ImageLoader(num_workers=5, db_creds=db_creds)"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": 10,
+   "metadata": {},
+   "outputs": [
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "Done.\n"
+     ]
+    },
+    {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 10,
+     "metadata": {},
+     "output_type": "execute_result"
+    },
+    {
+     "name": "stdout",
+     "output_type": "stream",
+     "text": [
+      "MainProcess: Connected to madlib db.\n",
+      "Executing: CREATE TABLE train_lt5 (id SERIAL, x REAL[], y TEXT[])\n",
+      "CREATE TABLE\n",
+      "Created table train_lt5 in madlib db\n",
+      "Spawning 5 workers...\n",
+      "Initializing PoolWorker-1 [pid 84275]\n",
+      "PoolWorker-1: Created temporary directory /tmp/madlib_5TU8FybuWQ\n",
+      "Initializing PoolWorker-2 [pid 84276]\n",
+      "PoolWorker-2: Created temporary directory /tmp/madlib_LjDRu2RVLy\n",
+      "Initializing PoolWorker-3 [pid 84277]\n",
+      "PoolWorker-3: Created temporary directory /tmp/madlib_ksuUrx0mOn\n",
+      "Initializing PoolWorker-4 [pid 84278]\n",
+      "PoolWorker-4: Created temporary directory /tmp/madlib_f2SlPjS13H\n",
+      "PoolWorker-5: Created temporary directory /tmp/madlib_8GA0SlnXzj\n",
+      "Initializing PoolWorker-5 [pid 84279]\n",
+      "PoolWorker-4: Connected to madlib db.\n",
+      "PoolWorker-5: Connected to madlib db.\n",
+      "PoolWorker-2: Connected to madlib db.\n",
+      "PoolWorker-1: Connected to madlib db.\n",
+      "PoolWorker-3: Connected to madlib db.\n",
+      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_8GA0SlnXzj/train_lt50000.tmp\n",
+      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_LjDRu2RVLy/train_lt50000.tmp\n",
+      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_f2SlPjS13H/train_lt50000.tmp\n",
+      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_5TU8FybuWQ/train_lt50000.tmp\n",
+      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_ksuUrx0mOn/train_lt50000.tmp\n",
+      "PoolWorker-5: Removed temporary directory /tmp/madlib_8GA0SlnXzj\n",
+      "\n",
+      "Error in PoolWorker-5 while loading images\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=18042)\n",
+      "CONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "\n",
+      "PoolWorker-5: Can't find temporary directory... exiting.\n",
+      "PoolWorker-5: Can't find temporary directory... exiting.\n",
+      "PoolWorker-2: Removed temporary directory /tmp/madlib_LjDRu2RVLy\n",
+      "PoolWorker-1: Removed temporary directory /tmp/madlib_5TU8FybuWQ\n",
+      "\n",
+      "Error in PoolWorker-1 while loading images\n",
+      "Error in PoolWorker-2 while loading images\n",
+      "\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=18054)\n",
+      "CONTEXT:  COPY train_lt5, line 2, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=18046)\n",
+      "CONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "\n",
+      "\n",
+      "PoolWorker-1: Can't find temporary directory... exiting.\n",
+      "PoolWorker-2: Can't find temporary directory... exiting.\n",
+      "PoolWorker-4: Removed temporary directory /tmp/madlib_f2SlPjS13H\n",
+      "PoolWorker-3: Removed temporary directory /tmp/madlib_ksuUrx0mOn\n",
+      "\n",
+      "Error in PoolWorker-4 while loading images\n",
+      "Error in PoolWorker-3 while loading images\n",
+      "\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=18050)\n",
+      "CONTEXT:  COPY train_lt5, line 2, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "Traceback (most recent call last):\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 184, in _call_np_worker\n",
+      "    iloader._write_tmp_file_and_load(data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 396, in _write_tmp_file_and_load\n",
+      "    self._copy_into_db(f, data)\n",
+      "  File \"/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.py\", line 362, in _copy_into_db\n",
+      "    self.db_cur.copy_from(f, table_name, sep='|', columns=['x','y'])\n",
+      "BadCopyFileFormat: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=18058)\n",
+      "CONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+      "\n",
+      "\n",
+      "\n",
+      "PoolWorker-4: Can't find temporary directory... exiting.\n",
+      "PoolWorker-3: Can't find temporary directory... exiting.\n",
+      "PoolWorker-5: Can't find temporary directory... exiting.\n",
+      "PoolWorker-1: Can't find temporary directory... exiting.\n",
+      "PoolWorker-2: Can't find temporary directory... exiting.\n",
+      "PoolWorker-4: Can't find temporary directory... exiting.\n",
+      "PoolWorker-3: Can't find temporary directory... exiting.\n",
+      "PoolWorker-5: Can't find temporary directory... exiting.\n",
+      "PoolWorker-1: Can't find temporary directory... exiting.\n",
+      "PoolWorker-2: Can't find temporary directory... exiting.\n",
+      "PoolWorker-4: Can't find temporary directory... exiting.\n",
+      "PoolWorker-3: Can't find temporary directory... exiting.\n",
+      "PoolWorker-5: Can't find temporary directory... exiting.\n",
+      "PoolWorker-1: Can't find temporary directory... exiting.\n",
+      "PoolWorker-2: Can't find temporary directory... exiting.\n",
+      "PoolWorker-4: Can't find temporary directory... exiting.\n",
+      "PoolWorker-3: Can't find temporary directory... exiting.\n",
+      "5 workers terminated.\n"
+     ]
+    },
+    {
+     "ename": "BadCopyFileFormat",
+     "evalue": "array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=18042)\nCONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n",
+     "output_type": "error",
+     "traceback": [
+      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
+      "\u001b[0;31mBadCopyFileFormat\u001b[0m                         Traceback (most recent call last)",
+      "\u001b[0;32m<ipython-input-10-3c25ba51b8fc>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m()\u001b[0m\n\u001b[1;32m      3\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      4\u001b[0m \u001b[0;31m# Save images to temporary directories and load into database\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 5\u001b[0;31m \u001b[0miloader\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload_dataset_from_np\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx_train_lt5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_train_lt5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'train_lt5'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mappend\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mFalse\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m      6\u001b[0m \u001b[0miloader\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload_dataset_from_np\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx_test_lt5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_test_lt5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'test_lt5'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mappend\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mFalse\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      7\u001b[0m \u001b[0miloader\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload_dataset_from_np\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx_train_gte5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_train_gte5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'train_gte5'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mappend\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mFalse\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/Documents/Product/MADlib/Demos/data/madlib_image_loader.pyc\u001b[0m in \u001b[0;36mload_dataset_from_np\u001b[0;34m(self, data_x, data_y, table_name, append, label_datatype)\u001b[0m\n\u001b[1;32m    523\u001b[0m         \u001b[0;32mexcept\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mException\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    524\u001b[0m             \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mterminate_workers\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 525\u001b[0;31m             \u001b[0;32mraise\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    526\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    527\u001b[0m         \u001b[0mend_time\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtime\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtime\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;31mBadCopyFileFormat\u001b[0m: array value must start with \"{\" or dimension information  (seg0 10.128.0.41:40000 pid=18042)\nCONTEXT:  COPY train_lt5, line 1, column {{{0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}...\n"
+     ]
+    }
+   ],
+   "source": [
+    "# Drop tables\n",
+    "%sql DROP TABLE IF EXISTS train_lt5, test_lt5, train_gte5, test_gte5\n",
+    "\n",
+    "# Save images to temporary directories and load into database\n",
+    "iloader.load_dataset_from_np(x_train_lt5, y_train_lt5, 'train_lt5', append=False)\n",
+    "iloader.load_dataset_from_np(x_test_lt5, y_test_lt5, 'test_lt5', append=False)\n",
+    "iloader.load_dataset_from_np(x_train_gte5, y_train_gte5, 'train_gte5', append=False)\n",
+    "iloader.load_dataset_from_np(x_test_gte5, y_test_gte5, 'test_gte5', append=False)"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"image_preproc\"></a>\n",
+    "# 3. Call image preprocessor\n",
+    "\n",
+    "Transforms from one image per row to multiple images per row for batch optimization.  Also normalizes and one-hot encodes.\n",
+    "\n",
+    "Training dataset < 5"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {
+    "scrolled": true
+   },
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS train_lt5_packed, train_lt5_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('train_lt5',               -- Source table\n",
+    "                                       'train_lt5_packed',        -- Output table\n",
+    "                                       'y',                       -- Dependent variable\n",
+    "                                       'x',                       -- Independent variable\n",
+    "                                        1000,                     -- Buffer size\n",
+    "                                        255                       -- Normalizing constant\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM train_lt5_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Test dataset < 5"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS test_lt5_packed, test_lt5_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('test_lt5',                -- Source table\n",
+    "                                         'test_lt5_packed',         -- Output table\n",
+    "                                         'y',                       -- Dependent variable\n",
+    "                                         'x',                       -- Independent variable\n",
+    "                                         'train_lt5_packed'         -- Training preproc table\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM test_lt5_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Training dataset >= 5"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS train_gte5_packed, train_gte5_packed_summary;\n",
+    "\n",
+    "SELECT madlib.training_preprocessor_dl('train_gte5',              -- Source table\n",
+    "                                       'train_gte5_packed',       -- Output table\n",
+    "                                       'y',                       -- Dependent variable\n",
+    "                                       'x',                       -- Independent variable\n",
+    "                                        1000,                     -- Buffer size\n",
+    "                                        255                       -- Normalizing constant\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM train_gte5_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Test dataset >= 5"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS test_gte5_packed, test_gte5_packed_summary;\n",
+    "\n",
+    "SELECT madlib.validation_preprocessor_dl('test_gte5',             -- Source table\n",
+    "                                         'test_gte5_packed',      -- Output table\n",
+    "                                         'y',                     -- Dependent variable\n",
+    "                                         'x',                     -- Independent variable\n",
+    "                                         'train_gte5_packed'      -- Training preproc table\n",
+    "                                        );\n",
+    "\n",
+    "SELECT * FROM test_gte5_packed_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"define_and_load_model\"></a>\n",
+    "# 4. Define and load model architecture\n",
+    "\n",
+    "Model with feature and classification layers trainable"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# define two groups of layers: feature (convolutions) and classification (dense)\n",
+    "feature_layers = [\n",
+    "    Conv2D(filters, kernel_size,\n",
+    "           padding='valid',\n",
+    "           input_shape=input_shape),\n",
+    "    Activation('relu'),\n",
+    "    Conv2D(filters, kernel_size),\n",
+    "    Activation('relu'),\n",
+    "    MaxPooling2D(pool_size=pool_size),\n",
+    "    Dropout(0.25),\n",
+    "    Flatten(),\n",
+    "]\n",
+    "\n",
+    "classification_layers = [\n",
+    "    Dense(128),\n",
+    "    Activation('relu'),\n",
+    "    Dropout(0.5),\n",
+    "    Dense(num_classes),\n",
+    "    Activation('softmax')\n",
+    "]\n",
+    "\n",
+    "# create complete model\n",
+    "model = Sequential(feature_layers + classification_layers)\n",
+    "\n",
+    "model.summary()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into model architecture table using psycopg2"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "import psycopg2 as p2\n",
+    "#conn = p2.connect('postgresql://gpadmin@35.239.240.26:5432/madlib')\n",
+    "conn = p2.connect('postgresql://gpadmin@localhost:8000/madlib')\n",
+    "cur = conn.cursor()\n",
+    "\n",
+    "%sql DROP TABLE IF EXISTS model_arch_library;\n",
+    "query = \"SELECT madlib.load_keras_model('model_arch_library', %s, NULL, %s)\"\n",
+    "cur.execute(query,[model.to_json(), \"feature + classification layers trainable\"])\n",
+    "conn.commit()\n",
+    "\n",
+    "# check model loaded OK\n",
+    "%sql SELECT model_id, name FROM model_arch_library;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Model with feature layers frozen"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "# freeze feature layers\n",
+    "for l in feature_layers:\n",
+    "    l.trainable = False\n",
+    "\n",
+    "model.summary()"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Load into transfer model architecture table using psycopg2"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "cur.execute(query,[model.to_json(), \"only classification layers trainable\"])\n",
+    "conn.commit()\n",
+    "\n",
+    "# check model loaded OK\n",
+    "%sql SELECT model_id, name FROM model_arch_library ORDER BY model_id;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"train\"></a>\n",
+    "# 5.  Train\n",
+    "Train the model for 5-digit classification [0..4]  "
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mnist_model, mnist_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('train_lt5_packed',    -- source table\n",
+    "                               'mnist_model',         -- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                1,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']$$,  -- compile_params\n",
+    "                                $$ batch_size=128, epochs=1 $$,  -- fit_params\n",
+    "                                5                     -- num_iterations\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM mnist_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Evaluate using test data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mnist_validate;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_evaluate('mnist_model',      -- model\n",
+    "                                   'test_lt5_packed',   -- test table\n",
+    "                                   'mnist_validate'     -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM mnist_validate;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "<a id=\"transfer_learning\"></a>\n",
+    "# 6. Transfer learning"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Use UPDATE to load trained weights from previous run into the model library table:"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "UPDATE model_arch_library\n",
+    "SET model_weights = mnist_model.model_weights\n",
+    "FROM mnist_model\n",
+    "WHERE model_arch_library.model_id = 2;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Transfer: train dense layers for new classification task [5..9]"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mnist_transfer_model, mnist_transfer_model_summary;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_fit('train_gte5_packed',   -- source table\n",
+    "                               'mnist_transfer_model',-- model output table\n",
+    "                               'model_arch_library',  -- model arch table\n",
+    "                                2,                    -- model arch id\n",
+    "                                $$ loss='categorical_crossentropy', optimizer='adadelta', metrics=['accuracy']$$,  -- compile_params\n",
+    "                                $$ batch_size=128, epochs=1 $$,  -- fit_params\n",
+    "                                5                     -- num_iterations\n",
+    "                              );"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "View the model summary"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "SELECT * FROM mnist_transfer_model_summary;"
+   ]
+  },
+  {
+   "cell_type": "markdown",
+   "metadata": {},
+   "source": [
+    "Evaluate using test data"
+   ]
+  },
+  {
+   "cell_type": "code",
+   "execution_count": null,
+   "metadata": {},
+   "outputs": [],
+   "source": [
+    "%%sql\n",
+    "DROP TABLE IF EXISTS mnist_transfer_validate;\n",
+    "\n",
+    "SELECT madlib.madlib_keras_evaluate('mnist_transfer_model',      -- model\n",
+    "                                   'test_gte5_packed',           -- test table\n",
+    "                                   'mnist_transfer_validate'     -- output table\n",
+    "                                   );\n",
+    "\n",
+    "SELECT * FROM mnist_transfer_validate;"
+   ]
+  }
+ ],
+ "metadata": {
+  "kernelspec": {
+   "display_name": "Python 2",
+   "language": "python",
+   "name": "python2"
+  },
+  "language_info": {
+   "codemirror_mode": {
+    "name": "ipython",
+    "version": 2
+   },
+   "file_extension": ".py",
+   "mimetype": "text/x-python",
+   "name": "python",
+   "nbconvert_exporter": "python",
+   "pygments_lexer": "ipython2",
+   "version": "2.7.16"
+  }
+ },
+ "nbformat": 4,
+ "nbformat_minor": 1
+}
diff --git a/community-artifacts/Deep-learning/Load-images-v1.ipynb b/community-artifacts/Deep-learning/Utilities/.ipynb_checkpoints/Load-images-v1-checkpoint.ipynb
similarity index 96%
rename from community-artifacts/Deep-learning/Load-images-v1.ipynb
rename to community-artifacts/Deep-learning/Utilities/.ipynb_checkpoints/Load-images-v1-checkpoint.ipynb
index 61929e9..d158888 100644
--- a/community-artifacts/Deep-learning/Load-images-v1.ipynb
+++ b/community-artifacts/Deep-learning/Utilities/.ipynb_checkpoints/Load-images-v1-checkpoint.ipynb
@@ -25,44 +25,22 @@
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
     "%load_ext sql"
    ]
   },
   {
    "cell_type": "code",
-   "execution_count": 2,
+   "execution_count": null,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: fmcquillan@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP for deep learning (PM demo machine)\n",
-    "#%sql postgresql://gpadmin@35.239.240.26:5432/madlib\n",
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
     "# PostgreSQL local\n",
-    "%sql postgresql://fmcquillan@localhost:5432/madlib"
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
    ]
   },
   {
@@ -832,13 +810,7 @@
       "PoolWorker-12: Wrote 1000 images to /tmp/madlib_QOyefNO7bi/cifar_10_train_data_int0009.tmp\n",
       "PoolWorker-14: Loaded 1000 images into cifar_10_train_data_int\n",
       "PoolWorker-11: Loaded 1000 images into cifar_10_train_data_int\n",
-      "PoolWorker-12: Loaded 1000 images into cifar_10_train_data_int\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
+      "PoolWorker-12: Loaded 1000 images into cifar_10_train_data_int\n",
       "PoolWorker-11: Wrote 1000 images to /tmp/madlib_8Pc3rPCJJL/cifar_10_train_data_int0010.tmp\n",
       "PoolWorker-12: Wrote 1000 images to /tmp/madlib_QOyefNO7bi/cifar_10_train_data_int0010.tmp\n",
       "PoolWorker-11: Loaded 1000 images into cifar_10_train_data_int\n",
diff --git a/community-artifacts/Deep-learning/Load-images-v1.ipynb b/community-artifacts/Deep-learning/Utilities/Load-images-v1.ipynb
similarity index 72%
copy from community-artifacts/Deep-learning/Load-images-v1.ipynb
copy to community-artifacts/Deep-learning/Utilities/Load-images-v1.ipynb
index 61929e9..9d39228 100644
--- a/community-artifacts/Deep-learning/Load-images-v1.ipynb
+++ b/community-artifacts/Deep-learning/Utilities/Load-images-v1.ipynb
@@ -25,18 +25,7 @@
    "cell_type": "code",
    "execution_count": 1,
    "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/config.py:13: ShimWarning: The `IPython.config` package has been deprecated since IPython 4.0. You should import from traitlets.config instead.\n",
-      "  \"You should import from traitlets.config instead.\", ShimWarning)\n",
-      "/Users/fmcquillan/anaconda/lib/python2.7/site-packages/IPython/utils/traitlets.py:5: UserWarning: IPython.utils.traitlets has moved to a top-level traitlets package.\n",
-      "  warn(\"IPython.utils.traitlets has moved to a top-level traitlets package.\")\n"
-     ]
-    }
-   ],
+   "outputs": [],
    "source": [
     "%load_ext sql"
    ]
@@ -45,24 +34,13 @@
    "cell_type": "code",
    "execution_count": 2,
    "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: fmcquillan@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
+   "outputs": [],
    "source": [
-    "# Greenplum Database 5.x on GCP for deep learning (PM demo machine)\n",
-    "#%sql postgresql://gpadmin@35.239.240.26:5432/madlib\n",
+    "# Greenplum Database 5.x on GCP - via tunnel\n",
+    "%sql postgresql://gpadmin@localhost:8000/madlib\n",
     "        \n",
     "# PostgreSQL local\n",
-    "%sql postgresql://fmcquillan@localhost:5432/madlib"
+    "#%sql postgresql://fmcquillan@localhost:5432/madlib"
    ]
   },
   {
@@ -85,12 +63,12 @@
        "        <th>version</th>\n",
        "    </tr>\n",
        "    <tr>\n",
-       "        <td>MADlib version: 1.16, git revision: rc/1.16-rc1, cmake configuration time: Mon Jul  1 17:45:09 UTC 2019, build type: Release, build system: Darwin-16.7.0, C compiler: Clang, C++ compiler: Clang</td>\n",
+       "        <td>MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5</td>\n",
        "    </tr>\n",
        "</table>"
       ],
       "text/plain": [
-       "[(u'MADlib version: 1.16, git revision: rc/1.16-rc1, cmake configuration time: Mon Jul  1 17:45:09 UTC 2019, build type: Release, build system: Darwin-16.7.0, C compiler: Clang, C++ compiler: Clang',)]"
+       "[(u'MADlib version: 1.18.0-dev, git revision: rel/v1.17.0-89-g14a91ce, cmake configuration time: Fri Mar  5 23:08:38 UTC 2021, build type: release, build system: Linux-3.10.0-1160.11.1.el7.x86_64, C compiler: gcc 4.8.5, C++ compiler: g++ 4.8.5',)]"
       ]
      },
      "execution_count": 3,
@@ -124,13 +102,6 @@
      "text": [
       "Using TensorFlow backend.\n"
      ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Couldn't import dot_parser, loading of dot files will not be possible.\n"
-     ]
     }
    ],
    "source": [
@@ -197,187 +168,38 @@
      "name": "stdout",
      "output_type": "stream",
      "text": [
-      "Done.\n",
-      "MainProcess: Connected to madlib db.\n",
-      "Executing: CREATE TABLE cifar_10_train_data (id SERIAL, x REAL[], y TEXT)\n",
-      "CREATE TABLE\n",
-      "Created table cifar_10_train_data in madlib db\n",
-      "Spawning 5 workers...\n",
-      "Initializing PoolWorker-1 [pid 82412]\n",
-      "PoolWorker-1: Created temporary directory /tmp/madlib_Bt85aChbv0\n",
-      "Initializing PoolWorker-2 [pid 82413]\n",
-      "PoolWorker-2: Created temporary directory /tmp/madlib_cSyCSiEhHT\n",
-      "Initializing PoolWorker-3 [pid 82414]\n",
-      "PoolWorker-3: Created temporary directory /tmp/madlib_uvtHjGCU5S\n",
-      "PoolWorker-1: Connected to madlib db.\n",
-      "Initializing PoolWorker-4 [pid 82415]\n",
-      "PoolWorker-4: Created temporary directory /tmp/madlib_eJmkoDZTr8\n",
-      "PoolWorker-2: Connected to madlib db.\n",
-      "Initializing PoolWorker-5 [pid 82417]\n",
-      "PoolWorker-5: Created temporary directory /tmp/madlib_websbk05x2\n",
-      "PoolWorker-3: Connected to madlib db.\n",
-      "PoolWorker-4: Connected to madlib db.\n",
-      "PoolWorker-5: Connected to madlib db.\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0000.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0000.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_uvtHjGCU5S/cifar_10_train_data0000.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_eJmkoDZTr8/cifar_10_train_data0000.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_websbk05x2/cifar_10_train_data0000.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-3: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-4: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-5: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0001.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0001.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_uvtHjGCU5S/cifar_10_train_data0001.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_eJmkoDZTr8/cifar_10_train_data0001.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_websbk05x2/cifar_10_train_data0001.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-3: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-4: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-5: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0002.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0002.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_uvtHjGCU5S/cifar_10_train_data0002.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_eJmkoDZTr8/cifar_10_train_data0002.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_websbk05x2/cifar_10_train_data0002.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-3: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-4: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-5: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0003.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0003.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_uvtHjGCU5S/cifar_10_train_data0003.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_eJmkoDZTr8/cifar_10_train_data0003.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_websbk05x2/cifar_10_train_data0003.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-3: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-4: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-5: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0004.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0004.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_uvtHjGCU5S/cifar_10_train_data0004.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_eJmkoDZTr8/cifar_10_train_data0004.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_websbk05x2/cifar_10_train_data0004.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-3: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-4: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-5: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0005.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0005.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_uvtHjGCU5S/cifar_10_train_data0005.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_eJmkoDZTr8/cifar_10_train_data0005.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_websbk05x2/cifar_10_train_data0005.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-3: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-4: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-5: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0006.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0006.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_uvtHjGCU5S/cifar_10_train_data0006.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_eJmkoDZTr8/cifar_10_train_data0006.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_websbk05x2/cifar_10_train_data0006.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-3: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-4: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-5: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0007.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0007.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_uvtHjGCU5S/cifar_10_train_data0007.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_eJmkoDZTr8/cifar_10_train_data0007.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_websbk05x2/cifar_10_train_data0007.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-3: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-4: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-5: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0008.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0008.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_uvtHjGCU5S/cifar_10_train_data0008.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_eJmkoDZTr8/cifar_10_train_data0008.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_websbk05x2/cifar_10_train_data0008.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-3: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-4: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-5: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0009.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0009.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0010.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_cSyCSiEhHT/cifar_10_train_data0010.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_Bt85aChbv0/cifar_10_train_data0011.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar_10_train_data\n",
-      "PoolWorker-2: Removed temporary directory /tmp/madlib_cSyCSiEhHT\n",
-      "PoolWorker-3: Removed temporary directory /tmp/madlib_uvtHjGCU5S\n"
+      "Done.\n"
      ]
     },
     {
+     "data": {
+      "text/plain": [
+       "[]"
+      ]
+     },
+     "execution_count": 6,
+     "metadata": {},
+     "output_type": "execute_result"
+    },
+    {
      "name": "stdout",
      "output_type": "stream",
      "text": [
-      "PoolWorker-5: Removed temporary directory /tmp/madlib_websbk05x2\n",
-      "PoolWorker-4: Removed temporary directory /tmp/madlib_eJmkoDZTr8\n",
-      "PoolWorker-1: Removed temporary directory /tmp/madlib_Bt85aChbv0\n",
-      "Done!  Loaded 50000 images in 24.2222080231s\n",
-      "5 workers terminated.\n",
       "MainProcess: Connected to madlib db.\n",
-      "Executing: CREATE TABLE cifar_10_test_data (id SERIAL, x REAL[], y TEXT)\n",
-      "CREATE TABLE\n",
-      "Created table cifar_10_test_data in madlib db\n",
-      "Spawning 5 workers...\n",
-      "Initializing PoolWorker-6 [pid 82423]\n",
-      "PoolWorker-6: Created temporary directory /tmp/madlib_e615zVgkaE\n",
-      "Initializing PoolWorker-7 [pid 82424]\n",
-      "PoolWorker-7: Created temporary directory /tmp/madlib_iRi2oMNIFA\n",
-      "Initializing PoolWorker-8 [pid 82425]\n",
-      "PoolWorker-8: Created temporary directory /tmp/madlib_kkSktVCq3n\n",
-      "PoolWorker-6: Connected to madlib db.\n",
-      "Initializing PoolWorker-9 [pid 82426]\n",
-      "PoolWorker-7: Connected to madlib db.\n",
-      "PoolWorker-9: Created temporary directory /tmp/madlib_0To3XX96yI\n",
-      "Initializing PoolWorker-10 [pid 82428]\n",
-      "PoolWorker-8: Connected to madlib db.\n",
-      "PoolWorker-10: Created temporary directory /tmp/madlib_8zwK04IJsc\n",
-      "PoolWorker-9: Connected to madlib db.\n",
-      "PoolWorker-10: Connected to madlib db.\n",
-      "PoolWorker-6: Wrote 1000 images to /tmp/madlib_e615zVgkaE/cifar_10_test_data0000.tmp\n",
-      "PoolWorker-7: Wrote 1000 images to /tmp/madlib_iRi2oMNIFA/cifar_10_test_data0000.tmp\n",
-      "PoolWorker-8: Wrote 1000 images to /tmp/madlib_kkSktVCq3n/cifar_10_test_data0000.tmp\n",
-      "PoolWorker-9: Wrote 1000 images to /tmp/madlib_0To3XX96yI/cifar_10_test_data0000.tmp\n",
-      "PoolWorker-10: Wrote 1000 images to /tmp/madlib_8zwK04IJsc/cifar_10_test_data0000.tmp\n",
-      "PoolWorker-6: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-7: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-8: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-10: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-9: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-6: Wrote 1000 images to /tmp/madlib_e615zVgkaE/cifar_10_test_data0001.tmp\n",
-      "PoolWorker-7: Wrote 1000 images to /tmp/madlib_iRi2oMNIFA/cifar_10_test_data0001.tmp\n",
-      "PoolWorker-8: Wrote 1000 images to /tmp/madlib_kkSktVCq3n/cifar_10_test_data0001.tmp\n",
-      "PoolWorker-9: Wrote 1000 images to /tmp/madlib_0To3XX96yI/cifar_10_test_data0001.tmp\n",
-      "PoolWorker-10: Wrote 1000 images to /tmp/madlib_8zwK04IJsc/cifar_10_test_data0001.tmp\n",
-      "PoolWorker-6: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-7: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-8: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-9: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-10: Loaded 1000 images into cifar_10_test_data\n",
-      "PoolWorker-10: Removed temporary directory /tmp/madlib_8zwK04IJsc\n",
-      "PoolWorker-8: Removed temporary directory /tmp/madlib_kkSktVCq3n\n",
-      "PoolWorker-7: Removed temporary directory /tmp/madlib_iRi2oMNIFA\n",
-      "PoolWorker-6: Removed temporary directory /tmp/madlib_e615zVgkaE\n",
-      "PoolWorker-9: Removed temporary directory /tmp/madlib_0To3XX96yI\n",
-      "Done!  Loaded 10000 images in 4.6932258606s\n",
-      "5 workers terminated.\n"
+      "Executing: CREATE TABLE cifar_10_train_data (id SERIAL, x REAL[], y TEXT)\n"
+     ]
+    },
+    {
+     "ename": "RuntimeError",
+     "evalue": "relation \"cifar_10_train_data\" already exists while creating cifar_10_train_data in db madlib.\nIf the table already exists, you can use append=True to append more images to it.",
+     "output_type": "error",
+     "traceback": [
+      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
+      "\u001b[0;31mRuntimeError\u001b[0m                              Traceback (most recent call last)",
+      "\u001b[0;32m<ipython-input-6-046119109a65>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m()\u001b[0m\n\u001b[1;32m      2\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      3\u001b[0m \u001b[0;31m# Save images to temporary directories and load into database\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 4\u001b[0;31m \u001b[0miloader\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload_dataset_from_np\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx_train\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_train\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'cifar_10_train_data'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mappend\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mFalse\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m      5\u001b[0m \u001b[0miloader\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload_dataset_from_np\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx_test\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_test\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'cifar_10_test_data'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mappend\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mFalse\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/workspace/madlib-site/community-artifacts/Deep-learning/utilities/madlib_image_loader.pyc\u001b[0m in \u001b[0;36mload_dataset_from_np\u001b[0;34m(self, data_x, data_y, table_name, append, label_datatype)\u001b[0m\n\u001b[1;32m    485\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    486\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 487\u001b[0;31m         \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_validate_input_and_create_table\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mdata_x\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdata_y\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    488\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    489\u001b[0m         \u001b[0mdata\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mzip\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mdata_x\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdata_y\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
+      "\u001b[0;32m/Users/fmcquillan/workspace/madlib-site/community-artifacts/Deep-learning/utilities/madlib_image_loader.pyc\u001b[0m in \u001b[0;36m_validate_input_and_create_table\u001b[0;34m(self, data_x, data_y)\u001b[0m\n\u001b[1;32m    438\u001b[0m                                    \u001b[0;34m\"append=True to append more images to it.\"\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    439\u001b[0m                                 .format(e.message.strip(), self.table_name,\n\u001b[0;32m--> 440\u001b[0;31m                                         self.db_creds.db_name))\n\u001b[0m\u001b[1;32m    441\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    442\u001b[0m             print \"Created table {0} in {1} db\".format(self.table_name,\n",
+      "\u001b[0;31mRuntimeError\u001b[0m: relation \"cifar_10_train_data\" already exists while creating cifar_10_train_data in db madlib.\nIf the table already exists, you can use append=True to append more images to it."
      ]
     }
    ],
@@ -832,13 +654,7 @@
       "PoolWorker-12: Wrote 1000 images to /tmp/madlib_QOyefNO7bi/cifar_10_train_data_int0009.tmp\n",
       "PoolWorker-14: Loaded 1000 images into cifar_10_train_data_int\n",
       "PoolWorker-11: Loaded 1000 images into cifar_10_train_data_int\n",
-      "PoolWorker-12: Loaded 1000 images into cifar_10_train_data_int\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
+      "PoolWorker-12: Loaded 1000 images into cifar_10_train_data_int\n",
       "PoolWorker-11: Wrote 1000 images to /tmp/madlib_8Pc3rPCJJL/cifar_10_train_data_int0010.tmp\n",
       "PoolWorker-12: Wrote 1000 images to /tmp/madlib_QOyefNO7bi/cifar_10_train_data_int0010.tmp\n",
       "PoolWorker-11: Loaded 1000 images into cifar_10_train_data_int\n",
diff --git a/community-artifacts/Deep-learning/madlib_image_loader.py b/community-artifacts/Deep-learning/Utilities/madlib_image_loader.py
similarity index 100%
rename from community-artifacts/Deep-learning/madlib_image_loader.py
rename to community-artifacts/Deep-learning/Utilities/madlib_image_loader.py
diff --git a/community-artifacts/Deep-learning/automl/hyperband-diag-cifar10-v1.ipynb b/community-artifacts/Deep-learning/automl/hyperband-diag-cifar10-v1.ipynb
deleted file mode 100644
index 05a7143..0000000
--- a/community-artifacts/Deep-learning/automl/hyperband-diag-cifar10-v1.ipynb
+++ /dev/null
@@ -1,5288 +0,0 @@
-{
- "cells": [
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "# Hyperband diagonal using CIFAR-10\n",
-    "\n",
-    "Implemention of Hyperband https://arxiv.org/pdf/1603.06560.pdf for MPP with a synchronous barrier. Uses the Hyperband schedule but runs it on a diagonal across brackets, instead of one bracket at a time, to be more efficient with cluster resources.\n",
-    "\n",
-    "The CIFAR-10 dataset consists of 60,000 32x32 colour images in 10 classes, with 6,000 images per class. There are 50,000 training images and 10,000 test images.\n",
-    "https://www.cs.toronto.edu/~kriz/cifar.html\n",
-    "\n",
-    "\n",
-    "## Table of contents \n",
-    "\n",
-    "<a href=\"#setup\">0. Setup</a>\n",
-    "\n",
-    "<a href=\"#load_dataset\">1. Load dataset into table</a>\n",
-    "\n",
-    "<a href=\"#distr\">2. Setup distribution rules and call preprocessor</a>\n",
-    "\n",
-    "<a href=\"#arch\">3. Define and load model architectures</a>\n",
-    "\n",
-    "<a href=\"#hyperband\">4. Hyperband diagonal</a>\n",
-    "\n",
-    "<a href=\"#plot\">5. Plot results</a>\n",
-    "\n",
-    "<a href=\"#print\">6. Pretty print schedules</a>\n",
-    "\n",
-    "<a href=\"#predict\">7. Inference</a>"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"setup\"></a>\n",
-    "# 0. Setup"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 1,
-   "metadata": {
-    "scrolled": false
-   },
-   "outputs": [],
-   "source": [
-    "%load_ext sql"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 2,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "u'Connected: fmcquillan@madlib'"
-      ]
-     },
-     "execution_count": 2,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "# Greenplum Database 5.x on GCP - via tunnel\n",
-    "#%sql postgresql://gpadmin@localhost:8000/madlib\n",
-    "#%sql postgresql://gpadmin@35.230.53.21:5432/cifar_demo\n",
-    "\n",
-    "# PostgreSQL local\n",
-    "%sql postgresql://fmcquillan@localhost:5432/madlib"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 3,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      " * postgresql://fmcquillan@localhost:5432/madlib\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>version</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>MADlib version: 1.16, git revision: rc/1.16-rc1, cmake configuration time: Mon Jul  1 17:45:09 UTC 2019, build type: Release, build system: Darwin-16.7.0, C compiler: Clang, C++ compiler: Clang</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'MADlib version: 1.16, git revision: rc/1.16-rc1, cmake configuration time: Mon Jul  1 17:45:09 UTC 2019, build type: Release, build system: Darwin-16.7.0, C compiler: Clang, C++ compiler: Clang',)]"
-      ]
-     },
-     "execution_count": 3,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql select madlib.version();\n",
-    "#%sql select version();"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Import libraries and define some params"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 4,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stderr",
-     "output_type": "stream",
-     "text": [
-      "Using TensorFlow backend.\n"
-     ]
-    }
-   ],
-   "source": [
-    "from __future__ import print_function\n",
-    "import keras\n",
-    "from keras.datasets import cifar10\n",
-    "from keras.preprocessing.image import ImageDataGenerator\n",
-    "from keras.models import Sequential\n",
-    "from keras.layers import Dense, Dropout, Activation, Flatten, BatchNormalization\n",
-    "from keras.layers import Conv2D, MaxPooling2D\n",
-    "import os"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Others needed in this workbook"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 5,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "import pandas as pd\n",
-    "import numpy as np\n",
-    "import sys\n",
-    "import os\n",
-    "from matplotlib import pyplot as plt"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"load_dataset\"></a>\n",
-    "# 1.  Load dataset into table"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "PXF can be used to load image data.  \n",
-    "\n",
-    "For this demo, we will get the dataset from Keras and use the script called madlib_image_loader.py located at https://github.com/apache/madlib-site/tree/asf-site/community-artifacts/Deep-learning .\n",
-    "\n",
-    "If the script is not in the same folder as the notebook, you can use the following lines to import it."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "import sys\n",
-    "sys.path.insert(1, '/Users/fmcquillan/workspace/madlib-site/community-artifacts/Deep-learning')\n",
-    "from madlib_image_loader import ImageLoader, DbCredentials\n",
-    "\n",
-    "# Specify database credentials, for connecting to db\n",
-    "db_creds = DbCredentials(user='gpadmin',\n",
-    "                         host='localhost',\n",
-    "                         port='8000',\n",
-    "                         password='')\n",
-    "\n",
-    "#db_creds = DbCredentials(user='fmcquillan',\n",
-    "#                         host='localhost',\n",
-    "#                         port='5432',\n",
-    "#                         password='')\n",
-    "\n",
-    "# Initialize ImageLoader (increase num_workers to run faster)\n",
-    "iloader = ImageLoader(num_workers=5, db_creds=db_creds)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Load the training and test data"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 7,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      " * postgresql://fmcquillan@localhost:5432/madlib\n",
-      "Done.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "[]"
-      ]
-     },
-     "execution_count": 7,
-     "metadata": {},
-     "output_type": "execute_result"
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "MainProcess: Connected to madlib db.\n",
-      "Executing: CREATE TABLE cifar10_train (id SERIAL, x REAL[], y TEXT)\n",
-      "CREATE TABLE\n",
-      "Created table cifar10_train in madlib db\n",
-      "Spawning 5 workers...\n",
-      "Initializing PoolWorker-1 [pid 10828]\n",
-      "Initializing PoolWorker-2 [pid 10829]\n",
-      "PoolWorker-1: Created temporary directory /tmp/madlib_DaP40IOgzi\n",
-      "Initializing PoolWorker-3 [pid 10830]\n",
-      "PoolWorker-2: Created temporary directory /tmp/madlib_n5XjJvXs5s\n",
-      "PoolWorker-3: Created temporary directory /tmp/madlib_99mTsCxOFF\n",
-      "Initializing PoolWorker-4 [pid 10831]\n",
-      "PoolWorker-5: Connected to madlib db.\n",
-      "PoolWorker-4: Created temporary directory /tmp/madlib_zGujxaoQIb\n",
-      "Initializing PoolWorker-5 [pid 10832]\n",
-      "PoolWorker-1: Connected to madlib db.\n",
-      "PoolWorker-5: Created temporary directory /tmp/madlib_D6q8olnown\n",
-      "PoolWorker-2: Connected to madlib db.\n",
-      "PoolWorker-3: Connected to madlib db.\n",
-      "PoolWorker-4: Connected to madlib db.\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0000.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0000.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_n5XjJvXs5s/cifar10_train0000.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_zGujxaoQIb/cifar10_train0000.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_D6q8olnown/cifar10_train0000.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-2: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-5: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0001.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0001.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_n5XjJvXs5s/cifar10_train0001.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_zGujxaoQIb/cifar10_train0001.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_D6q8olnown/cifar10_train0001.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-2: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-5: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0002.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0002.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_n5XjJvXs5s/cifar10_train0002.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_zGujxaoQIb/cifar10_train0002.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_D6q8olnown/cifar10_train0002.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-2: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-5: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0003.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0003.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_n5XjJvXs5s/cifar10_train0003.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_zGujxaoQIb/cifar10_train0003.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_D6q8olnown/cifar10_train0003.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-2: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-5: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0004.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0004.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_n5XjJvXs5s/cifar10_train0004.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_zGujxaoQIb/cifar10_train0004.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_D6q8olnown/cifar10_train0004.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-2: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-5: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0005.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0005.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_n5XjJvXs5s/cifar10_train0005.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_zGujxaoQIb/cifar10_train0005.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_D6q8olnown/cifar10_train0005.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-2: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-5: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0006.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0006.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_n5XjJvXs5s/cifar10_train0006.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_zGujxaoQIb/cifar10_train0006.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_D6q8olnown/cifar10_train0006.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-2: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-5: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0007.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0007.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_n5XjJvXs5s/cifar10_train0007.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_zGujxaoQIb/cifar10_train0007.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_D6q8olnown/cifar10_train0007.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-2: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-5: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0008.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0008.tmp\n",
-      "PoolWorker-2: Wrote 1000 images to /tmp/madlib_n5XjJvXs5s/cifar10_train0008.tmp\n",
-      "PoolWorker-4: Wrote 1000 images to /tmp/madlib_zGujxaoQIb/cifar10_train0008.tmp\n",
-      "PoolWorker-5: Wrote 1000 images to /tmp/madlib_D6q8olnown/cifar10_train0008.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-2: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-5: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0009.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0009.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0010.tmp\n",
-      "PoolWorker-3: Wrote 1000 images to /tmp/madlib_99mTsCxOFF/cifar10_train0010.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-3: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-1: Wrote 1000 images to /tmp/madlib_DaP40IOgzi/cifar10_train0011.tmp\n",
-      "PoolWorker-1: Loaded 1000 images into cifar10_train\n",
-      "PoolWorker-4: Removed temporary directory /tmp/madlib_zGujxaoQIb\n",
-      "PoolWorker-5: Removed temporary directory /tmp/madlib_D6q8olnown\n",
-      "PoolWorker-2: Removed temporary directory /tmp/madlib_n5XjJvXs5s\n",
-      "PoolWorker-1: Removed temporary directory /tmp/madlib_DaP40IOgzi\n",
-      "PoolWorker-3: Removed temporary directory /tmp/madlib_99mTsCxOFF\n",
-      "Done!  Loaded 50000 images in 19.7727279663s\n",
-      "5 workers terminated.\n",
-      "MainProcess: Connected to madlib db.\n",
-      "Executing: CREATE TABLE cifar10_val (id SERIAL, x REAL[], y TEXT)\n",
-      "CREATE TABLE\n",
-      "Created table cifar10_val in madlib db\n",
-      "Spawning 5 workers...\n",
-      "Initializing PoolWorker-6 [pid 10850]\n",
-      "PoolWorker-6: Created temporary directory /tmp/madlib_OqFarH4eVS\n",
-      "Initializing PoolWorker-7 [pid 10851]\n"
-     ]
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "PoolWorker-7: Created temporary directory /tmp/madlib_BHhah9z53T\n",
-      "Initializing PoolWorker-8 [pid 10852]\n",
-      "PoolWorker-8: Created temporary directory /tmp/madlib_G5oLCmXwQN\n",
-      "Initializing PoolWorker-9 [pid 10853]\n",
-      "PoolWorker-6: Connected to madlib db.\n",
-      "PoolWorker-9: Created temporary directory /tmp/madlib_THDiiymnsM\n",
-      "Initializing PoolWorker-10 [pid 10854]\n",
-      "PoolWorker-7: Connected to madlib db.\n",
-      "PoolWorker-10: Created temporary directory /tmp/madlib_DLO1TEiyo6\n",
-      "PoolWorker-8: Connected to madlib db.\n",
-      "PoolWorker-9: Connected to madlib db.\n",
-      "PoolWorker-10: Connected to madlib db.\n",
-      "PoolWorker-6: Wrote 1000 images to /tmp/madlib_OqFarH4eVS/cifar10_val0000.tmp\n",
-      "PoolWorker-7: Wrote 1000 images to /tmp/madlib_BHhah9z53T/cifar10_val0000.tmp\n",
-      "PoolWorker-8: Wrote 1000 images to /tmp/madlib_G5oLCmXwQN/cifar10_val0000.tmp\n",
-      "PoolWorker-9: Wrote 1000 images to /tmp/madlib_THDiiymnsM/cifar10_val0000.tmp\n",
-      "PoolWorker-10: Wrote 1000 images to /tmp/madlib_DLO1TEiyo6/cifar10_val0000.tmp\n",
-      "PoolWorker-6: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-7: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-8: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-9: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-10: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-6: Wrote 1000 images to /tmp/madlib_OqFarH4eVS/cifar10_val0001.tmp\n",
-      "PoolWorker-7: Wrote 1000 images to /tmp/madlib_BHhah9z53T/cifar10_val0001.tmp\n",
-      "PoolWorker-8: Wrote 1000 images to /tmp/madlib_G5oLCmXwQN/cifar10_val0001.tmp\n",
-      "PoolWorker-9: Wrote 1000 images to /tmp/madlib_THDiiymnsM/cifar10_val0001.tmp\n",
-      "PoolWorker-10: Wrote 1000 images to /tmp/madlib_DLO1TEiyo6/cifar10_val0001.tmp\n",
-      "PoolWorker-6: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-7: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-8: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-9: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-10: Loaded 1000 images into cifar10_val\n",
-      "PoolWorker-8: Removed temporary directory /tmp/madlib_G5oLCmXwQN\n",
-      "PoolWorker-7: Removed temporary directory /tmp/madlib_BHhah9z53T\n",
-      "PoolWorker-10: Removed temporary directory /tmp/madlib_DLO1TEiyo6\n",
-      "PoolWorker-6: Removed temporary directory /tmp/madlib_OqFarH4eVS\n",
-      "PoolWorker-9: Removed temporary directory /tmp/madlib_THDiiymnsM\n",
-      "Done!  Loaded 10000 images in 4.03977298737s\n",
-      "5 workers terminated.\n"
-     ]
-    }
-   ],
-   "source": [
-    "# Load dataset into np array\n",
-    "(x_train, y_train), (x_test, y_test) = cifar10.load_data()\n",
-    "\n",
-    "%sql DROP TABLE IF EXISTS cifar10_train, cifar10_val;\n",
-    "\n",
-    "# Save images to temporary directories and load into database\n",
-    "iloader.load_dataset_from_np(x_train, y_train, 'cifar10_train', append=False)\n",
-    "iloader.load_dataset_from_np(x_test, y_test, 'cifar10_val', append=False)"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 15,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      " * postgresql://gpadmin@localhost:8000/madlib\n",
-      "(psycopg2.errors.UndefinedTable) relation \"cifar_10_train_data\" does not exist\n",
-      "LINE 1: SELECT COUNT(*) FROM cifar_10_train_data;\n",
-      "                             ^\n",
-      "\n",
-      "[SQL: SELECT COUNT(*) FROM cifar_10_train_data;]\n",
-      "(Background on this error at: http://sqlalche.me/e/f405)\n"
-     ]
-    }
-   ],
-   "source": [
-    "%sql SELECT COUNT(*) FROM cifar10_train;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 6,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>10000</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(10000L,)]"
-      ]
-     },
-     "execution_count": 6,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql SELECT COUNT(*) FROM cifar10_val;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"distr\"></a>\n",
-    "# 2.  Setup distribution rules and call preprocessor\n",
-    "\n",
-    "Get cluster configuration\n"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 8,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "20 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>hostname</th>\n",
-       "        <th>gpu_descr</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix1</td>\n",
-       "        <td>device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix1</td>\n",
-       "        <td>device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix1</td>\n",
-       "        <td>device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix1</td>\n",
-       "        <td>device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix2</td>\n",
-       "        <td>device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix2</td>\n",
-       "        <td>device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix2</td>\n",
-       "        <td>device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix2</td>\n",
-       "        <td>device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix3</td>\n",
-       "        <td>device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix3</td>\n",
-       "        <td>device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix3</td>\n",
-       "        <td>device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix3</td>\n",
-       "        <td>device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix4</td>\n",
-       "        <td>device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix4</td>\n",
-       "        <td>device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix4</td>\n",
-       "        <td>device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>phoenix4</td>\n",
-       "        <td>device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'phoenix0', u'device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0'),\n",
-       " (u'phoenix0', u'device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0'),\n",
-       " (u'phoenix0', u'device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0'),\n",
-       " (u'phoenix0', u'device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0'),\n",
-       " (u'phoenix1', u'device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0'),\n",
-       " (u'phoenix1', u'device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0'),\n",
-       " (u'phoenix1', u'device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0'),\n",
-       " (u'phoenix1', u'device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0'),\n",
-       " (u'phoenix2', u'device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0'),\n",
-       " (u'phoenix2', u'device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0'),\n",
-       " (u'phoenix2', u'device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0'),\n",
-       " (u'phoenix2', u'device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0'),\n",
-       " (u'phoenix3', u'device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0'),\n",
-       " (u'phoenix3', u'device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0'),\n",
-       " (u'phoenix3', u'device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0'),\n",
-       " (u'phoenix3', u'device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0'),\n",
-       " (u'phoenix4', u'device: 0, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:04.0, compute capability: 6.0'),\n",
-       " (u'phoenix4', u'device: 1, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:05.0, compute capability: 6.0'),\n",
-       " (u'phoenix4', u'device: 2, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:06.0, compute capability: 6.0'),\n",
-       " (u'phoenix4', u'device: 3, name: Tesla P100-PCIE-16GB, pci bus id: 0000:00:07.0, compute capability: 6.0')]"
-      ]
-     },
-     "execution_count": 8,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS host_gpu_mapping_tf;\n",
-    "SELECT * FROM madlib.gpu_configuration('host_gpu_mapping_tf');\n",
-    "SELECT * FROM host_gpu_mapping_tf ORDER BY hostname, gpu_descr;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Below are examples of setting up different distribution rules tables.  You can customize this to your needs.\n",
-    "\n",
-    "Build distribution rules table for 4 VMs"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS segments_to_use_4VMs;\n",
-    "CREATE TABLE segments_to_use_4VMs AS\n",
-    "  SELECT DISTINCT dbid, hostname FROM gp_segment_configuration JOIN host_gpu_mapping_tf USING (hostname)\n",
-    "  WHERE role='p' AND content>=0 AND hostname!='phoenix4';\n",
-    "SELECT * FROM segments_to_use_4VMs ORDER BY hostname, dbid;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Build distribution rules table for 2 VMs"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS segments_to_use_2VMs;\n",
-    "CREATE TABLE segments_to_use_2VMs AS\n",
-    "  SELECT DISTINCT dbid, hostname FROM gp_segment_configuration JOIN host_gpu_mapping_tf USING (hostname)\n",
-    "  WHERE role='p' AND content>=0 AND (hostname='phoenix0' OR hostname='phoenix1');\n",
-    "SELECT * FROM segments_to_use_2VMs ORDER BY hostname, dbid;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Build distribution rules table for 1 VMs"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS segments_to_use_1VM;\n",
-    "CREATE TABLE segments_to_use_1VM AS\n",
-    "  SELECT DISTINCT dbid, hostname FROM gp_segment_configuration JOIN host_gpu_mapping_tf USING (hostname)\n",
-    "  WHERE role='p' AND content>=0 AND hostname='phoenix0';\n",
-    "SELECT * FROM segments_to_use_1VM ORDER BY hostname, dbid;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Build distribution rules table for 1 segment"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 26,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "5 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>dbid</th>\n",
-       "        <th>content</th>\n",
-       "        <th>role</th>\n",
-       "        <th>preferred_role</th>\n",
-       "        <th>mode</th>\n",
-       "        <th>status</th>\n",
-       "        <th>port</th>\n",
-       "        <th>hostname</th>\n",
-       "        <th>address</th>\n",
-       "        <th>replication_port</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>-1</td>\n",
-       "        <td>p</td>\n",
-       "        <td>p</td>\n",
-       "        <td>s</td>\n",
-       "        <td>u</td>\n",
-       "        <td>5432</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>None</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>0</td>\n",
-       "        <td>p</td>\n",
-       "        <td>p</td>\n",
-       "        <td>c</td>\n",
-       "        <td>u</td>\n",
-       "        <td>40000</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>70000</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>1</td>\n",
-       "        <td>p</td>\n",
-       "        <td>p</td>\n",
-       "        <td>c</td>\n",
-       "        <td>u</td>\n",
-       "        <td>40001</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>70001</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>2</td>\n",
-       "        <td>p</td>\n",
-       "        <td>p</td>\n",
-       "        <td>c</td>\n",
-       "        <td>u</td>\n",
-       "        <td>40002</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>70002</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>3</td>\n",
-       "        <td>p</td>\n",
-       "        <td>p</td>\n",
-       "        <td>c</td>\n",
-       "        <td>u</td>\n",
-       "        <td>40003</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>phoenix0</td>\n",
-       "        <td>70003</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, -1, u'p', u'p', u's', u'u', 5432, u'phoenix0', u'phoenix0', None),\n",
-       " (2, 0, u'p', u'p', u'c', u'u', 40000, u'phoenix0', u'phoenix0', 70000),\n",
-       " (3, 1, u'p', u'p', u'c', u'u', 40001, u'phoenix0', u'phoenix0', 70001),\n",
-       " (4, 2, u'p', u'p', u'c', u'u', 40002, u'phoenix0', u'phoenix0', 70002),\n",
-       " (5, 3, u'p', u'p', u'c', u'u', 40003, u'phoenix0', u'phoenix0', 70003)]"
-      ]
-     },
-     "execution_count": 26,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM gp_segment_configuration WHERE role='p' AND hostname='phoenix0' ORDER BY dbid;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 27,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>dbid</th>\n",
-       "        <th>hostname</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>phoenix0</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(2, u'phoenix0')]"
-      ]
-     },
-     "execution_count": 27,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS segments_to_use_1seg;\n",
-    "CREATE TABLE segments_to_use_1seg AS\n",
-    "  SELECT DISTINCT dbid, hostname FROM gp_segment_configuration JOIN host_gpu_mapping_tf USING (hostname)\n",
-    "  WHERE dbid=2;\n",
-    "SELECT * FROM segments_to_use_1seg ORDER BY hostname, dbid;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Training dataset (uses training preprocessor):"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 7,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "16 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
-       "        <th>buffer_id</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>2</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>3</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>4</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>5</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>6</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>7</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>8</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>9</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>10</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>11</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>12</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>13</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>14</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[3125, 32, 32, 3]</td>\n",
-       "        <td>[3125, 10]</td>\n",
-       "        <td>15</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[([3125, 32, 32, 3], [3125, 10], 0),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 1),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 2),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 3),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 4),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 5),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 6),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 7),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 8),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 9),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 10),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 11),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 12),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 13),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 14),\n",
-       " ([3125, 32, 32, 3], [3125, 10], 15)]"
-      ]
-     },
-     "execution_count": 7,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS cifar10_train_packed, cifar10_train_packed_summary;\n",
-    "\n",
-    "SELECT madlib.training_preprocessor_dl('cifar10_train',        -- Source table\n",
-    "                                       'cifar10_train_packed', -- Output table\n",
-    "                                       'y',                    -- Dependent variable\n",
-    "                                       'x',                    -- Independent variable\n",
-    "                                        NULL,                  -- Buffer size\n",
-    "                                        256.0,                 -- Normalizing constant\n",
-    "                                        NULL,                  -- Number of classes\n",
-    "                                       'gpu_segments'          -- Distribution rules\n",
-    "                                        );\n",
-    "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM cifar10_train_packed ORDER BY buffer_id;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 8,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>cifar10_train</td>\n",
-       "        <td>cifar10_train_packed</td>\n",
-       "        <td>y</td>\n",
-       "        <td>x</td>\n",
-       "        <td>smallint</td>\n",
-       "        <td>[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]</td>\n",
-       "        <td>3125</td>\n",
-       "        <td>256.0</td>\n",
-       "        <td>10</td>\n",
-       "        <td>[2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17]</td>\n",
-       "        <td>[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'cifar10_train', u'cifar10_train_packed', u'y', u'x', u'smallint', [0, 1, 2, 3, 4, 5, 6, 7, 8, 9], 3125, 256.0, 10, [2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17], [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15])]"
-      ]
-     },
-     "execution_count": 8,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM cifar10_train_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Validation dataset (uses validation preprocessor):"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 9,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "16 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>independent_var_shape</th>\n",
-       "        <th>dependent_var_shape</th>\n",
-       "        <th>buffer_id</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>1</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>2</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>3</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>4</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>5</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>6</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>7</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>8</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>9</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>10</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>11</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>12</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>13</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>14</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[625, 32, 32, 3]</td>\n",
-       "        <td>[625, 10]</td>\n",
-       "        <td>15</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[([625, 32, 32, 3], [625, 10], 0),\n",
-       " ([625, 32, 32, 3], [625, 10], 1),\n",
-       " ([625, 32, 32, 3], [625, 10], 2),\n",
-       " ([625, 32, 32, 3], [625, 10], 3),\n",
-       " ([625, 32, 32, 3], [625, 10], 4),\n",
-       " ([625, 32, 32, 3], [625, 10], 5),\n",
-       " ([625, 32, 32, 3], [625, 10], 6),\n",
-       " ([625, 32, 32, 3], [625, 10], 7),\n",
-       " ([625, 32, 32, 3], [625, 10], 8),\n",
-       " ([625, 32, 32, 3], [625, 10], 9),\n",
-       " ([625, 32, 32, 3], [625, 10], 10),\n",
-       " ([625, 32, 32, 3], [625, 10], 11),\n",
-       " ([625, 32, 32, 3], [625, 10], 12),\n",
-       " ([625, 32, 32, 3], [625, 10], 13),\n",
-       " ([625, 32, 32, 3], [625, 10], 14),\n",
-       " ([625, 32, 32, 3], [625, 10], 15)]"
-      ]
-     },
-     "execution_count": 9,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS cifar10_val_packed, cifar10_val_packed_summary;\n",
-    "\n",
-    "SELECT madlib.validation_preprocessor_dl('cifar10_val',          -- Source table\n",
-    "                                         'cifar10_val_packed',   -- Output table\n",
-    "                                         'y',                    -- Dependent variable\n",
-    "                                         'x',                    -- Independent variable\n",
-    "                                         'cifar10_train_packed', -- From training preprocessor step\n",
-    "                                         NULL,                   -- Buffer size\n",
-    "                                         'gpu_segments'          -- Distribution rules\n",
-    "                                          ); \n",
-    "\n",
-    "SELECT independent_var_shape, dependent_var_shape, buffer_id FROM cifar10_val_packed ORDER BY buffer_id;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 10,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>source_table</th>\n",
-       "        <th>output_table</th>\n",
-       "        <th>dependent_varname</th>\n",
-       "        <th>independent_varname</th>\n",
-       "        <th>dependent_vartype</th>\n",
-       "        <th>class_values</th>\n",
-       "        <th>buffer_size</th>\n",
-       "        <th>normalizing_const</th>\n",
-       "        <th>num_classes</th>\n",
-       "        <th>distribution_rules</th>\n",
-       "        <th>__internal_gpu_config__</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>cifar10_val</td>\n",
-       "        <td>cifar10_val_packed</td>\n",
-       "        <td>y</td>\n",
-       "        <td>x</td>\n",
-       "        <td>smallint</td>\n",
-       "        <td>[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]</td>\n",
-       "        <td>625</td>\n",
-       "        <td>256.0</td>\n",
-       "        <td>10</td>\n",
-       "        <td>[2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17]</td>\n",
-       "        <td>[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(u'cifar10_val', u'cifar10_val_packed', u'y', u'x', u'smallint', [0, 1, 2, 3, 4, 5, 6, 7, 8, 9], 625, 256.0, 10, [2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17], [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15])]"
-      ]
-     },
-     "execution_count": 10,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT * FROM cifar10_val_packed_summary;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"arch\"></a>\n",
-    "# 3. Define and load model architectures\n",
-    "\n",
-    "Here we load some example model architectures from published sources.\n",
-    "\n",
-    "a. Model architecture from https://keras.io/examples/cifar10_cnn/"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 6,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "num_classes = 10\n",
-    "\n",
-    "#to be removed\n",
-    "#do this just to get shape for model architecture \n",
-    "(x_train, y_train), (x_test, y_test) = cifar10.load_data()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 7,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "conv2d_1 (Conv2D)            (None, 32, 32, 32)        896       \n",
-      "_________________________________________________________________\n",
-      "activation_1 (Activation)    (None, 32, 32, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "conv2d_2 (Conv2D)            (None, 30, 30, 32)        9248      \n",
-      "_________________________________________________________________\n",
-      "activation_2 (Activation)    (None, 30, 30, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_1 (MaxPooling2 (None, 15, 15, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "dropout_1 (Dropout)          (None, 15, 15, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "conv2d_3 (Conv2D)            (None, 15, 15, 64)        18496     \n",
-      "_________________________________________________________________\n",
-      "activation_3 (Activation)    (None, 15, 15, 64)        0         \n",
-      "_________________________________________________________________\n",
-      "conv2d_4 (Conv2D)            (None, 13, 13, 64)        36928     \n",
-      "_________________________________________________________________\n",
-      "activation_4 (Activation)    (None, 13, 13, 64)        0         \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_2 (MaxPooling2 (None, 6, 6, 64)          0         \n",
-      "_________________________________________________________________\n",
-      "dropout_2 (Dropout)          (None, 6, 6, 64)          0         \n",
-      "_________________________________________________________________\n",
-      "flatten_1 (Flatten)          (None, 2304)              0         \n",
-      "_________________________________________________________________\n",
-      "dense_1 (Dense)              (None, 512)               1180160   \n",
-      "_________________________________________________________________\n",
-      "activation_5 (Activation)    (None, 512)               0         \n",
-      "_________________________________________________________________\n",
-      "dropout_3 (Dropout)          (None, 512)               0         \n",
-      "_________________________________________________________________\n",
-      "dense_2 (Dense)              (None, 10)                5130      \n",
-      "_________________________________________________________________\n",
-      "activation_6 (Activation)    (None, 10)                0         \n",
-      "=================================================================\n",
-      "Total params: 1,250,858\n",
-      "Trainable params: 1,250,858\n",
-      "Non-trainable params: 0\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "model1 = Sequential()\n",
-    "\n",
-    "model1.add(Conv2D(32, (3, 3), padding='same',\n",
-    "                 input_shape=x_train.shape[1:]))\n",
-    "model1.add(Activation('relu'))\n",
-    "model1.add(Conv2D(32, (3, 3)))\n",
-    "model1.add(Activation('relu'))\n",
-    "model1.add(MaxPooling2D(pool_size=(2, 2)))\n",
-    "model1.add(Dropout(0.25))\n",
-    "\n",
-    "model1.add(Conv2D(64, (3, 3), padding='same'))\n",
-    "model1.add(Activation('relu'))\n",
-    "model1.add(Conv2D(64, (3, 3)))\n",
-    "model1.add(Activation('relu'))\n",
-    "model1.add(MaxPooling2D(pool_size=(2, 2)))\n",
-    "model1.add(Dropout(0.25))\n",
-    "\n",
-    "model1.add(Flatten())\n",
-    "model1.add(Dense(512))\n",
-    "model1.add(Activation('relu'))\n",
-    "model1.add(Dropout(0.5))\n",
-    "model1.add(Dense(num_classes))\n",
-    "model1.add(Activation('softmax'))\n",
-    "\n",
-    "model1.summary()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 8,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"conv2d_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"filters\": 32, \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"batch_input_shape\": [null, 32, 32, 3], \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_1\"}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_constraint\": null, \"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"conv2d_2\", \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"valid\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 32, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_2\"}}, {\"class_name\": \"MaxPooling2D\", \"config\": {\"name\": \"max_pooling2d_1\", \"trainable\": true, \"data_format\": \"channels_last\", \"pool_size\": [2, 2], \"padding\": \"valid\", \"strides\": [2, 2]}}, {\"class_name\": \"Dropout\", \"config\": {\"rate\": 0.25, \"noise_shape\": null, \"trainable\": true, \"seed\": null, \"name\": \"dropout_1\"}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_constraint\": null, \"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"conv2d_3\", \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 64, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_3\"}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_constraint\": null, \"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"conv2d_4\", \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"linear\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"valid\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 64, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"Activation\", \"config\": {\"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_4\"}}, {\"class_name\": \"MaxPooling2D\", \"config\": {\"name\": \"max_pooling2d_2\", \"trainable\": true, \"data_format\": \"channels_last\", \"pool_size\": [2, 2], \"padding\": \"valid\", \"strides\": [2, 2]}}, {\"class_name\": \"Dropout\", \"config\": {\"rate\": 0.25, \"noise_shape\": null, \"trainable\": true, \"seed\": null, \"name\": \"dropout_2\"}}, {\"class_name\": \"Flatten\", \"config\": {\"trainable\": true, \"name\": \"flatten_1\", \"data_format\": \"channels_last\"}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_1\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 512, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Activation\", \"config\": {\"activation\": \"relu\", \"trainable\": true, \"name\": \"activation_5\"}}, {\"class_name\": \"Dropout\", \"config\": {\"rate\": 0.5, \"noise_shape\": null, \"trainable\": true, \"seed\": null, \"name\": \"dropout_3\"}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_2\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"linear\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"Activation\", \"config\": {\"activation\": \"softmax\", \"trainable\": true, \"name\": \"activation_6\"}}], \"backend\": \"tensorflow\"}'"
-      ]
-     },
-     "execution_count": 8,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "model1.to_json()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "b. Model architecture from https://machinelearningmastery.com/how-to-develop-a-cnn-from-scratch-for-cifar-10-photo-classification/"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 9,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "conv2d_5 (Conv2D)            (None, 32, 32, 32)        896       \n",
-      "_________________________________________________________________\n",
-      "batch_normalization_1 (Batch (None, 32, 32, 32)        128       \n",
-      "_________________________________________________________________\n",
-      "conv2d_6 (Conv2D)            (None, 32, 32, 32)        9248      \n",
-      "_________________________________________________________________\n",
-      "batch_normalization_2 (Batch (None, 32, 32, 32)        128       \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_3 (MaxPooling2 (None, 16, 16, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "dropout_4 (Dropout)          (None, 16, 16, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "conv2d_7 (Conv2D)            (None, 16, 16, 64)        18496     \n",
-      "_________________________________________________________________\n",
-      "batch_normalization_3 (Batch (None, 16, 16, 64)        256       \n",
-      "_________________________________________________________________\n",
-      "conv2d_8 (Conv2D)            (None, 16, 16, 64)        36928     \n",
-      "_________________________________________________________________\n",
-      "batch_normalization_4 (Batch (None, 16, 16, 64)        256       \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_4 (MaxPooling2 (None, 8, 8, 64)          0         \n",
-      "_________________________________________________________________\n",
-      "dropout_5 (Dropout)          (None, 8, 8, 64)          0         \n",
-      "_________________________________________________________________\n",
-      "conv2d_9 (Conv2D)            (None, 8, 8, 128)         73856     \n",
-      "_________________________________________________________________\n",
-      "batch_normalization_5 (Batch (None, 8, 8, 128)         512       \n",
-      "_________________________________________________________________\n",
-      "conv2d_10 (Conv2D)           (None, 8, 8, 128)         147584    \n",
-      "_________________________________________________________________\n",
-      "batch_normalization_6 (Batch (None, 8, 8, 128)         512       \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_5 (MaxPooling2 (None, 4, 4, 128)         0         \n",
-      "_________________________________________________________________\n",
-      "dropout_6 (Dropout)          (None, 4, 4, 128)         0         \n",
-      "_________________________________________________________________\n",
-      "flatten_2 (Flatten)          (None, 2048)              0         \n",
-      "_________________________________________________________________\n",
-      "dense_3 (Dense)              (None, 128)               262272    \n",
-      "_________________________________________________________________\n",
-      "batch_normalization_7 (Batch (None, 128)               512       \n",
-      "_________________________________________________________________\n",
-      "dropout_7 (Dropout)          (None, 128)               0         \n",
-      "_________________________________________________________________\n",
-      "dense_4 (Dense)              (None, 10)                1290      \n",
-      "=================================================================\n",
-      "Total params: 552,874\n",
-      "Trainable params: 551,722\n",
-      "Non-trainable params: 1,152\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "model2 = Sequential()\n",
-    "\n",
-    "model2.add(Conv2D(32, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same', input_shape=(32, 32, 3)))\n",
-    "model2.add(BatchNormalization())\n",
-    "model2.add(Conv2D(32, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model2.add(BatchNormalization())\n",
-    "model2.add(MaxPooling2D((2, 2)))\n",
-    "model2.add(Dropout(0.2))\n",
-    "\n",
-    "model2.add(Conv2D(64, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model2.add(BatchNormalization())\n",
-    "model2.add(Conv2D(64, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model2.add(BatchNormalization())\n",
-    "model2.add(MaxPooling2D((2, 2)))\n",
-    "model2.add(Dropout(0.3))\n",
-    "\n",
-    "model2.add(Conv2D(128, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model2.add(BatchNormalization())\n",
-    "model2.add(Conv2D(128, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model2.add(BatchNormalization())\n",
-    "model2.add(MaxPooling2D((2, 2)))\n",
-    "model2.add(Dropout(0.4))\n",
-    "\n",
-    "model2.add(Flatten())\n",
-    "model2.add(Dense(128, activation='relu', kernel_initializer='he_uniform'))\n",
-    "model2.add(BatchNormalization())\n",
-    "model2.add(Dropout(0.5))\n",
-    "model2.add(Dense(10, activation='softmax'))\n",
-    "\n",
-    "model2.summary()"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 10,
-   "metadata": {},
-   "outputs": [
-    {
-     "data": {
-      "text/plain": [
-       "'{\"class_name\": \"Sequential\", \"keras_version\": \"2.1.6\", \"config\": [{\"class_name\": \"Conv2D\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 2.0, \"seed\": null, \"mode\": \"fan_in\"}}, \"name\": \"conv2d_5\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"dtype\": \"float32\", \"activation\": \"relu\", \"trainable\": true, \"data_format\": \"channels_last\", \"filters\": 32, \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"batch_input_shape\": [null, 32, 32, 3], \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"BatchNormalization\", \"config\": {\"beta_constraint\": null, \"gamma_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"moving_mean_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"name\": \"batch_normalization_1\", \"epsilon\": 0.001, \"trainable\": true, \"moving_variance_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"beta_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"scale\": true, \"axis\": -1, \"gamma_constraint\": null, \"gamma_regularizer\": null, \"beta_regularizer\": null, \"momentum\": 0.99, \"center\": true}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_constraint\": null, \"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 2.0, \"seed\": null, \"mode\": \"fan_in\"}}, \"name\": \"conv2d_6\", \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 32, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"BatchNormalization\", \"config\": {\"beta_constraint\": null, \"gamma_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"moving_mean_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"name\": \"batch_normalization_2\", \"epsilon\": 0.001, \"trainable\": true, \"moving_variance_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"beta_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"scale\": true, \"axis\": -1, \"gamma_constraint\": null, \"gamma_regularizer\": null, \"beta_regularizer\": null, \"momentum\": 0.99, \"center\": true}}, {\"class_name\": \"MaxPooling2D\", \"config\": {\"name\": \"max_pooling2d_3\", \"trainable\": true, \"data_format\": \"channels_last\", \"pool_size\": [2, 2], \"padding\": \"valid\", \"strides\": [2, 2]}}, {\"class_name\": \"Dropout\", \"config\": {\"rate\": 0.2, \"noise_shape\": null, \"trainable\": true, \"seed\": null, \"name\": \"dropout_4\"}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_constraint\": null, \"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 2.0, \"seed\": null, \"mode\": \"fan_in\"}}, \"name\": \"conv2d_7\", \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 64, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"BatchNormalization\", \"config\": {\"beta_constraint\": null, \"gamma_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"moving_mean_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"name\": \"batch_normalization_3\", \"epsilon\": 0.001, \"trainable\": true, \"moving_variance_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"beta_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"scale\": true, \"axis\": -1, \"gamma_constraint\": null, \"gamma_regularizer\": null, \"beta_regularizer\": null, \"momentum\": 0.99, \"center\": true}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_constraint\": null, \"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 2.0, \"seed\": null, \"mode\": \"fan_in\"}}, \"name\": \"conv2d_8\", \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 64, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"BatchNormalization\", \"config\": {\"beta_constraint\": null, \"gamma_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"moving_mean_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"name\": \"batch_normalization_4\", \"epsilon\": 0.001, \"trainable\": true, \"moving_variance_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"beta_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"scale\": true, \"axis\": -1, \"gamma_constraint\": null, \"gamma_regularizer\": null, \"beta_regularizer\": null, \"momentum\": 0.99, \"center\": true}}, {\"class_name\": \"MaxPooling2D\", \"config\": {\"name\": \"max_pooling2d_4\", \"trainable\": true, \"data_format\": \"channels_last\", \"pool_size\": [2, 2], \"padding\": \"valid\", \"strides\": [2, 2]}}, {\"class_name\": \"Dropout\", \"config\": {\"rate\": 0.3, \"noise_shape\": null, \"trainable\": true, \"seed\": null, \"name\": \"dropout_5\"}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_constraint\": null, \"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 2.0, \"seed\": null, \"mode\": \"fan_in\"}}, \"name\": \"conv2d_9\", \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 128, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"BatchNormalization\", \"config\": {\"beta_constraint\": null, \"gamma_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"moving_mean_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"name\": \"batch_normalization_5\", \"epsilon\": 0.001, \"trainable\": true, \"moving_variance_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"beta_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"scale\": true, \"axis\": -1, \"gamma_constraint\": null, \"gamma_regularizer\": null, \"beta_regularizer\": null, \"momentum\": 0.99, \"center\": true}}, {\"class_name\": \"Conv2D\", \"config\": {\"kernel_constraint\": null, \"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 2.0, \"seed\": null, \"mode\": \"fan_in\"}}, \"name\": \"conv2d_10\", \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"data_format\": \"channels_last\", \"padding\": \"same\", \"strides\": [1, 1], \"dilation_rate\": [1, 1], \"kernel_regularizer\": null, \"filters\": 128, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"use_bias\": true, \"activity_regularizer\": null, \"kernel_size\": [3, 3]}}, {\"class_name\": \"BatchNormalization\", \"config\": {\"beta_constraint\": null, \"gamma_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"moving_mean_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"name\": \"batch_normalization_6\", \"epsilon\": 0.001, \"trainable\": true, \"moving_variance_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"beta_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"scale\": true, \"axis\": -1, \"gamma_constraint\": null, \"gamma_regularizer\": null, \"beta_regularizer\": null, \"momentum\": 0.99, \"center\": true}}, {\"class_name\": \"MaxPooling2D\", \"config\": {\"name\": \"max_pooling2d_5\", \"trainable\": true, \"data_format\": \"channels_last\", \"pool_size\": [2, 2], \"padding\": \"valid\", \"strides\": [2, 2]}}, {\"class_name\": \"Dropout\", \"config\": {\"rate\": 0.4, \"noise_shape\": null, \"trainable\": true, \"seed\": null, \"name\": \"dropout_6\"}}, {\"class_name\": \"Flatten\", \"config\": {\"trainable\": true, \"name\": \"flatten_2\", \"data_format\": \"channels_last\"}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 2.0, \"seed\": null, \"mode\": \"fan_in\"}}, \"name\": \"dense_3\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"relu\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 128, \"use_bias\": true, \"activity_regularizer\": null}}, {\"class_name\": \"BatchNormalization\", \"config\": {\"beta_constraint\": null, \"gamma_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"moving_mean_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"name\": \"batch_normalization_7\", \"epsilon\": 0.001, \"trainable\": true, \"moving_variance_initializer\": {\"class_name\": \"Ones\", \"config\": {}}, \"beta_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"scale\": true, \"axis\": -1, \"gamma_constraint\": null, \"gamma_regularizer\": null, \"beta_regularizer\": null, \"momentum\": 0.99, \"center\": true}}, {\"class_name\": \"Dropout\", \"config\": {\"rate\": 0.5, \"noise_shape\": null, \"trainable\": true, \"seed\": null, \"name\": \"dropout_7\"}}, {\"class_name\": \"Dense\", \"config\": {\"kernel_initializer\": {\"class_name\": \"VarianceScaling\", \"config\": {\"distribution\": \"uniform\", \"scale\": 1.0, \"seed\": null, \"mode\": \"fan_avg\"}}, \"name\": \"dense_4\", \"kernel_constraint\": null, \"bias_regularizer\": null, \"bias_constraint\": null, \"activation\": \"softmax\", \"trainable\": true, \"kernel_regularizer\": null, \"bias_initializer\": {\"class_name\": \"Zeros\", \"config\": {}}, \"units\": 10, \"use_bias\": true, \"activity_regularizer\": null}}], \"backend\": \"tensorflow\"}'"
-      ]
-     },
-     "execution_count": 10,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "model2.to_json()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "c. Another model architecture from https://machinelearningmastery.com/how-to-develop-a-cnn-from-scratch-for-cifar-10-photo-classification/"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 11,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "_________________________________________________________________\n",
-      "Layer (type)                 Output Shape              Param #   \n",
-      "=================================================================\n",
-      "conv2d_11 (Conv2D)           (None, 32, 32, 32)        896       \n",
-      "_________________________________________________________________\n",
-      "conv2d_12 (Conv2D)           (None, 32, 32, 32)        9248      \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_6 (MaxPooling2 (None, 16, 16, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "dropout_8 (Dropout)          (None, 16, 16, 32)        0         \n",
-      "_________________________________________________________________\n",
-      "conv2d_13 (Conv2D)           (None, 16, 16, 64)        18496     \n",
-      "_________________________________________________________________\n",
-      "conv2d_14 (Conv2D)           (None, 16, 16, 64)        36928     \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_7 (MaxPooling2 (None, 8, 8, 64)          0         \n",
-      "_________________________________________________________________\n",
-      "dropout_9 (Dropout)          (None, 8, 8, 64)          0         \n",
-      "_________________________________________________________________\n",
-      "conv2d_15 (Conv2D)           (None, 8, 8, 128)         73856     \n",
-      "_________________________________________________________________\n",
-      "conv2d_16 (Conv2D)           (None, 8, 8, 128)         147584    \n",
-      "_________________________________________________________________\n",
-      "max_pooling2d_8 (MaxPooling2 (None, 4, 4, 128)         0         \n",
-      "_________________________________________________________________\n",
-      "dropout_10 (Dropout)         (None, 4, 4, 128)         0         \n",
-      "_________________________________________________________________\n",
-      "flatten_3 (Flatten)          (None, 2048)              0         \n",
-      "_________________________________________________________________\n",
-      "dense_5 (Dense)              (None, 128)               262272    \n",
-      "_________________________________________________________________\n",
-      "dropout_11 (Dropout)         (None, 128)               0         \n",
-      "_________________________________________________________________\n",
-      "dense_6 (Dense)              (None, 10)                1290      \n",
-      "=================================================================\n",
-      "Total params: 550,570\n",
-      "Trainable params: 550,570\n",
-      "Non-trainable params: 0\n",
-      "_________________________________________________________________\n"
-     ]
-    }
-   ],
-   "source": [
-    "model3 = Sequential()\n",
-    "\n",
-    "model3.add(Conv2D(32, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same', input_shape=(32, 32, 3)))\n",
-    "model3.add(Conv2D(32, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model3.add(MaxPooling2D((2, 2)))\n",
-    "model3.add(Dropout(0.2))\n",
-    "model3.add(Conv2D(64, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model3.add(Conv2D(64, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model3.add(MaxPooling2D((2, 2)))\n",
-    "model3.add(Dropout(0.3))\n",
-    "model3.add(Conv2D(128, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model3.add(Conv2D(128, (3, 3), activation='relu', kernel_initializer='he_uniform', padding='same'))\n",
-    "model3.add(MaxPooling2D((2, 2)))\n",
-    "model3.add(Dropout(0.4))\n",
-    "model3.add(Flatten())\n",
-    "model3.add(Dense(128, activation='relu', kernel_initializer='he_uniform'))\n",
-    "model3.add(Dropout(0.5))\n",
-    "model3.add(Dense(10, activation='softmax'))\n",
-    "\n",
-    "model3.summary()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Load into model architecture table using psycopg2"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 12,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "3 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>model_id</th>\n",
-       "        <th>name</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>CNN from Keras docs for CIFAR-10</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>CNN from Jason Brownlee blog post</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>CNN from Jason Brownlee blog post - no batch normalization</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, u'CNN from Keras docs for CIFAR-10'),\n",
-       " (2, u'CNN from Jason Brownlee blog post'),\n",
-       " (3, u'CNN from Jason Brownlee blog post - no batch normalization')]"
-      ]
-     },
-     "execution_count": 12,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "import psycopg2 as p2\n",
-    "#conn = p2.connect('postgresql://gpadmin@35.239.240.26:5432/madlib')\n",
-    "#conn = p2.connect('postgresql://fmcquillan@localhost:5432/madlib')\n",
-    "conn = p2.connect('postgresql://gpadmin@localhost:8000/cifar_demo')\n",
-    "cur = conn.cursor()\n",
-    "\n",
-    "%sql DROP TABLE IF EXISTS model_arch_table_cifar10;\n",
-    "query = \"SELECT madlib.load_keras_model('model_arch_table_cifar10', %s, NULL, %s)\"\n",
-    "cur.execute(query,[model1.to_json(), \"CNN from Keras docs for CIFAR-10\"])\n",
-    "conn.commit()\n",
-    "\n",
-    "query = \"SELECT madlib.load_keras_model('model_arch_table_cifar10', %s, NULL, %s)\"\n",
-    "cur.execute(query,[model2.to_json(), \"CNN from Jason Brownlee blog post\"])\n",
-    "conn.commit()\n",
-    "\n",
-    "query = \"SELECT madlib.load_keras_model('model_arch_table_cifar10', %s, NULL, %s)\"\n",
-    "cur.execute(query,[model3.to_json(), \"CNN from Jason Brownlee blog post - no batch normalization\"])\n",
-    "conn.commit()\n",
-    "\n",
-    "# check model loaded OK\n",
-    "%sql SELECT model_id, name FROM model_arch_table_cifar10 ORDER BY model_id;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"hyperband\"></a>\n",
-    "# 4.  Hyperband diagonal\n",
-    "\n",
-    "Create tables for intermediate and overall results from Hyperband, which is running on top of MADlib model selection methods."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 13,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "Done.\n",
-      "Done.\n",
-      "Done.\n",
-      "Done.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "Done.\n",
-      "Done.\n",
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "[]"
-      ]
-     },
-     "execution_count": 13,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "-- overall results table\n",
-    "DROP TABLE IF EXISTS results_cifar10;\n",
-    "CREATE TABLE results_cifar10 ( \n",
-    "                      mst_key INTEGER,  -- note not SERIAL\n",
-    "                      model_id INTEGER, \n",
-    "                      compile_params TEXT,\n",
-    "                      fit_params TEXT, \n",
-    "                      model_type TEXT, \n",
-    "                      model_size DOUBLE PRECISION, \n",
-    "                      metrics_elapsed_time DOUBLE PRECISION[], \n",
-    "                      metrics_type TEXT[], \n",
-    "                      training_metrics_final DOUBLE PRECISION, \n",
-    "                      training_loss_final DOUBLE PRECISION, \n",
-    "                      training_metrics DOUBLE PRECISION[], \n",
-    "                      training_loss DOUBLE PRECISION[], \n",
-    "                      validation_metrics_final DOUBLE PRECISION, \n",
-    "                      validation_loss_final DOUBLE PRECISION, \n",
-    "                      validation_metrics DOUBLE PRECISION[], \n",
-    "                      validation_loss DOUBLE PRECISION[], \n",
-    "                      model_arch_table TEXT, \n",
-    "                      num_iterations INTEGER, \n",
-    "                      start_training_time TIMESTAMP, \n",
-    "                      end_training_time TIMESTAMP,\n",
-    "                      s INTEGER,            -- bracket number from Hyperband\n",
-    "                      i INTEGER,            -- iteration corresponding to successive having within a bracket\n",
-    "                      run_id SERIAL         -- global counter for the training runs\n",
-    "                     );\n",
-    "\n",
-    "-- all model selections:\n",
-    "-- model selection table containing all model configs (all brackets)\n",
-    "DROP TABLE IF EXISTS mst_table_hb_cifar10;\n",
-    "CREATE TABLE mst_table_hb_cifar10 (\n",
-    "                           mst_key SERIAL, \n",
-    "                           s INTEGER,        -- bracket\n",
-    "                           model_id INTEGER, \n",
-    "                           compile_params VARCHAR, \n",
-    "                           fit_params VARCHAR\n",
-    "                          );\n",
-    "\n",
-    "-- model selection summary table\n",
-    "DROP TABLE IF EXISTS mst_table_hb_cifar10_summary;\n",
-    "CREATE TABLE mst_table_hb_cifar10_summary (model_arch_table VARCHAR);\n",
-    "INSERT INTO mst_table_hb_cifar10_summary VALUES ('model_arch_table_cifar10');\n",
-    "\n",
-    "-- diagonal model selections:\n",
-    "-- model selection table for diagonal: fit() will be called on a per diagonal basis\n",
-    "DROP TABLE IF EXISTS mst_diag_table_hb_cifar10;\n",
-    "CREATE TABLE mst_diag_table_hb_cifar10 (\n",
-    "                           mst_key INTEGER, -- note not SERIAL since this table derived from main model selection table\n",
-    "                           s INTEGER,          -- bracket\n",
-    "                           model_id INTEGER, \n",
-    "                           compile_params VARCHAR, \n",
-    "                           fit_params VARCHAR\n",
-    "                          );\n",
-    "\n",
-    "-- model selection summary table for diagonal table\n",
-    "DROP TABLE IF EXISTS mst_diag_table_hb_cifar10_summary;\n",
-    "CREATE TABLE mst_diag_table_hb_cifar10_summary (model_arch_table VARCHAR);\n",
-    "INSERT INTO mst_diag_table_hb_cifar10_summary VALUES ('model_arch_table_cifar10');"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Generalize table names"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 14,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "results_table = 'results_cifar10'\n",
-    "\n",
-    "output_table = 'cifar10_multi_model'\n",
-    "output_table_info = '_'.join([output_table, 'info'])\n",
-    "output_table_summary = '_'.join([output_table, 'summary'])\n",
-    "\n",
-    "best_model = 'cifar10_best_model'\n",
-    "best_model_info = '_'.join([best_model, 'info'])\n",
-    "best_model_summary = '_'.join([best_model, 'summary'])\n",
-    "\n",
-    "\n",
-    "mst_table = 'mst_table_hb_cifar10'\n",
-    "mst_table_summary = '_'.join([mst_table, 'summary'])\n",
-    "\n",
-    "mst_diag_table = 'mst_diag_table_hb_cifar10'\n",
-    "mst_diag_table_summary = '_'.join([mst_diag_table, 'summary'])\n",
-    "\n",
-    "model_arch_table = 'model_arch_table_cifar10'"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Hyperband diagonal logic"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Define variables for Hyperband"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 8,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "max_iter = 27   # maximum iterations per configuration\n",
-    "eta = 3        # defines downsampling rate (default = 3)\n",
-    "skip_last = 0  # 1 means skip last run in each bracket, 0 means run full bracket"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 9,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "import numpy as np\n",
-    "from random import random\n",
-    "from math import log, ceil\n",
-    "from time import time, ctime\n",
-    "\n",
-    "class Hyperband_diagonal:\n",
-    "    \n",
-    "    def __init__( self, get_params_function, try_params_function ):\n",
-    "        self.get_params = get_params_function #\n",
-    "        self.try_params = try_params_function\n",
-    "\n",
-    "        self.max_iter = max_iter \n",
-    "        self.eta = eta \n",
-    "        self.skip_last = skip_last  \n",
-    "\n",
-    "        self.logeta = lambda x: log( x ) / log( self.eta )\n",
-    "        self.s_max = int( self.logeta( self.max_iter ))\n",
-    "        self.B = ( self.s_max + 1 ) * self.max_iter\n",
-    "        \n",
-    "        #echo output\n",
-    "        print (\"max_iter = \" + str(self.max_iter))\n",
-    "        print (\"eta = \" + str(self.eta))\n",
-    "        print (\"B = \" + str(self.s_max+1) + \"*max_iter = \" + str(self.B))\n",
-    "        print (\"skip_last = \" + str(self.skip_last))\n",
-    "        \n",
-    "        self.setup_full_schedule()\n",
-    "        self.create_mst_superset()\n",
-    "        \n",
-    "        self.best_loss = np.inf\n",
-    "        self.best_accuracy = 0.0\n",
-    "\n",
-    "    # create full Hyperband schedule for all brackets ahead of time\n",
-    "    def setup_full_schedule(self):\n",
-    "        self.n_vals = np.zeros((self.s_max+1, self.s_max+1), dtype=int)\n",
-    "        self.r_vals = np.zeros((self.s_max+1, self.s_max+1), dtype=int)\n",
-    "        \n",
-    "        print (\" \")\n",
-    "        print (\"Hyperband brackets\")\n",
-    "\n",
-    "        # loop through each bracket in reverse order\n",
-    "        for s in reversed(range(self.s_max+1)):\n",
-    "            \n",
-    "            print (\" \")\n",
-    "            print (\"s=\" + str(s))\n",
-    "            print (\"n_i      r_i\")\n",
-    "            print (\"------------\")\n",
-    "\n",
-    "            for i in range(s+1):\n",
-    "                # n_i configs for r_i iterations\n",
-    "                n_i = n*self.eta**(-i)\n",
-    "                r_i = r*self.eta**(i)\n",
-    "\n",
-    "                self.n_vals[s][i] = n_i\n",
-    "                self.r_vals[s][i] = r_i\n",
-    "\n",
-    "                print (str(n_i) + \"     \" + str (r_i))\n",
-    "           \n",
-    "        \n",
-    "    # generate model selection tuples for all brackets\n",
-    "    def create_mst_superset(self):\n",
-    "        \n",
-    "        print (\" \")\n",
-    "        print (\"Create superset of MSTs for each bracket s\")\n",
-    "        \n",
-    "        # get hyper parameter configs for each bracket s\n",
-    "        for s in reversed(range(self.s_max+1)):\n",
-    "            n = int(ceil(int(self.B/self.max_iter/(s+1))*self.eta**s)) # initial number of configurations\n",
-    "            r = self.max_iter*self.eta**(-s) # initial number of iterations to run configurations for\n",
-    "\n",
-    "            print (\" \")\n",
-    "            print (\"s=\" + str(s))\n",
-    "            print (\"n=\" + str(n))\n",
-    "            print (\"r=\" + str(r))\n",
-    "            print (\" \")\n",
-    "            \n",
-    "            # n random configurations for each bracket s\n",
-    "            self.get_params(n, s)\n",
-    "            \n",
-    "            \n",
-    "    # Hyperband diagonal logic\n",
-    "    def run(self):   \n",
-    "        \n",
-    "        print (\" \")\n",
-    "        print (\"Hyperband diagonal\")\n",
-    "        print (\"Outer loop on diagonal:\")\n",
-    "        \n",
-    "        # outer loop on diagonal\n",
-    "        #for i in range(self.s_max+1):\n",
-    "        for i in range((self.s_max+1) - int(self.skip_last)):\n",
-    "            print (\" \")\n",
-    "            print (\"i=\" + str(i))\n",
-    "    \n",
-    "            # zero out diagonal table\n",
-    "            %sql TRUNCATE TABLE $mst_diag_table\n",
-    "            \n",
-    "            # loop on brackets s desc to create diagonal table\n",
-    "            print (\"Loop on s desc to create diagonal table:\")\n",
-    "            for s in range(self.s_max, self.s_max-i-1, -1):\n",
-    "\n",
-    "                # build up mst table for diagonal\n",
-    "                %sql INSERT INTO $mst_diag_table (SELECT * FROM $mst_table WHERE s=$s);\n",
-    "            \n",
-    "            # first pass\n",
-    "            if i == 0:\n",
-    "                first_pass = True\n",
-    "            else:\n",
-    "                first_pass = False\n",
-    "                \n",
-    "            # multi-model training\n",
-    "            print (\" \")\n",
-    "            print (\"Try params for i = \" + str(i))\n",
-    "            U = self.try_params(i, self.r_vals[self.s_max][i], first_pass) # r_i is the same for all diagonal elements\n",
-    "            \n",
-    "            # loop on brackets s desc to prune model selection table\n",
-    "            # don't need to prune if finished last diagonal\n",
-    "            #if i < (self.s_max):\n",
-    "            if i < (self.s_max - int(self.skip_last)):\n",
-    "                print (\"Loop on s desc to prune mst table:\")\n",
-    "                for s in range(self.s_max, self.s_max-i-1, -1):\n",
-    "                    \n",
-    "                    # compute number of configs to keep\n",
-    "                    # remember i value is different for each bracket s on the diagonal\n",
-    "                    k = int( self.n_vals[s][s-self.s_max+i] / self.eta)\n",
-    "                    print (\"Pruning s = {} with k = {}\".format(s, k))\n",
-    "\n",
-    "                    # temporarily re-define table names due to weird Python scope issues\n",
-    "                    results_table = 'results_cifar10'\n",
-    "\n",
-    "                    output_table = 'cifar10_multi_model'\n",
-    "                    output_table_info = '_'.join([output_table, 'info'])\n",
-    "                    output_table_summary = '_'.join([output_table, 'summary'])\n",
-    "\n",
-    "                    mst_table = 'mst_table_hb_cifar10'\n",
-    "                    mst_table_summary = '_'.join([mst_table, 'summary'])\n",
-    "\n",
-    "                    mst_diag_table = 'mst_diag_table_hb_cifar10'\n",
-    "                    mst_diag_table_summary = '_'.join([mst_diag_table, 'summary'])\n",
-    "\n",
-    "                    model_arch_table = 'model_arch_table_cifar10'\n",
-    "            \n",
-    "                    query = \"\"\"\n",
-    "                    DELETE FROM {mst_table} WHERE s={s} AND mst_key NOT IN (SELECT {output_table_info}.mst_key FROM {output_table_info} JOIN {mst_table} ON {output_table_info}.mst_key={mst_table}.mst_key WHERE s={s} ORDER BY validation_loss_final ASC LIMIT {k}::INT);\n",
-    "                    \"\"\".format(**locals())\n",
-    "                    cur.execute(query)\n",
-    "                    conn.commit()\n",
-    "                    \n",
-    "                    # these were not working so used cursor instead\n",
-    "                    #%sql DELETE FROM $mst_table WHERE s=$s AND mst_key NOT IN (SELECT $output_table_info.mst_key FROM $output_table_info JOIN $mst_table ON $output_table_info.mst_key=$mst_table.mst_key WHERE s=$s ORDER BY validation_loss_final ASC LIMIT $k::INT);\n",
-    "                    #%sql DELETE FROM mst_table_hb_cifar10 WHERE s=1 AND mst_key NOT IN (SELECT cifar10_multi_model_info.mst_key FROM cifar10_multi_model_info JOIN mst_table_hb_cifar10 ON cifar10_multi_model_info.mst_key=mst_table_hb_cifar10.mst_key WHERE s=1 ORDER BY validation_loss_final ASC LIMIT 1);\n",
-    "        \n",
-    "            # keep track of best loss so far and save the model for inference\n",
-    "            # get best loss and accuracy from this diagonal run\n",
-    "            # (need to check if this will work OK if don't evaluate metrics every iteration)\n",
-    "            loss = %sql SELECT validation_loss_final FROM $output_table_info ORDER BY validation_loss_final ASC LIMIT 1;\n",
-    "            accuracy = %sql SELECT validation_metrics_final FROM $output_table_info ORDER BY validation_metrics_final DESC LIMIT 1;\n",
-    "                    \n",
-    "            # save best model based on accuracy (could do loss if you wanted)\n",
-    "            if accuracy > self.best_accuracy:\n",
-    "                \n",
-    "                self.best_accuracy = accuracy\n",
-    "                \n",
-    "                # get best mst_key\n",
-    "                best_mst_key = %sql SELECT mst_key FROM $output_table_info ORDER BY validation_metrics_final DESC LIMIT 1; \n",
-    "                best_mst_key = best_mst_key.DataFrame().to_numpy()[0][0]\n",
-    "\n",
-    "                # save model table (1 row for best model)\n",
-    "                %sql DROP TABLE IF EXISTS $best_model;\n",
-    "                %sql CREATE TABLE $best_model AS SELECT * FROM $output_table WHERE mst_key = $best_mst_key;\n",
-    "\n",
-    "                # save info table (1 row for best model)\n",
-    "                %sql DROP TABLE IF EXISTS $best_model_info;\n",
-    "                %sql CREATE TABLE $best_model_info AS SELECT * FROM $output_table_info WHERE mst_key = $best_mst_key;\n",
-    " \n",
-    "                # save summary table\n",
-    "                %sql DROP TABLE IF EXISTS $best_model_summary;\n",
-    "                %sql CREATE TABLE $best_model_summary AS SELECT * FROM $output_table_summary;\n",
-    "            \n",
-    "            if loss < self.best_loss:\n",
-    "                self.best_loss = loss\n",
-    "                \n",
-    "            print (\" \")\n",
-    "            print (\"Best validation loss so far = \")\n",
-    "            print (str(loss))\n",
-    "            print (\"Best validation accuracy so far = \")\n",
-    "            print (str(accuracy))\n",
-    "            \n",
-    "\n",
-    "            \n",
-    "        return"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Generate params and insert into MST table.  This version of get_params uses the same compile parameters for all optimizers, and the same compile/fit parameters for all model architectures.  (This may be too restrictive in some cases.) -- Note 3/13: check SIGMOID paper runs which I think I may have addressed this to some extent"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 20,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "def get_params(n, s):\n",
-    "    \n",
-    "    from sklearn.model_selection import ParameterSampler\n",
-    "    from scipy.stats.distributions import uniform\n",
-    "    import numpy as np\n",
-    "    \n",
-    "    # model architecture\n",
-    "    model_id = [1,2]\n",
-    "\n",
-    "    # compile params\n",
-    "    # loss function\n",
-    "    loss = ['categorical_crossentropy']\n",
-    "    # optimizer\n",
-    "    optimizer = ['sgd', 'adam', 'rmsprop']\n",
-    "    # learning rate (sample on log scale here not in ParameterSampler)\n",
-    "    lr_range = [0.0001, 0.01]\n",
-    "    lr = 10**np.random.uniform(np.log10(lr_range[0]), np.log10(lr_range[1]), n)\n",
-    "    # metrics\n",
-    "    metrics = ['accuracy']\n",
-    "\n",
-    "    # fit params\n",
-    "    # batch size\n",
-    "    batch_size = [32, 64, 128, 256]\n",
-    "    # epochs\n",
-    "    epochs = [5]\n",
-    "\n",
-    "    # create random param list\n",
-    "    param_grid = {\n",
-    "        'model_id': model_id,\n",
-    "        'loss': loss,\n",
-    "        'optimizer': optimizer,\n",
-    "        'lr': lr,\n",
-    "        'metrics': metrics,\n",
-    "        'batch_size': batch_size,\n",
-    "        'epochs': epochs\n",
-    "    }\n",
-    "    param_list = list(ParameterSampler(param_grid, n_iter=n))\n",
-    "    \n",
-    "    for params in param_list:\n",
-    "\n",
-    "        model_id = str(params.get(\"model_id\"))\n",
-    "        compile_params = \"$$loss='\" + str(params.get(\"loss\")) + \"',optimizer='\" + str(params.get(\"optimizer\")) + \"(lr=\" + str(params.get(\"lr\")) + \")',metrics=['\" + str(params.get(\"metrics\")) + \"']$$\" \n",
-    "        fit_params = \"$$batch_size=\" + str(params.get(\"batch_size\")) + \",epochs=\" + str(params.get(\"epochs\")) + \"$$\"  \n",
-    "        row_content = \"(\" + str(s) + \", \" + model_id + \", \" + compile_params + \", \" + fit_params + \");\"\n",
-    "        \n",
-    "        %sql INSERT INTO $mst_table (s, model_id, compile_params, fit_params) VALUES $row_content\n",
-    "    \n",
-    "    return"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Generate params and insert into MST table.  This version of get_params allows for more customization by optimizer and model architecture.  This is sort of brute force and can be improved."
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 20,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "def get_params(n, s):\n",
-    "    \n",
-    "    from sklearn.model_selection import ParameterSampler\n",
-    "    from scipy.stats.distributions import uniform\n",
-    "    import numpy as np\n",
-    "    \n",
-    "    # number of samples by optimizer\n",
-    "    #n_adam = int(n/3)\n",
-    "    n_adam = int(n/2)\n",
-    "    #n_rmsprop = int(n/3)\n",
-    "    n_rmsprop = 0\n",
-    "    n_sgd = int(n - n_adam - n_rmsprop)\n",
-    "\n",
-    "    # 1) adam\n",
-    "    \n",
-    "    # model architecture\n",
-    "    model_id = [2,3]\n",
-    "\n",
-    "    # compile params\n",
-    "    # loss function\n",
-    "    loss = ['categorical_crossentropy']\n",
-    "    # optimizer\n",
-    "    optimizer = ['adam']\n",
-    "    # learning rate (sample on log scale here not in ParameterSampler)\n",
-    "    lr_range = [0.0001, 0.001]\n",
-    "    lr = 10**np.random.uniform(np.log10(lr_range[0]), np.log10(lr_range[1]), n_adam)\n",
-    "    # metrics\n",
-    "    metrics = ['accuracy']\n",
-    "\n",
-    "    # fit params\n",
-    "    # batch size\n",
-    "    batch_size = [128, 256]\n",
-    "    # epochs\n",
-    "    epochs = [5]\n",
-    "\n",
-    "    # create random param list\n",
-    "    param_grid = {\n",
-    "        'model_id': model_id,\n",
-    "        'loss': loss,\n",
-    "        'optimizer': optimizer,\n",
-    "        'lr': lr,\n",
-    "        'metrics': metrics,\n",
-    "        'batch_size': batch_size,\n",
-    "        'epochs': epochs\n",
-    "    }\n",
-    "    param_list_adam = list(ParameterSampler(param_grid, n_iter=n_adam))\n",
-    "\n",
-    "    # iterate over params\n",
-    "    for params in param_list_adam:\n",
-    "\n",
-    "        model_id = str(params.get(\"model_id\"))\n",
-    "        compile_params = \"$$loss='\" + str(params.get(\"loss\")) + \"',optimizer='\" + str(params.get(\"optimizer\")) + \"(lr=\" + str(params.get(\"lr\")) + \")',metrics=['\" + str(params.get(\"metrics\")) + \"']$$\" \n",
-    "        fit_params = \"$$batch_size=\" + str(params.get(\"batch_size\")) + \",epochs=\" + str(params.get(\"epochs\")) + \"$$\"  \n",
-    "        row_content = \"(\" + str(s) + \", \" + model_id + \", \" + compile_params + \", \" + fit_params + \");\"\n",
-    "    \n",
-    "        # populate mst table\n",
-    "        %sql INSERT INTO $mst_table (s, model_id, compile_params, fit_params) VALUES $row_content\n",
-    "    \n",
-    "    \n",
-    "    # 2) rmsprop\n",
-    "    \n",
-    "    # model architecture\n",
-    "    model_id = [1,2,3]\n",
-    "\n",
-    "    # compile params\n",
-    "    # loss function\n",
-    "    loss = ['categorical_crossentropy']\n",
-    "    # optimizer\n",
-    "    optimizer = ['rmsprop']\n",
-    "    # learning rate (sample on log scale here not in ParameterSampler)\n",
-    "    lr_range = [0.0001, 0.001]\n",
-    "    lr = 10**np.random.uniform(np.log10(lr_range[0]), np.log10(lr_range[1]), n_rmsprop)\n",
-    "    # decay (sample on log scale here not in ParameterSampler if want multiple values)\n",
-    "    decay = [1e-6]\n",
-    "\n",
-    "    # metrics\n",
-    "    metrics = ['accuracy']\n",
-    "\n",
-    "    # fit params\n",
-    "    # batch size\n",
-    "    batch_size = [32, 64, 128, 256]\n",
-    "    # epochs\n",
-    "    epochs = [5]\n",
-    "\n",
-    "    # create random param list\n",
-    "    param_grid = {\n",
-    "        'model_id': model_id,\n",
-    "        'loss': loss,\n",
-    "        'optimizer': optimizer,\n",
-    "        'lr': lr,\n",
-    "        'decay': decay,\n",
-    "        'metrics': metrics,\n",
-    "        'batch_size': batch_size,\n",
-    "        'epochs': epochs\n",
-    "    }\n",
-    "    param_list_rmsprop = list(ParameterSampler(param_grid, n_iter=n_rmsprop))\n",
-    "\n",
-    "    # iterate over params\n",
-    "    for params in param_list_rmsprop:\n",
-    "\n",
-    "        model_id = str(params.get(\"model_id\"))\n",
-    "        compile_params = \"$$loss='\" + str(params.get(\"loss\")) + \"',optimizer='\" + str(params.get(\"optimizer\")) + \"(lr=\" + str(params.get(\"lr\")) + \",decay=\" + str(params.get(\"decay\")) + \")',metrics=['\" + str(params.get(\"metrics\")) + \"']$$\" \n",
-    "        fit_params = \"$$batch_size=\" + str(params.get(\"batch_size\")) + \",epochs=\" + str(params.get(\"epochs\")) + \"$$\"  \n",
-    "        row_content = \"(\" + str(s) + \", \" + model_id + \", \" + compile_params + \", \" + fit_params + \");\"\n",
-    "    \n",
-    "        # populate mst table\n",
-    "        %sql INSERT INTO $mst_table (s, model_id, compile_params, fit_params) VALUES $row_content\n",
-    "\n",
-    "\n",
-    "    # 3) sgd\n",
-    "    \n",
-    "    # model architecture\n",
-    "    model_id = [2,3]\n",
-    "\n",
-    "    # compile params\n",
-    "    # loss function\n",
-    "    loss = ['categorical_crossentropy']\n",
-    "    # optimizer\n",
-    "    optimizer = ['sgd']\n",
-    "    # learning rate (sample on log scale here not in ParameterSampler)\n",
-    "    lr_range = [0.001, 0.005]\n",
-    "    lr = 10**np.random.uniform(np.log10(lr_range[0]), np.log10(lr_range[1]), n_sgd)\n",
-    "    # momentum (sample on log scale here not in ParameterSampler)\n",
-    "    # recall momentum is an exponentially weighted array\n",
-    "    beta_range = [0.9, 0.95]\n",
-    "    beta = 1.0 - 10**np.random.uniform(np.log10(1.0-beta_range[0]), np.log10(1.0-beta_range[1]), n_sgd)\n",
-    "    # metrics\n",
-    "    metrics = ['accuracy']\n",
-    "\n",
-    "    # fit params\n",
-    "    # batch size\n",
-    "    batch_size = [128, 256]\n",
-    "    # epochs\n",
-    "    epochs = [5]\n",
-    "\n",
-    "    # create random param list\n",
-    "    param_grid = {\n",
-    "        'model_id': model_id,\n",
-    "        'loss': loss,\n",
-    "        'optimizer': optimizer,\n",
-    "        'lr': lr,\n",
-    "        'beta': beta,\n",
-    "        'metrics': metrics,\n",
-    "        'batch_size': batch_size,\n",
-    "        'epochs': epochs\n",
-    "    }\n",
-    "    param_list_sgd = list(ParameterSampler(param_grid, n_iter=n_sgd))\n",
-    "\n",
-    "    # iterate over params\n",
-    "    for params in param_list_sgd:\n",
-    "\n",
-    "        model_id = str(params.get(\"model_id\"))\n",
-    "        compile_params = \"$$loss='\" + str(params.get(\"loss\")) + \"',optimizer='\" + str(params.get(\"optimizer\")) + \"(lr=\" + str(params.get(\"lr\")) + \",momentum=\" + str(params.get(\"beta\")) + \")',metrics=['\" + str(params.get(\"metrics\")) + \"']$$\" \n",
-    "        fit_params = \"$$batch_size=\" + str(params.get(\"batch_size\")) + \",epochs=\" + str(params.get(\"epochs\")) + \"$$\"  \n",
-    "        row_content = \"(\" + str(s) + \", \" + model_id + \", \" + compile_params + \", \" + fit_params + \");\"\n",
-    "    \n",
-    "        # populate mst table\n",
-    "        %sql INSERT INTO $mst_table (s, model_id, compile_params, fit_params) VALUES $row_content\n",
-    "\n",
-    "    \n",
-    "    #4) organize mst table\n",
-    "\n",
-    "    #down sample\n",
-    "    #%sql DELETE from $mst_table WHERE mst_key NOT IN (SELECT mst_key FROM $mst_table ORDER BY random() LIMIT $n);\n",
-    "\n",
-    "    # make mst_keys contiguous\n",
-    "    #%sql ALTER TABLE $mst_table DROP COLUMN mst_key;\n",
-    "    #%sql ALTER TABLE $mst_table ADD COLUMN mst_key SERIAL;\n",
-    "    \n",
-    "    return"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Run model hopper for candidates in MST table"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 21,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "def try_params(i, r, first_pass):\n",
-    "    \n",
-    "    # multi-model fit\n",
-    "    if first_pass:\n",
-    "        # cold start\n",
-    "        %sql DROP TABLE IF EXISTS $output_table, $output_table_summary, $output_table_info;\n",
-    "        # passing vars as madlib args does not seem to work\n",
-    "        #%sql SELECT madlib.madlib_keras_fit_multiple_model('cifar10_train_packed', $output_table, $mst_diag_table, $r_i::INT, 0);\n",
-    "        %sql SELECT madlib.madlib_keras_fit_multiple_model('cifar10_train_packed', 'cifar10_multi_model', 'mst_diag_table_hb_cifar10', $r::INT, True, 'cifar10_val_packed',1);\n",
-    "\n",
-    "    else:\n",
-    "        # warm start to continue from previous run\n",
-    "        %sql SELECT madlib.madlib_keras_fit_multiple_model('cifar10_train_packed', 'cifar10_multi_model', 'mst_diag_table_hb_cifar10', $r::INT, True, 'cifar10_val_packed', 1, True);\n",
-    "\n",
-    "    # save results via temp table\n",
-    "    # add everything from info table\n",
-    "    %sql DROP TABLE IF EXISTS temp_results;\n",
-    "    %sql CREATE TABLE temp_results AS (SELECT * FROM $output_table_info);\n",
-    "    \n",
-    "    # add summary table info and i value (same for each row)\n",
-    "    %sql ALTER TABLE temp_results ADD COLUMN model_arch_table TEXT, ADD COLUMN num_iterations INTEGER, ADD COLUMN start_training_time TIMESTAMP, ADD COLUMN end_training_time TIMESTAMP, ADD COLUMN s INTEGER, ADD COLUMN i INTEGER;\n",
-    "    %sql UPDATE temp_results SET model_arch_table = (SELECT model_arch_table FROM $output_table_summary), num_iterations = (SELECT num_iterations FROM $output_table_summary), start_training_time = (SELECT start_training_time FROM $output_table_summary), end_training_time = (SELECT end_training_time FROM $output_table_summary), i = $i;\n",
-    "    \n",
-    "    # get the s value for each run (not the same for each row since diagonal table crosses multiple brackets)\n",
-    "    %sql UPDATE temp_results SET s = m.s FROM mst_diag_table_hb_cifar10 AS m WHERE m.mst_key = temp_results.mst_key;\n",
-    "    \n",
-    "    # copy temp table into results table\n",
-    "    %sql INSERT INTO $results_table (SELECT * FROM temp_results);\n",
-    "\n",
-    "    return"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Call Hyperband diagonal"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 22,
-   "metadata": {
-    "scrolled": false
-   },
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "max_iter = 27\n",
-      "eta = 3\n",
-      "B = 4*max_iter = 108\n",
-      "skip_last = 0\n",
-      " \n",
-      "Hyperband brackets\n",
-      " \n",
-      "s=3\n",
-      "n_i      r_i\n",
-      "------------\n",
-      "27     1.0\n",
-      "9.0     3.0\n",
-      "3.0     9.0\n",
-      "1.0     27.0\n",
-      " \n",
-      "s=2\n",
-      "n_i      r_i\n",
-      "------------\n",
-      "9     3.0\n",
-      "3.0     9.0\n",
-      "1.0     27.0\n",
-      " \n",
-      "s=1\n",
-      "n_i      r_i\n",
-      "------------\n",
-      "6     9.0\n",
-      "2.0     27.0\n",
-      " \n",
-      "s=0\n",
-      "n_i      r_i\n",
-      "------------\n",
-      "4     27\n",
-      " \n",
-      "Create superset of MSTs for each bracket s\n",
-      " \n",
-      "s=3\n",
-      "n=27\n",
-      "r=1.0\n",
-      " \n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      " \n",
-      "s=2\n",
-      "n=9\n",
-      "r=3.0\n",
-      " \n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      " \n",
-      "s=1\n",
-      "n=6\n",
-      "r=9.0\n",
-      " \n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      " \n",
-      "s=0\n",
-      "n=4\n",
-      "r=27\n",
-      " \n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      " \n",
-      "Hyperband diagonal\n",
-      "Outer loop on diagonal:\n",
-      " \n",
-      "i=0\n",
-      "Done.\n",
-      "Loop on s desc to create diagonal table:\n",
-      "27 rows affected.\n",
-      " \n",
-      "Try params for i = 0\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "27 rows affected.\n",
-      "Done.\n",
-      "27 rows affected.\n",
-      "27 rows affected.\n",
-      "27 rows affected.\n",
-      "Loop on s desc to prune mst table:\n",
-      "Pruning s = 3 with k = 9\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      " \n",
-      "Best validation loss so far = \n",
-      "+-----------------------+\n",
-      "| validation_loss_final |\n",
-      "+-----------------------+\n",
-      "|     0.782763898373    |\n",
-      "+-----------------------+\n",
-      "Best validation accuracy so far = \n",
-      "+--------------------------+\n",
-      "| validation_metrics_final |\n",
-      "+--------------------------+\n",
-      "|      0.72729998827       |\n",
-      "+--------------------------+\n",
-      " \n",
-      "i=1\n",
-      "Done.\n",
-      "Loop on s desc to create diagonal table:\n",
-      "9 rows affected.\n",
-      "9 rows affected.\n",
-      " \n",
-      "Try params for i = 1\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "18 rows affected.\n",
-      "Done.\n",
-      "18 rows affected.\n",
-      "18 rows affected.\n",
-      "18 rows affected.\n",
-      "Loop on s desc to prune mst table:\n",
-      "Pruning s = 3 with k = 3\n",
-      "Pruning s = 2 with k = 3\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      " \n",
-      "Best validation loss so far = \n",
-      "+-----------------------+\n",
-      "| validation_loss_final |\n",
-      "+-----------------------+\n",
-      "|     0.602479159832    |\n",
-      "+-----------------------+\n",
-      "Best validation accuracy so far = \n",
-      "+--------------------------+\n",
-      "| validation_metrics_final |\n",
-      "+--------------------------+\n",
-      "|      0.805599987507      |\n",
-      "+--------------------------+\n",
-      " \n",
-      "i=2\n",
-      "Done.\n",
-      "Loop on s desc to create diagonal table:\n",
-      "3 rows affected.\n",
-      "3 rows affected.\n",
-      "6 rows affected.\n",
-      " \n",
-      "Try params for i = 2\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "12 rows affected.\n",
-      "Done.\n",
-      "12 rows affected.\n",
-      "12 rows affected.\n",
-      "12 rows affected.\n",
-      "Loop on s desc to prune mst table:\n",
-      "Pruning s = 3 with k = 1\n",
-      "Pruning s = 2 with k = 1\n",
-      "Pruning s = 1 with k = 2\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      " \n",
-      "Best validation loss so far = \n",
-      "+-----------------------+\n",
-      "| validation_loss_final |\n",
-      "+-----------------------+\n",
-      "|     0.595765888691    |\n",
-      "+-----------------------+\n",
-      "Best validation accuracy so far = \n",
-      "+--------------------------+\n",
-      "| validation_metrics_final |\n",
-      "+--------------------------+\n",
-      "|      0.824999988079      |\n",
-      "+--------------------------+\n",
-      " \n",
-      "i=3\n",
-      "Done.\n",
-      "Loop on s desc to create diagonal table:\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "2 rows affected.\n",
-      "4 rows affected.\n",
-      " \n",
-      "Try params for i = 3\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "8 rows affected.\n",
-      "Done.\n",
-      "8 rows affected.\n",
-      "8 rows affected.\n",
-      "8 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      "Done.\n",
-      "1 rows affected.\n",
-      " \n",
-      "Best validation loss so far = \n",
-      "+-----------------------+\n",
-      "| validation_loss_final |\n",
-      "+-----------------------+\n",
-      "|     0.580716967583    |\n",
-      "+-----------------------+\n",
-      "Best validation accuracy so far = \n",
-      "+--------------------------+\n",
-      "| validation_metrics_final |\n",
-      "+--------------------------+\n",
-      "|      0.834100008011      |\n",
-      "+--------------------------+\n"
-     ]
-    }
-   ],
-   "source": [
-    "hp = Hyperband_diagonal(get_params, try_params )\n",
-    "results = hp.run()"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"plot\"></a>\n",
-    "# 5. Review and plot results"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 29,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>mst_key</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "        <th>model_arch_table</th>\n",
-       "        <th>num_iterations</th>\n",
-       "        <th>start_training_time</th>\n",
-       "        <th>end_training_time</th>\n",
-       "        <th>s</th>\n",
-       "        <th>i</th>\n",
-       "        <th>run_id</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>45</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='sgd(lr=0.004501919010538727,momentum=0.9002808952996391)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=256,epochs=5</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>2159.70019531</td>\n",
-       "        <td>[121.955986022949, 245.619317054749, 368.365077972412, 490.415205955505, 614.768485069275, 737.048167943954, 860.508330106735, 984.307431936264, 1106.31793498993, 1229.54079914093, 1352.66811394691, 1477.57317709923, 1599.99458003044, 1723.35215711594, 1847.86346912384, 1971.57312297821, 2096.37913298607, 2221.54790210724, 2346.08665895462, 2470.83494997025, 2595.6411960125, 2722.25887513161, 2846.48335313797, 2971.13271403313, 3097.49445009232, 3222.44972395897, 3348.5662779808]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.941940009594</td>\n",
-       "        <td>0.169452220201</td>\n",
-       "        <td>[0.574479997158051, 0.658760011196136, 0.695840001106262, 0.72733998298645, 0.733219981193542, 0.771200001239777, 0.778680026531219, 0.808700025081635, 0.809000015258789, 0.818579971790314, 0.835739970207214, 0.84799998998642, 0.853200018405914, 0.858900010585785, 0.872919976711273, 0.878780007362366, 0.88808000087738, 0.880240023136139, 0.894320011138916, 0.903779983520508, 0.912299990653992, 0.908439993858337, 0.919539988040924, 0.924639999866486, 0.929180026054382, 0.9375, 0.941940009593964]</td>\n",
-       "        <td>[1.19219434261322, 0.959131419658661, 0.861107409000397, 0.770956337451935, 0.747268915176392, 0.64410811662674, 0.628470838069916, 0.539423823356628, 0.541868448257446, 0.514527797698975, 0.469026476144791, 0.432008743286133, 0.416983753442764, 0.402583330869675, 0.363078087568283, 0.346161216497421, 0.317243546247482, 0.340911239385605, 0.304346263408661, 0.274338334798813, 0.253901869058609, 0.262585163116455, 0.231020957231522, 0.218931555747986, 0.206650838255882, 0.184870630502701, 0.169452220201492]</td>\n",
-       "        <td>0.816399991512</td>\n",
-       "        <td>0.580716967583</td>\n",
-       "        <td>[0.565699994564056, 0.641200006008148, 0.674899995326996, 0.704500019550323, 0.708000004291534, 0.740499973297119, 0.739799976348877, 0.766499996185303, 0.762099981307983, 0.76690000295639, 0.780900001525879, 0.785000026226044, 0.785300016403198, 0.79009997844696, 0.79449999332428, 0.795799970626831, 0.802600026130676, 0.792599976062775, 0.798399984836578, 0.807299971580505, 0.810500025749207, 0.801699995994568, 0.805400013923645, 0.811600029468536, 0.810100018978119, 0.813899993896484, 0.816399991512299]</td>\n",
-       "        <td>[1.20952260494232, 1.00138294696808, 0.919946014881134, 0.846988558769226, 0.835236310958862, 0.748137712478638, 0.745132148265839, 0.670836567878723, 0.688502311706543, 0.673530399799347, 0.646275579929352, 0.626095473766327, 0.629233837127686, 0.623023450374603, 0.601795375347137, 0.603216171264648, 0.587353229522705, 0.635767936706543, 0.61867493391037, 0.594616591930389, 0.586753845214844, 0.60888147354126, 0.601007521152496, 0.593143999576569, 0.601291477680206, 0.583372294902802, 0.580716967582703]</td>\n",
-       "        <td>model_arch_table_cifar10</td>\n",
-       "        <td>27</td>\n",
-       "        <td>2020-01-23 21:12:04.749779</td>\n",
-       "        <td>2020-01-23 22:07:53.819497</td>\n",
-       "        <td>0</td>\n",
-       "        <td>3</td>\n",
-       "        <td>65</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(45, 2, u\"loss='categorical_crossentropy',optimizer='sgd(lr=0.004501919010538727,momentum=0.9002808952996391)',metrics=['accuracy']\", u'batch_size=256,epochs=5', u'madlib_keras', 2159.70019531, [121.955986022949, 245.619317054749, 368.365077972412, 490.415205955505, 614.768485069275, 737.048167943954, 860.508330106735, 984.307431936264, 1106.31793498993, 1229.54079914093, 1352.66811394691, 1477.57317709923, 1599.99458003044, 1723.35215711594, 1847.86346912384, 1971.57312297821, 2096.37913298607, 2221.54790210724, 2346.08665895462, 2470.83494997025, 2595.6411960125, 2722.25887513161, 2846.48335313797, 2971.13271403313, 3097.49445009232, 3222.44972395897, 3348.5662779808], [u'accuracy'], 0.941940009594, 0.169452220201, [0.574479997158051, 0.658760011196136, 0.695840001106262, 0.72733998298645, 0.733219981193542, 0.771200001239777, 0.778680026531219, 0.808700025081635, 0.809000015258789, 0.818579971790314, 0.835739970207214, 0.84799998998642, 0.853200018405914, 0.858900010585785, 0.872919976711273, 0.878780007362366, 0.88808000087738, 0.880240023136139, 0.894320011138916, 0.903779983520508, 0.912299990653992, 0.908439993858337, 0.919539988040924, 0.924639999866486, 0.929180026054382, 0.9375, 0.941940009593964], [1.19219434261322, 0.959131419658661, 0.861107409000397, 0.770956337451935, 0.747268915176392, 0.64410811662674, 0.628470838069916, 0.539423823356628, 0.541868448257446, 0.514527797698975, 0.469026476144791, 0.432008743286133, 0.416983753442764, 0.402583330869675, 0.363078087568283, 0.346161216497421, 0.317243546247482, 0.340911239385605, 0.304346263408661, 0.274338334798813, 0.253901869058609, 0.262585163116455, 0.231020957231522, 0.218931555747986, 0.206650838255882, 0.184870630502701, 0.169452220201492], 0.816399991512, 0.580716967583, [0.565699994564056, 0.641200006008148, 0.674899995326996, 0.704500019550323, 0.708000004291534, 0.740499973297119, 0.739799976348877, 0.766499996185303, 0.762099981307983, 0.76690000295639, 0.780900001525879, 0.785000026226044, 0.785300016403198, 0.79009997844696, 0.79449999332428, 0.795799970626831, 0.802600026130676, 0.792599976062775, 0.798399984836578, 0.807299971580505, 0.810500025749207, 0.801699995994568, 0.805400013923645, 0.811600029468536, 0.810100018978119, 0.813899993896484, 0.816399991512299], [1.20952260494232, 1.00138294696808, 0.919946014881134, 0.846988558769226, 0.835236310958862, 0.748137712478638, 0.745132148265839, 0.670836567878723, 0.688502311706543, 0.673530399799347, 0.646275579929352, 0.626095473766327, 0.629233837127686, 0.623023450374603, 0.601795375347137, 0.603216171264648, 0.587353229522705, 0.635767936706543, 0.61867493391037, 0.594616591930389, 0.586753845214844, 0.60888147354126, 0.601007521152496, 0.593143999576569, 0.601291477680206, 0.583372294902802, 0.580716967582703], u'model_arch_table_cifar10', 27, datetime.datetime(2020, 1, 23, 21, 12, 4, 749779), datetime.datetime(2020, 1, 23, 22, 7, 53, 819497), 0, 3, 65)]"
-      ]
-     },
-     "execution_count": 29,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql SELECT * FROM $results_table ORDER BY validation_loss_final ASC LIMIT 1;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 24,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "%matplotlib notebook\n",
-    "import matplotlib.pyplot as plt\n",
-    "from matplotlib.ticker import MaxNLocator\n",
-    "from collections import defaultdict\n",
-    "import pandas as pd\n",
-    "import seaborn as sns\n",
-    "sns.set_palette(sns.color_palette(\"hls\", 20))\n",
-    "plt.rcParams.update({'font.size': 12})\n",
-    "pd.set_option('display.max_colwidth', -1)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Training dataset"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 26,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "65 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "application/javascript": [
-       "/* Put everything inside the global mpl namespace */\n",
-       "window.mpl = {};\n",
-       "\n",
-       "\n",
-       "mpl.get_websocket_type = function() {\n",
-       "    if (typeof(WebSocket) !== 'undefined') {\n",
-       "        return WebSocket;\n",
-       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
-       "        return MozWebSocket;\n",
-       "    } else {\n",
-       "        alert('Your browser does not have WebSocket support.' +\n",
-       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
-       "              'Firefox 4 and 5 are also supported but you ' +\n",
-       "              'have to enable WebSockets in about:config.');\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
-       "    this.id = figure_id;\n",
-       "\n",
-       "    this.ws = websocket;\n",
-       "\n",
-       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
-       "\n",
-       "    if (!this.supports_binary) {\n",
-       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
-       "        if (warnings) {\n",
-       "            warnings.style.display = 'block';\n",
-       "            warnings.textContent = (\n",
-       "                \"This browser does not support binary websocket messages. \" +\n",
-       "                    \"Performance may be slow.\");\n",
-       "        }\n",
-       "    }\n",
-       "\n",
-       "    this.imageObj = new Image();\n",
-       "\n",
-       "    this.context = undefined;\n",
-       "    this.message = undefined;\n",
-       "    this.canvas = undefined;\n",
-       "    this.rubberband_canvas = undefined;\n",
-       "    this.rubberband_context = undefined;\n",
-       "    this.format_dropdown = undefined;\n",
-       "\n",
-       "    this.image_mode = 'full';\n",
-       "\n",
-       "    this.root = $('<div/>');\n",
-       "    this._root_extra_style(this.root)\n",
-       "    this.root.attr('style', 'display: inline-block');\n",
-       "\n",
-       "    $(parent_element).append(this.root);\n",
-       "\n",
-       "    this._init_header(this);\n",
-       "    this._init_canvas(this);\n",
-       "    this._init_toolbar(this);\n",
-       "\n",
-       "    var fig = this;\n",
-       "\n",
-       "    this.waiting = false;\n",
-       "\n",
-       "    this.ws.onopen =  function () {\n",
-       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
-       "            fig.send_message(\"send_image_mode\", {});\n",
-       "            if (mpl.ratio != 1) {\n",
-       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
-       "            }\n",
-       "            fig.send_message(\"refresh\", {});\n",
-       "        }\n",
-       "\n",
-       "    this.imageObj.onload = function() {\n",
-       "            if (fig.image_mode == 'full') {\n",
-       "                // Full images could contain transparency (where diff images\n",
-       "                // almost always do), so we need to clear the canvas so that\n",
-       "                // there is no ghosting.\n",
-       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
-       "            }\n",
-       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
-       "        };\n",
-       "\n",
-       "    this.imageObj.onunload = function() {\n",
-       "        fig.ws.close();\n",
-       "    }\n",
-       "\n",
-       "    this.ws.onmessage = this._make_on_message_function(this);\n",
-       "\n",
-       "    this.ondownload = ondownload;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_header = function() {\n",
-       "    var titlebar = $(\n",
-       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
-       "        'ui-helper-clearfix\"/>');\n",
-       "    var titletext = $(\n",
-       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
-       "        'text-align: center; padding: 3px;\"/>');\n",
-       "    titlebar.append(titletext)\n",
-       "    this.root.append(titlebar);\n",
-       "    this.header = titletext[0];\n",
-       "}\n",
-       "\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
-       "\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
-       "\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_canvas = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var canvas_div = $('<div/>');\n",
-       "\n",
-       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
-       "\n",
-       "    function canvas_keyboard_event(event) {\n",
-       "        return fig.key_event(event, event['data']);\n",
-       "    }\n",
-       "\n",
-       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
-       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
-       "    this.canvas_div = canvas_div\n",
-       "    this._canvas_extra_style(canvas_div)\n",
-       "    this.root.append(canvas_div);\n",
-       "\n",
-       "    var canvas = $('<canvas/>');\n",
-       "    canvas.addClass('mpl-canvas');\n",
-       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
-       "\n",
-       "    this.canvas = canvas[0];\n",
-       "    this.context = canvas[0].getContext(\"2d\");\n",
-       "\n",
-       "    var backingStore = this.context.backingStorePixelRatio ||\n",
-       "\tthis.context.webkitBackingStorePixelRatio ||\n",
-       "\tthis.context.mozBackingStorePixelRatio ||\n",
-       "\tthis.context.msBackingStorePixelRatio ||\n",
-       "\tthis.context.oBackingStorePixelRatio ||\n",
-       "\tthis.context.backingStorePixelRatio || 1;\n",
-       "\n",
-       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
-       "\n",
-       "    var rubberband = $('<canvas/>');\n",
-       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
-       "\n",
-       "    var pass_mouse_events = true;\n",
-       "\n",
-       "    canvas_div.resizable({\n",
-       "        start: function(event, ui) {\n",
-       "            pass_mouse_events = false;\n",
-       "        },\n",
-       "        resize: function(event, ui) {\n",
-       "            fig.request_resize(ui.size.width, ui.size.height);\n",
-       "        },\n",
-       "        stop: function(event, ui) {\n",
-       "            pass_mouse_events = true;\n",
-       "            fig.request_resize(ui.size.width, ui.size.height);\n",
-       "        },\n",
-       "    });\n",
-       "\n",
-       "    function mouse_event_fn(event) {\n",
-       "        if (pass_mouse_events)\n",
-       "            return fig.mouse_event(event, event['data']);\n",
-       "    }\n",
-       "\n",
-       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
-       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
-       "    // Throttle sequential mouse events to 1 every 20ms.\n",
-       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
-       "\n",
-       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
-       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
-       "\n",
-       "    canvas_div.on(\"wheel\", function (event) {\n",
-       "        event = event.originalEvent;\n",
-       "        event['data'] = 'scroll'\n",
-       "        if (event.deltaY < 0) {\n",
-       "            event.step = 1;\n",
-       "        } else {\n",
-       "            event.step = -1;\n",
-       "        }\n",
-       "        mouse_event_fn(event);\n",
-       "    });\n",
-       "\n",
-       "    canvas_div.append(canvas);\n",
-       "    canvas_div.append(rubberband);\n",
-       "\n",
-       "    this.rubberband = rubberband;\n",
-       "    this.rubberband_canvas = rubberband[0];\n",
-       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
-       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
-       "\n",
-       "    this._resize_canvas = function(width, height) {\n",
-       "        // Keep the size of the canvas, canvas container, and rubber band\n",
-       "        // canvas in synch.\n",
-       "        canvas_div.css('width', width)\n",
-       "        canvas_div.css('height', height)\n",
-       "\n",
-       "        canvas.attr('width', width * mpl.ratio);\n",
-       "        canvas.attr('height', height * mpl.ratio);\n",
-       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
-       "\n",
-       "        rubberband.attr('width', width);\n",
-       "        rubberband.attr('height', height);\n",
-       "    }\n",
-       "\n",
-       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
-       "    // upon first draw.\n",
-       "    this._resize_canvas(600, 600);\n",
-       "\n",
-       "    // Disable right mouse context menu.\n",
-       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
-       "        return false;\n",
-       "    });\n",
-       "\n",
-       "    function set_focus () {\n",
-       "        canvas.focus();\n",
-       "        canvas_div.focus();\n",
-       "    }\n",
-       "\n",
-       "    window.setTimeout(set_focus, 100);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_toolbar = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var nav_element = $('<div/>')\n",
-       "    nav_element.attr('style', 'width: 100%');\n",
-       "    this.root.append(nav_element);\n",
-       "\n",
-       "    // Define a callback function for later on.\n",
-       "    function toolbar_event(event) {\n",
-       "        return fig.toolbar_button_onclick(event['data']);\n",
-       "    }\n",
-       "    function toolbar_mouse_event(event) {\n",
-       "        return fig.toolbar_button_onmouseover(event['data']);\n",
-       "    }\n",
-       "\n",
-       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
-       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
-       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
-       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
-       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
-       "\n",
-       "        if (!name) {\n",
-       "            // put a spacer in here.\n",
-       "            continue;\n",
-       "        }\n",
-       "        var button = $('<button/>');\n",
-       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
-       "                        'ui-button-icon-only');\n",
-       "        button.attr('role', 'button');\n",
-       "        button.attr('aria-disabled', 'false');\n",
-       "        button.click(method_name, toolbar_event);\n",
-       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
-       "\n",
-       "        var icon_img = $('<span/>');\n",
-       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
-       "        icon_img.addClass(image);\n",
-       "        icon_img.addClass('ui-corner-all');\n",
-       "\n",
-       "        var tooltip_span = $('<span/>');\n",
-       "        tooltip_span.addClass('ui-button-text');\n",
-       "        tooltip_span.html(tooltip);\n",
-       "\n",
-       "        button.append(icon_img);\n",
-       "        button.append(tooltip_span);\n",
-       "\n",
-       "        nav_element.append(button);\n",
-       "    }\n",
-       "\n",
-       "    var fmt_picker_span = $('<span/>');\n",
-       "\n",
-       "    var fmt_picker = $('<select/>');\n",
-       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
-       "    fmt_picker_span.append(fmt_picker);\n",
-       "    nav_element.append(fmt_picker_span);\n",
-       "    this.format_dropdown = fmt_picker[0];\n",
-       "\n",
-       "    for (var ind in mpl.extensions) {\n",
-       "        var fmt = mpl.extensions[ind];\n",
-       "        var option = $(\n",
-       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
-       "        fmt_picker.append(option)\n",
-       "    }\n",
-       "\n",
-       "    // Add hover states to the ui-buttons\n",
-       "    $( \".ui-button\" ).hover(\n",
-       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
-       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
-       "    );\n",
-       "\n",
-       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
-       "    nav_element.append(status_bar);\n",
-       "    this.message = status_bar[0];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
-       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
-       "    // which will in turn request a refresh of the image.\n",
-       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.send_message = function(type, properties) {\n",
-       "    properties['type'] = type;\n",
-       "    properties['figure_id'] = this.id;\n",
-       "    this.ws.send(JSON.stringify(properties));\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.send_draw_message = function() {\n",
-       "    if (!this.waiting) {\n",
-       "        this.waiting = true;\n",
-       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
-       "    var format_dropdown = fig.format_dropdown;\n",
-       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
-       "    fig.ondownload(fig, format);\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
-       "    var size = msg['size'];\n",
-       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
-       "        fig._resize_canvas(size[0], size[1]);\n",
-       "        fig.send_message(\"refresh\", {});\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
-       "    var x0 = msg['x0'] / mpl.ratio;\n",
-       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
-       "    var x1 = msg['x1'] / mpl.ratio;\n",
-       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
-       "    x0 = Math.floor(x0) + 0.5;\n",
-       "    y0 = Math.floor(y0) + 0.5;\n",
-       "    x1 = Math.floor(x1) + 0.5;\n",
-       "    y1 = Math.floor(y1) + 0.5;\n",
-       "    var min_x = Math.min(x0, x1);\n",
-       "    var min_y = Math.min(y0, y1);\n",
-       "    var width = Math.abs(x1 - x0);\n",
-       "    var height = Math.abs(y1 - y0);\n",
-       "\n",
-       "    fig.rubberband_context.clearRect(\n",
-       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
-       "\n",
-       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
-       "    // Updates the figure title.\n",
-       "    fig.header.textContent = msg['label'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
-       "    var cursor = msg['cursor'];\n",
-       "    switch(cursor)\n",
-       "    {\n",
-       "    case 0:\n",
-       "        cursor = 'pointer';\n",
-       "        break;\n",
-       "    case 1:\n",
-       "        cursor = 'default';\n",
-       "        break;\n",
-       "    case 2:\n",
-       "        cursor = 'crosshair';\n",
-       "        break;\n",
-       "    case 3:\n",
-       "        cursor = 'move';\n",
-       "        break;\n",
-       "    }\n",
-       "    fig.rubberband_canvas.style.cursor = cursor;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
-       "    fig.message.textContent = msg['message'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
-       "    // Request the server to send over a new figure.\n",
-       "    fig.send_draw_message();\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
-       "    fig.image_mode = msg['mode'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.updated_canvas_event = function() {\n",
-       "    // Called whenever the canvas gets updated.\n",
-       "    this.send_message(\"ack\", {});\n",
-       "}\n",
-       "\n",
-       "// A function to construct a web socket function for onmessage handling.\n",
-       "// Called in the figure constructor.\n",
-       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
-       "    return function socket_on_message(evt) {\n",
-       "        if (evt.data instanceof Blob) {\n",
-       "            /* FIXME: We get \"Resource interpreted as Image but\n",
-       "             * transferred with MIME type text/plain:\" errors on\n",
-       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
-       "             * to be part of the websocket stream */\n",
-       "            evt.data.type = \"image/png\";\n",
-       "\n",
-       "            /* Free the memory for the previous frames */\n",
-       "            if (fig.imageObj.src) {\n",
-       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
-       "                    fig.imageObj.src);\n",
-       "            }\n",
-       "\n",
-       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
-       "                evt.data);\n",
-       "            fig.updated_canvas_event();\n",
-       "            fig.waiting = false;\n",
-       "            return;\n",
-       "        }\n",
-       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
-       "            fig.imageObj.src = evt.data;\n",
-       "            fig.updated_canvas_event();\n",
-       "            fig.waiting = false;\n",
-       "            return;\n",
-       "        }\n",
-       "\n",
-       "        var msg = JSON.parse(evt.data);\n",
-       "        var msg_type = msg['type'];\n",
-       "\n",
-       "        // Call the  \"handle_{type}\" callback, which takes\n",
-       "        // the figure and JSON message as its only arguments.\n",
-       "        try {\n",
-       "            var callback = fig[\"handle_\" + msg_type];\n",
-       "        } catch (e) {\n",
-       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
-       "            return;\n",
-       "        }\n",
-       "\n",
-       "        if (callback) {\n",
-       "            try {\n",
-       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
-       "                callback(fig, msg);\n",
-       "            } catch (e) {\n",
-       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
-       "            }\n",
-       "        }\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
-       "mpl.findpos = function(e) {\n",
-       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
-       "    var targ;\n",
-       "    if (!e)\n",
-       "        e = window.event;\n",
-       "    if (e.target)\n",
-       "        targ = e.target;\n",
-       "    else if (e.srcElement)\n",
-       "        targ = e.srcElement;\n",
-       "    if (targ.nodeType == 3) // defeat Safari bug\n",
-       "        targ = targ.parentNode;\n",
-       "\n",
-       "    // jQuery normalizes the pageX and pageY\n",
-       "    // pageX,Y are the mouse positions relative to the document\n",
-       "    // offset() returns the position of the element relative to the document\n",
-       "    var x = e.pageX - $(targ).offset().left;\n",
-       "    var y = e.pageY - $(targ).offset().top;\n",
-       "\n",
-       "    return {\"x\": x, \"y\": y};\n",
-       "};\n",
-       "\n",
-       "/*\n",
-       " * return a copy of an object with only non-object keys\n",
-       " * we need this to avoid circular references\n",
-       " * http://stackoverflow.com/a/24161582/3208463\n",
-       " */\n",
-       "function simpleKeys (original) {\n",
-       "  return Object.keys(original).reduce(function (obj, key) {\n",
-       "    if (typeof original[key] !== 'object')\n",
-       "        obj[key] = original[key]\n",
-       "    return obj;\n",
-       "  }, {});\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
-       "    var canvas_pos = mpl.findpos(event)\n",
-       "\n",
-       "    if (name === 'button_press')\n",
-       "    {\n",
-       "        this.canvas.focus();\n",
-       "        this.canvas_div.focus();\n",
-       "    }\n",
-       "\n",
-       "    var x = canvas_pos.x * mpl.ratio;\n",
-       "    var y = canvas_pos.y * mpl.ratio;\n",
-       "\n",
-       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
-       "                             step: event.step,\n",
-       "                             guiEvent: simpleKeys(event)});\n",
-       "\n",
-       "    /* This prevents the web browser from automatically changing to\n",
-       "     * the text insertion cursor when the button is pressed.  We want\n",
-       "     * to control all of the cursor setting manually through the\n",
-       "     * 'cursor' event from matplotlib */\n",
-       "    event.preventDefault();\n",
-       "    return false;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
-       "    // Handle any extra behaviour associated with a key event\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.key_event = function(event, name) {\n",
-       "\n",
-       "    // Prevent repeat events\n",
-       "    if (name == 'key_press')\n",
-       "    {\n",
-       "        if (event.which === this._key)\n",
-       "            return;\n",
-       "        else\n",
-       "            this._key = event.which;\n",
-       "    }\n",
-       "    if (name == 'key_release')\n",
-       "        this._key = null;\n",
-       "\n",
-       "    var value = '';\n",
-       "    if (event.ctrlKey && event.which != 17)\n",
-       "        value += \"ctrl+\";\n",
-       "    if (event.altKey && event.which != 18)\n",
-       "        value += \"alt+\";\n",
-       "    if (event.shiftKey && event.which != 16)\n",
-       "        value += \"shift+\";\n",
-       "\n",
-       "    value += 'k';\n",
-       "    value += event.which.toString();\n",
-       "\n",
-       "    this._key_event_extra(event, name);\n",
-       "\n",
-       "    this.send_message(name, {key: value,\n",
-       "                             guiEvent: simpleKeys(event)});\n",
-       "    return false;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
-       "    if (name == 'download') {\n",
-       "        this.handle_save(this, null);\n",
-       "    } else {\n",
-       "        this.send_message(\"toolbar_button\", {name: name});\n",
-       "    }\n",
-       "};\n",
-       "\n",
-       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
-       "    this.message.textContent = tooltip;\n",
-       "};\n",
-       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
-       "\n",
-       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
-       "\n",
-       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
-       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
-       "    // object with the appropriate methods. Currently this is a non binary\n",
-       "    // socket, so there is still some room for performance tuning.\n",
-       "    var ws = {};\n",
-       "\n",
-       "    ws.close = function() {\n",
-       "        comm.close()\n",
-       "    };\n",
-       "    ws.send = function(m) {\n",
-       "        //console.log('sending', m);\n",
-       "        comm.send(m);\n",
-       "    };\n",
-       "    // Register the callback with on_msg.\n",
-       "    comm.on_msg(function(msg) {\n",
-       "        //console.log('receiving', msg['content']['data'], msg);\n",
-       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
-       "        ws.onmessage(msg['content']['data'])\n",
-       "    });\n",
-       "    return ws;\n",
-       "}\n",
-       "\n",
-       "mpl.mpl_figure_comm = function(comm, msg) {\n",
-       "    // This is the function which gets called when the mpl process\n",
-       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
-       "\n",
-       "    var id = msg.content.data.id;\n",
-       "    // Get hold of the div created by the display call when the Comm\n",
-       "    // socket was opened in Python.\n",
-       "    var element = $(\"#\" + id);\n",
-       "    var ws_proxy = comm_websocket_adapter(comm)\n",
-       "\n",
-       "    function ondownload(figure, format) {\n",
-       "        window.open(figure.imageObj.src);\n",
-       "    }\n",
-       "\n",
-       "    var fig = new mpl.figure(id, ws_proxy,\n",
-       "                           ondownload,\n",
-       "                           element.get(0));\n",
-       "\n",
-       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
-       "    // web socket which is closed, not our websocket->open comm proxy.\n",
-       "    ws_proxy.onopen();\n",
-       "\n",
-       "    fig.parent_element = element.get(0);\n",
-       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
-       "    if (!fig.cell_info) {\n",
-       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
-       "        return;\n",
-       "    }\n",
-       "\n",
-       "    var output_index = fig.cell_info[2]\n",
-       "    var cell = fig.cell_info[0];\n",
-       "\n",
-       "};\n",
-       "\n",
-       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
-       "    var width = fig.canvas.width/mpl.ratio\n",
-       "    fig.root.unbind('remove')\n",
-       "\n",
-       "    // Update the output cell to use the data from the current canvas.\n",
-       "    fig.push_to_output();\n",
-       "    var dataURL = fig.canvas.toDataURL();\n",
-       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
-       "    // the notebook keyboard shortcuts fail.\n",
-       "    IPython.keyboard_manager.enable()\n",
-       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
-       "    fig.close_ws(fig, msg);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
-       "    fig.send_message('closing', msg);\n",
-       "    // fig.ws.close()\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
-       "    // Turn the data on the canvas into data in the output cell.\n",
-       "    var width = this.canvas.width/mpl.ratio\n",
-       "    var dataURL = this.canvas.toDataURL();\n",
-       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.updated_canvas_event = function() {\n",
-       "    // Tell IPython that the notebook contents must change.\n",
-       "    IPython.notebook.set_dirty(true);\n",
-       "    this.send_message(\"ack\", {});\n",
-       "    var fig = this;\n",
-       "    // Wait a second, then push the new image to the DOM so\n",
-       "    // that it is saved nicely (might be nice to debounce this).\n",
-       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_toolbar = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var nav_element = $('<div/>')\n",
-       "    nav_element.attr('style', 'width: 100%');\n",
-       "    this.root.append(nav_element);\n",
-       "\n",
-       "    // Define a callback function for later on.\n",
-       "    function toolbar_event(event) {\n",
-       "        return fig.toolbar_button_onclick(event['data']);\n",
-       "    }\n",
-       "    function toolbar_mouse_event(event) {\n",
-       "        return fig.toolbar_button_onmouseover(event['data']);\n",
-       "    }\n",
-       "\n",
-       "    for(var toolbar_ind in mpl.toolbar_items){\n",
-       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
-       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
-       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
-       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
-       "\n",
-       "        if (!name) { continue; };\n",
-       "\n",
-       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
-       "        button.click(method_name, toolbar_event);\n",
-       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
-       "        nav_element.append(button);\n",
-       "    }\n",
-       "\n",
-       "    // Add the status bar.\n",
-       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
-       "    nav_element.append(status_bar);\n",
-       "    this.message = status_bar[0];\n",
-       "\n",
-       "    // Add the close button to the window.\n",
-       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
-       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
-       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
-       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
-       "    buttongrp.append(button);\n",
-       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
-       "    titlebar.prepend(buttongrp);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._root_extra_style = function(el){\n",
-       "    var fig = this\n",
-       "    el.on(\"remove\", function(){\n",
-       "\tfig.close_ws(fig, {});\n",
-       "    });\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
-       "    // this is important to make the div 'focusable\n",
-       "    el.attr('tabindex', 0)\n",
-       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
-       "    // off when our div gets focus\n",
-       "\n",
-       "    // location in version 3\n",
-       "    if (IPython.notebook.keyboard_manager) {\n",
-       "        IPython.notebook.keyboard_manager.register_events(el);\n",
-       "    }\n",
-       "    else {\n",
-       "        // location in version 2\n",
-       "        IPython.keyboard_manager.register_events(el);\n",
-       "    }\n",
-       "\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
-       "    var manager = IPython.notebook.keyboard_manager;\n",
-       "    if (!manager)\n",
-       "        manager = IPython.keyboard_manager;\n",
-       "\n",
-       "    // Check for shift+enter\n",
-       "    if (event.shiftKey && event.which == 13) {\n",
-       "        this.canvas_div.blur();\n",
-       "        event.shiftKey = false;\n",
-       "        // Send a \"J\" for go to next cell\n",
-       "        event.which = 74;\n",
-       "        event.keyCode = 74;\n",
-       "        manager.command_mode();\n",
-       "        manager.handle_keydown(event);\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
-       "    fig.ondownload(fig, null);\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.find_output_cell = function(html_output) {\n",
-       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
-       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
-       "    // IPython event is triggered only after the cells have been serialised, which for\n",
-       "    // our purposes (turning an active figure into a static one), is too late.\n",
-       "    var cells = IPython.notebook.get_cells();\n",
-       "    var ncells = cells.length;\n",
-       "    for (var i=0; i<ncells; i++) {\n",
-       "        var cell = cells[i];\n",
-       "        if (cell.cell_type === 'code'){\n",
-       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
-       "                var data = cell.output_area.outputs[j];\n",
-       "                if (data.data) {\n",
-       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
-       "                    data = data.data;\n",
-       "                }\n",
-       "                if (data['text/html'] == html_output) {\n",
-       "                    return [cell, data, j];\n",
-       "                }\n",
-       "            }\n",
-       "        }\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "// Register the function which deals with the matplotlib target/channel.\n",
-       "// The kernel may be null if the page has been refreshed.\n",
-       "if (IPython.notebook.kernel != null) {\n",
-       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
-       "}\n"
-      ],
-      "text/plain": [
-       "<IPython.core.display.Javascript object>"
-      ]
-     },
-     "metadata": {},
-     "output_type": "display_data"
-    },
-    {
-     "data": {
-      "text/html": [
-       "<img src=\"\" width=\"1000\">"
-      ],
-      "text/plain": [
-       "<IPython.core.display.HTML object>"
-      ]
-     },
-     "metadata": {},
-     "output_type": "display_data"
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    }
-   ],
-   "source": [
-    "#df_results = %sql SELECT * FROM $results_table ORDER BY run_id;\n",
-    "df_results = %sql SELECT * FROM $results_table ORDER BY training_loss_final ASC LIMIT 100;\n",
-    "df_results = df_results.DataFrame()\n",
-    "\n",
-    "#set up plots\n",
-    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
-    "fig.legend(ncol=4)\n",
-    "fig.tight_layout()\n",
-    "\n",
-    "ax_metric = axs[0]\n",
-    "ax_loss = axs[1]\n",
-    "\n",
-    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
-    "ax_metric.set_xlabel('Iteration')\n",
-    "ax_metric.set_ylabel('Metric')\n",
-    "ax_metric.set_title('Training metric curve')\n",
-    "\n",
-    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
-    "ax_loss.set_xlabel('Iteration')\n",
-    "ax_loss.set_ylabel('Loss')\n",
-    "ax_loss.set_title('Training loss curve')\n",
-    "\n",
-    "for run_id in df_results['run_id']:\n",
-    "    df_output_info = %sql SELECT training_metrics,training_loss FROM $results_table WHERE run_id = $run_id\n",
-    "    df_output_info = df_output_info.DataFrame()\n",
-    "    training_metrics = df_output_info['training_metrics'][0]\n",
-    "    training_loss = df_output_info['training_loss'][0]\n",
-    "    X = range(len(training_metrics))\n",
-    "    \n",
-    "    ax_metric.plot(X, training_metrics, label=run_id, marker='o')\n",
-    "    ax_loss.plot(X, training_loss, label=run_id, marker='o')\n",
-    "\n",
-    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Validation dataset"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 27,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "65 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "application/javascript": [
-       "/* Put everything inside the global mpl namespace */\n",
-       "window.mpl = {};\n",
-       "\n",
-       "\n",
-       "mpl.get_websocket_type = function() {\n",
-       "    if (typeof(WebSocket) !== 'undefined') {\n",
-       "        return WebSocket;\n",
-       "    } else if (typeof(MozWebSocket) !== 'undefined') {\n",
-       "        return MozWebSocket;\n",
-       "    } else {\n",
-       "        alert('Your browser does not have WebSocket support.' +\n",
-       "              'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
-       "              'Firefox 4 and 5 are also supported but you ' +\n",
-       "              'have to enable WebSockets in about:config.');\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
-       "    this.id = figure_id;\n",
-       "\n",
-       "    this.ws = websocket;\n",
-       "\n",
-       "    this.supports_binary = (this.ws.binaryType != undefined);\n",
-       "\n",
-       "    if (!this.supports_binary) {\n",
-       "        var warnings = document.getElementById(\"mpl-warnings\");\n",
-       "        if (warnings) {\n",
-       "            warnings.style.display = 'block';\n",
-       "            warnings.textContent = (\n",
-       "                \"This browser does not support binary websocket messages. \" +\n",
-       "                    \"Performance may be slow.\");\n",
-       "        }\n",
-       "    }\n",
-       "\n",
-       "    this.imageObj = new Image();\n",
-       "\n",
-       "    this.context = undefined;\n",
-       "    this.message = undefined;\n",
-       "    this.canvas = undefined;\n",
-       "    this.rubberband_canvas = undefined;\n",
-       "    this.rubberband_context = undefined;\n",
-       "    this.format_dropdown = undefined;\n",
-       "\n",
-       "    this.image_mode = 'full';\n",
-       "\n",
-       "    this.root = $('<div/>');\n",
-       "    this._root_extra_style(this.root)\n",
-       "    this.root.attr('style', 'display: inline-block');\n",
-       "\n",
-       "    $(parent_element).append(this.root);\n",
-       "\n",
-       "    this._init_header(this);\n",
-       "    this._init_canvas(this);\n",
-       "    this._init_toolbar(this);\n",
-       "\n",
-       "    var fig = this;\n",
-       "\n",
-       "    this.waiting = false;\n",
-       "\n",
-       "    this.ws.onopen =  function () {\n",
-       "            fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
-       "            fig.send_message(\"send_image_mode\", {});\n",
-       "            if (mpl.ratio != 1) {\n",
-       "                fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
-       "            }\n",
-       "            fig.send_message(\"refresh\", {});\n",
-       "        }\n",
-       "\n",
-       "    this.imageObj.onload = function() {\n",
-       "            if (fig.image_mode == 'full') {\n",
-       "                // Full images could contain transparency (where diff images\n",
-       "                // almost always do), so we need to clear the canvas so that\n",
-       "                // there is no ghosting.\n",
-       "                fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
-       "            }\n",
-       "            fig.context.drawImage(fig.imageObj, 0, 0);\n",
-       "        };\n",
-       "\n",
-       "    this.imageObj.onunload = function() {\n",
-       "        fig.ws.close();\n",
-       "    }\n",
-       "\n",
-       "    this.ws.onmessage = this._make_on_message_function(this);\n",
-       "\n",
-       "    this.ondownload = ondownload;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_header = function() {\n",
-       "    var titlebar = $(\n",
-       "        '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
-       "        'ui-helper-clearfix\"/>');\n",
-       "    var titletext = $(\n",
-       "        '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
-       "        'text-align: center; padding: 3px;\"/>');\n",
-       "    titlebar.append(titletext)\n",
-       "    this.root.append(titlebar);\n",
-       "    this.header = titletext[0];\n",
-       "}\n",
-       "\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
-       "\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
-       "\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_canvas = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var canvas_div = $('<div/>');\n",
-       "\n",
-       "    canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
-       "\n",
-       "    function canvas_keyboard_event(event) {\n",
-       "        return fig.key_event(event, event['data']);\n",
-       "    }\n",
-       "\n",
-       "    canvas_div.keydown('key_press', canvas_keyboard_event);\n",
-       "    canvas_div.keyup('key_release', canvas_keyboard_event);\n",
-       "    this.canvas_div = canvas_div\n",
-       "    this._canvas_extra_style(canvas_div)\n",
-       "    this.root.append(canvas_div);\n",
-       "\n",
-       "    var canvas = $('<canvas/>');\n",
-       "    canvas.addClass('mpl-canvas');\n",
-       "    canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
-       "\n",
-       "    this.canvas = canvas[0];\n",
-       "    this.context = canvas[0].getContext(\"2d\");\n",
-       "\n",
-       "    var backingStore = this.context.backingStorePixelRatio ||\n",
-       "\tthis.context.webkitBackingStorePixelRatio ||\n",
-       "\tthis.context.mozBackingStorePixelRatio ||\n",
-       "\tthis.context.msBackingStorePixelRatio ||\n",
-       "\tthis.context.oBackingStorePixelRatio ||\n",
-       "\tthis.context.backingStorePixelRatio || 1;\n",
-       "\n",
-       "    mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
-       "\n",
-       "    var rubberband = $('<canvas/>');\n",
-       "    rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
-       "\n",
-       "    var pass_mouse_events = true;\n",
-       "\n",
-       "    canvas_div.resizable({\n",
-       "        start: function(event, ui) {\n",
-       "            pass_mouse_events = false;\n",
-       "        },\n",
-       "        resize: function(event, ui) {\n",
-       "            fig.request_resize(ui.size.width, ui.size.height);\n",
-       "        },\n",
-       "        stop: function(event, ui) {\n",
-       "            pass_mouse_events = true;\n",
-       "            fig.request_resize(ui.size.width, ui.size.height);\n",
-       "        },\n",
-       "    });\n",
-       "\n",
-       "    function mouse_event_fn(event) {\n",
-       "        if (pass_mouse_events)\n",
-       "            return fig.mouse_event(event, event['data']);\n",
-       "    }\n",
-       "\n",
-       "    rubberband.mousedown('button_press', mouse_event_fn);\n",
-       "    rubberband.mouseup('button_release', mouse_event_fn);\n",
-       "    // Throttle sequential mouse events to 1 every 20ms.\n",
-       "    rubberband.mousemove('motion_notify', mouse_event_fn);\n",
-       "\n",
-       "    rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
-       "    rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
-       "\n",
-       "    canvas_div.on(\"wheel\", function (event) {\n",
-       "        event = event.originalEvent;\n",
-       "        event['data'] = 'scroll'\n",
-       "        if (event.deltaY < 0) {\n",
-       "            event.step = 1;\n",
-       "        } else {\n",
-       "            event.step = -1;\n",
-       "        }\n",
-       "        mouse_event_fn(event);\n",
-       "    });\n",
-       "\n",
-       "    canvas_div.append(canvas);\n",
-       "    canvas_div.append(rubberband);\n",
-       "\n",
-       "    this.rubberband = rubberband;\n",
-       "    this.rubberband_canvas = rubberband[0];\n",
-       "    this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
-       "    this.rubberband_context.strokeStyle = \"#000000\";\n",
-       "\n",
-       "    this._resize_canvas = function(width, height) {\n",
-       "        // Keep the size of the canvas, canvas container, and rubber band\n",
-       "        // canvas in synch.\n",
-       "        canvas_div.css('width', width)\n",
-       "        canvas_div.css('height', height)\n",
-       "\n",
-       "        canvas.attr('width', width * mpl.ratio);\n",
-       "        canvas.attr('height', height * mpl.ratio);\n",
-       "        canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
-       "\n",
-       "        rubberband.attr('width', width);\n",
-       "        rubberband.attr('height', height);\n",
-       "    }\n",
-       "\n",
-       "    // Set the figure to an initial 600x600px, this will subsequently be updated\n",
-       "    // upon first draw.\n",
-       "    this._resize_canvas(600, 600);\n",
-       "\n",
-       "    // Disable right mouse context menu.\n",
-       "    $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
-       "        return false;\n",
-       "    });\n",
-       "\n",
-       "    function set_focus () {\n",
-       "        canvas.focus();\n",
-       "        canvas_div.focus();\n",
-       "    }\n",
-       "\n",
-       "    window.setTimeout(set_focus, 100);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_toolbar = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var nav_element = $('<div/>')\n",
-       "    nav_element.attr('style', 'width: 100%');\n",
-       "    this.root.append(nav_element);\n",
-       "\n",
-       "    // Define a callback function for later on.\n",
-       "    function toolbar_event(event) {\n",
-       "        return fig.toolbar_button_onclick(event['data']);\n",
-       "    }\n",
-       "    function toolbar_mouse_event(event) {\n",
-       "        return fig.toolbar_button_onmouseover(event['data']);\n",
-       "    }\n",
-       "\n",
-       "    for(var toolbar_ind in mpl.toolbar_items) {\n",
-       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
-       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
-       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
-       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
-       "\n",
-       "        if (!name) {\n",
-       "            // put a spacer in here.\n",
-       "            continue;\n",
-       "        }\n",
-       "        var button = $('<button/>');\n",
-       "        button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
-       "                        'ui-button-icon-only');\n",
-       "        button.attr('role', 'button');\n",
-       "        button.attr('aria-disabled', 'false');\n",
-       "        button.click(method_name, toolbar_event);\n",
-       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
-       "\n",
-       "        var icon_img = $('<span/>');\n",
-       "        icon_img.addClass('ui-button-icon-primary ui-icon');\n",
-       "        icon_img.addClass(image);\n",
-       "        icon_img.addClass('ui-corner-all');\n",
-       "\n",
-       "        var tooltip_span = $('<span/>');\n",
-       "        tooltip_span.addClass('ui-button-text');\n",
-       "        tooltip_span.html(tooltip);\n",
-       "\n",
-       "        button.append(icon_img);\n",
-       "        button.append(tooltip_span);\n",
-       "\n",
-       "        nav_element.append(button);\n",
-       "    }\n",
-       "\n",
-       "    var fmt_picker_span = $('<span/>');\n",
-       "\n",
-       "    var fmt_picker = $('<select/>');\n",
-       "    fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
-       "    fmt_picker_span.append(fmt_picker);\n",
-       "    nav_element.append(fmt_picker_span);\n",
-       "    this.format_dropdown = fmt_picker[0];\n",
-       "\n",
-       "    for (var ind in mpl.extensions) {\n",
-       "        var fmt = mpl.extensions[ind];\n",
-       "        var option = $(\n",
-       "            '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
-       "        fmt_picker.append(option)\n",
-       "    }\n",
-       "\n",
-       "    // Add hover states to the ui-buttons\n",
-       "    $( \".ui-button\" ).hover(\n",
-       "        function() { $(this).addClass(\"ui-state-hover\");},\n",
-       "        function() { $(this).removeClass(\"ui-state-hover\");}\n",
-       "    );\n",
-       "\n",
-       "    var status_bar = $('<span class=\"mpl-message\"/>');\n",
-       "    nav_element.append(status_bar);\n",
-       "    this.message = status_bar[0];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
-       "    // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
-       "    // which will in turn request a refresh of the image.\n",
-       "    this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.send_message = function(type, properties) {\n",
-       "    properties['type'] = type;\n",
-       "    properties['figure_id'] = this.id;\n",
-       "    this.ws.send(JSON.stringify(properties));\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.send_draw_message = function() {\n",
-       "    if (!this.waiting) {\n",
-       "        this.waiting = true;\n",
-       "        this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
-       "    var format_dropdown = fig.format_dropdown;\n",
-       "    var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
-       "    fig.ondownload(fig, format);\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
-       "    var size = msg['size'];\n",
-       "    if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
-       "        fig._resize_canvas(size[0], size[1]);\n",
-       "        fig.send_message(\"refresh\", {});\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
-       "    var x0 = msg['x0'] / mpl.ratio;\n",
-       "    var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
-       "    var x1 = msg['x1'] / mpl.ratio;\n",
-       "    var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
-       "    x0 = Math.floor(x0) + 0.5;\n",
-       "    y0 = Math.floor(y0) + 0.5;\n",
-       "    x1 = Math.floor(x1) + 0.5;\n",
-       "    y1 = Math.floor(y1) + 0.5;\n",
-       "    var min_x = Math.min(x0, x1);\n",
-       "    var min_y = Math.min(y0, y1);\n",
-       "    var width = Math.abs(x1 - x0);\n",
-       "    var height = Math.abs(y1 - y0);\n",
-       "\n",
-       "    fig.rubberband_context.clearRect(\n",
-       "        0, 0, fig.canvas.width, fig.canvas.height);\n",
-       "\n",
-       "    fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
-       "    // Updates the figure title.\n",
-       "    fig.header.textContent = msg['label'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
-       "    var cursor = msg['cursor'];\n",
-       "    switch(cursor)\n",
-       "    {\n",
-       "    case 0:\n",
-       "        cursor = 'pointer';\n",
-       "        break;\n",
-       "    case 1:\n",
-       "        cursor = 'default';\n",
-       "        break;\n",
-       "    case 2:\n",
-       "        cursor = 'crosshair';\n",
-       "        break;\n",
-       "    case 3:\n",
-       "        cursor = 'move';\n",
-       "        break;\n",
-       "    }\n",
-       "    fig.rubberband_canvas.style.cursor = cursor;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_message = function(fig, msg) {\n",
-       "    fig.message.textContent = msg['message'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
-       "    // Request the server to send over a new figure.\n",
-       "    fig.send_draw_message();\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
-       "    fig.image_mode = msg['mode'];\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.updated_canvas_event = function() {\n",
-       "    // Called whenever the canvas gets updated.\n",
-       "    this.send_message(\"ack\", {});\n",
-       "}\n",
-       "\n",
-       "// A function to construct a web socket function for onmessage handling.\n",
-       "// Called in the figure constructor.\n",
-       "mpl.figure.prototype._make_on_message_function = function(fig) {\n",
-       "    return function socket_on_message(evt) {\n",
-       "        if (evt.data instanceof Blob) {\n",
-       "            /* FIXME: We get \"Resource interpreted as Image but\n",
-       "             * transferred with MIME type text/plain:\" errors on\n",
-       "             * Chrome.  But how to set the MIME type?  It doesn't seem\n",
-       "             * to be part of the websocket stream */\n",
-       "            evt.data.type = \"image/png\";\n",
-       "\n",
-       "            /* Free the memory for the previous frames */\n",
-       "            if (fig.imageObj.src) {\n",
-       "                (window.URL || window.webkitURL).revokeObjectURL(\n",
-       "                    fig.imageObj.src);\n",
-       "            }\n",
-       "\n",
-       "            fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
-       "                evt.data);\n",
-       "            fig.updated_canvas_event();\n",
-       "            fig.waiting = false;\n",
-       "            return;\n",
-       "        }\n",
-       "        else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
-       "            fig.imageObj.src = evt.data;\n",
-       "            fig.updated_canvas_event();\n",
-       "            fig.waiting = false;\n",
-       "            return;\n",
-       "        }\n",
-       "\n",
-       "        var msg = JSON.parse(evt.data);\n",
-       "        var msg_type = msg['type'];\n",
-       "\n",
-       "        // Call the  \"handle_{type}\" callback, which takes\n",
-       "        // the figure and JSON message as its only arguments.\n",
-       "        try {\n",
-       "            var callback = fig[\"handle_\" + msg_type];\n",
-       "        } catch (e) {\n",
-       "            console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
-       "            return;\n",
-       "        }\n",
-       "\n",
-       "        if (callback) {\n",
-       "            try {\n",
-       "                // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
-       "                callback(fig, msg);\n",
-       "            } catch (e) {\n",
-       "                console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
-       "            }\n",
-       "        }\n",
-       "    };\n",
-       "}\n",
-       "\n",
-       "// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
-       "mpl.findpos = function(e) {\n",
-       "    //this section is from http://www.quirksmode.org/js/events_properties.html\n",
-       "    var targ;\n",
-       "    if (!e)\n",
-       "        e = window.event;\n",
-       "    if (e.target)\n",
-       "        targ = e.target;\n",
-       "    else if (e.srcElement)\n",
-       "        targ = e.srcElement;\n",
-       "    if (targ.nodeType == 3) // defeat Safari bug\n",
-       "        targ = targ.parentNode;\n",
-       "\n",
-       "    // jQuery normalizes the pageX and pageY\n",
-       "    // pageX,Y are the mouse positions relative to the document\n",
-       "    // offset() returns the position of the element relative to the document\n",
-       "    var x = e.pageX - $(targ).offset().left;\n",
-       "    var y = e.pageY - $(targ).offset().top;\n",
-       "\n",
-       "    return {\"x\": x, \"y\": y};\n",
-       "};\n",
-       "\n",
-       "/*\n",
-       " * return a copy of an object with only non-object keys\n",
-       " * we need this to avoid circular references\n",
-       " * http://stackoverflow.com/a/24161582/3208463\n",
-       " */\n",
-       "function simpleKeys (original) {\n",
-       "  return Object.keys(original).reduce(function (obj, key) {\n",
-       "    if (typeof original[key] !== 'object')\n",
-       "        obj[key] = original[key]\n",
-       "    return obj;\n",
-       "  }, {});\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.mouse_event = function(event, name) {\n",
-       "    var canvas_pos = mpl.findpos(event)\n",
-       "\n",
-       "    if (name === 'button_press')\n",
-       "    {\n",
-       "        this.canvas.focus();\n",
-       "        this.canvas_div.focus();\n",
-       "    }\n",
-       "\n",
-       "    var x = canvas_pos.x * mpl.ratio;\n",
-       "    var y = canvas_pos.y * mpl.ratio;\n",
-       "\n",
-       "    this.send_message(name, {x: x, y: y, button: event.button,\n",
-       "                             step: event.step,\n",
-       "                             guiEvent: simpleKeys(event)});\n",
-       "\n",
-       "    /* This prevents the web browser from automatically changing to\n",
-       "     * the text insertion cursor when the button is pressed.  We want\n",
-       "     * to control all of the cursor setting manually through the\n",
-       "     * 'cursor' event from matplotlib */\n",
-       "    event.preventDefault();\n",
-       "    return false;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
-       "    // Handle any extra behaviour associated with a key event\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.key_event = function(event, name) {\n",
-       "\n",
-       "    // Prevent repeat events\n",
-       "    if (name == 'key_press')\n",
-       "    {\n",
-       "        if (event.which === this._key)\n",
-       "            return;\n",
-       "        else\n",
-       "            this._key = event.which;\n",
-       "    }\n",
-       "    if (name == 'key_release')\n",
-       "        this._key = null;\n",
-       "\n",
-       "    var value = '';\n",
-       "    if (event.ctrlKey && event.which != 17)\n",
-       "        value += \"ctrl+\";\n",
-       "    if (event.altKey && event.which != 18)\n",
-       "        value += \"alt+\";\n",
-       "    if (event.shiftKey && event.which != 16)\n",
-       "        value += \"shift+\";\n",
-       "\n",
-       "    value += 'k';\n",
-       "    value += event.which.toString();\n",
-       "\n",
-       "    this._key_event_extra(event, name);\n",
-       "\n",
-       "    this.send_message(name, {key: value,\n",
-       "                             guiEvent: simpleKeys(event)});\n",
-       "    return false;\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
-       "    if (name == 'download') {\n",
-       "        this.handle_save(this, null);\n",
-       "    } else {\n",
-       "        this.send_message(\"toolbar_button\", {name: name});\n",
-       "    }\n",
-       "};\n",
-       "\n",
-       "mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
-       "    this.message.textContent = tooltip;\n",
-       "};\n",
-       "mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to  previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
-       "\n",
-       "mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
-       "\n",
-       "mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
-       "    // Create a \"websocket\"-like object which calls the given IPython comm\n",
-       "    // object with the appropriate methods. Currently this is a non binary\n",
-       "    // socket, so there is still some room for performance tuning.\n",
-       "    var ws = {};\n",
-       "\n",
-       "    ws.close = function() {\n",
-       "        comm.close()\n",
-       "    };\n",
-       "    ws.send = function(m) {\n",
-       "        //console.log('sending', m);\n",
-       "        comm.send(m);\n",
-       "    };\n",
-       "    // Register the callback with on_msg.\n",
-       "    comm.on_msg(function(msg) {\n",
-       "        //console.log('receiving', msg['content']['data'], msg);\n",
-       "        // Pass the mpl event to the overridden (by mpl) onmessage function.\n",
-       "        ws.onmessage(msg['content']['data'])\n",
-       "    });\n",
-       "    return ws;\n",
-       "}\n",
-       "\n",
-       "mpl.mpl_figure_comm = function(comm, msg) {\n",
-       "    // This is the function which gets called when the mpl process\n",
-       "    // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
-       "\n",
-       "    var id = msg.content.data.id;\n",
-       "    // Get hold of the div created by the display call when the Comm\n",
-       "    // socket was opened in Python.\n",
-       "    var element = $(\"#\" + id);\n",
-       "    var ws_proxy = comm_websocket_adapter(comm)\n",
-       "\n",
-       "    function ondownload(figure, format) {\n",
-       "        window.open(figure.imageObj.src);\n",
-       "    }\n",
-       "\n",
-       "    var fig = new mpl.figure(id, ws_proxy,\n",
-       "                           ondownload,\n",
-       "                           element.get(0));\n",
-       "\n",
-       "    // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
-       "    // web socket which is closed, not our websocket->open comm proxy.\n",
-       "    ws_proxy.onopen();\n",
-       "\n",
-       "    fig.parent_element = element.get(0);\n",
-       "    fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
-       "    if (!fig.cell_info) {\n",
-       "        console.error(\"Failed to find cell for figure\", id, fig);\n",
-       "        return;\n",
-       "    }\n",
-       "\n",
-       "    var output_index = fig.cell_info[2]\n",
-       "    var cell = fig.cell_info[0];\n",
-       "\n",
-       "};\n",
-       "\n",
-       "mpl.figure.prototype.handle_close = function(fig, msg) {\n",
-       "    var width = fig.canvas.width/mpl.ratio\n",
-       "    fig.root.unbind('remove')\n",
-       "\n",
-       "    // Update the output cell to use the data from the current canvas.\n",
-       "    fig.push_to_output();\n",
-       "    var dataURL = fig.canvas.toDataURL();\n",
-       "    // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
-       "    // the notebook keyboard shortcuts fail.\n",
-       "    IPython.keyboard_manager.enable()\n",
-       "    $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
-       "    fig.close_ws(fig, msg);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.close_ws = function(fig, msg){\n",
-       "    fig.send_message('closing', msg);\n",
-       "    // fig.ws.close()\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
-       "    // Turn the data on the canvas into data in the output cell.\n",
-       "    var width = this.canvas.width/mpl.ratio\n",
-       "    var dataURL = this.canvas.toDataURL();\n",
-       "    this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.updated_canvas_event = function() {\n",
-       "    // Tell IPython that the notebook contents must change.\n",
-       "    IPython.notebook.set_dirty(true);\n",
-       "    this.send_message(\"ack\", {});\n",
-       "    var fig = this;\n",
-       "    // Wait a second, then push the new image to the DOM so\n",
-       "    // that it is saved nicely (might be nice to debounce this).\n",
-       "    setTimeout(function () { fig.push_to_output() }, 1000);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._init_toolbar = function() {\n",
-       "    var fig = this;\n",
-       "\n",
-       "    var nav_element = $('<div/>')\n",
-       "    nav_element.attr('style', 'width: 100%');\n",
-       "    this.root.append(nav_element);\n",
-       "\n",
-       "    // Define a callback function for later on.\n",
-       "    function toolbar_event(event) {\n",
-       "        return fig.toolbar_button_onclick(event['data']);\n",
-       "    }\n",
-       "    function toolbar_mouse_event(event) {\n",
-       "        return fig.toolbar_button_onmouseover(event['data']);\n",
-       "    }\n",
-       "\n",
-       "    for(var toolbar_ind in mpl.toolbar_items){\n",
-       "        var name = mpl.toolbar_items[toolbar_ind][0];\n",
-       "        var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
-       "        var image = mpl.toolbar_items[toolbar_ind][2];\n",
-       "        var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
-       "\n",
-       "        if (!name) { continue; };\n",
-       "\n",
-       "        var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
-       "        button.click(method_name, toolbar_event);\n",
-       "        button.mouseover(tooltip, toolbar_mouse_event);\n",
-       "        nav_element.append(button);\n",
-       "    }\n",
-       "\n",
-       "    // Add the status bar.\n",
-       "    var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
-       "    nav_element.append(status_bar);\n",
-       "    this.message = status_bar[0];\n",
-       "\n",
-       "    // Add the close button to the window.\n",
-       "    var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
-       "    var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
-       "    button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
-       "    button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
-       "    buttongrp.append(button);\n",
-       "    var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
-       "    titlebar.prepend(buttongrp);\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._root_extra_style = function(el){\n",
-       "    var fig = this\n",
-       "    el.on(\"remove\", function(){\n",
-       "\tfig.close_ws(fig, {});\n",
-       "    });\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._canvas_extra_style = function(el){\n",
-       "    // this is important to make the div 'focusable\n",
-       "    el.attr('tabindex', 0)\n",
-       "    // reach out to IPython and tell the keyboard manager to turn it's self\n",
-       "    // off when our div gets focus\n",
-       "\n",
-       "    // location in version 3\n",
-       "    if (IPython.notebook.keyboard_manager) {\n",
-       "        IPython.notebook.keyboard_manager.register_events(el);\n",
-       "    }\n",
-       "    else {\n",
-       "        // location in version 2\n",
-       "        IPython.keyboard_manager.register_events(el);\n",
-       "    }\n",
-       "\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype._key_event_extra = function(event, name) {\n",
-       "    var manager = IPython.notebook.keyboard_manager;\n",
-       "    if (!manager)\n",
-       "        manager = IPython.keyboard_manager;\n",
-       "\n",
-       "    // Check for shift+enter\n",
-       "    if (event.shiftKey && event.which == 13) {\n",
-       "        this.canvas_div.blur();\n",
-       "        event.shiftKey = false;\n",
-       "        // Send a \"J\" for go to next cell\n",
-       "        event.which = 74;\n",
-       "        event.keyCode = 74;\n",
-       "        manager.command_mode();\n",
-       "        manager.handle_keydown(event);\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "mpl.figure.prototype.handle_save = function(fig, msg) {\n",
-       "    fig.ondownload(fig, null);\n",
-       "}\n",
-       "\n",
-       "\n",
-       "mpl.find_output_cell = function(html_output) {\n",
-       "    // Return the cell and output element which can be found *uniquely* in the notebook.\n",
-       "    // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
-       "    // IPython event is triggered only after the cells have been serialised, which for\n",
-       "    // our purposes (turning an active figure into a static one), is too late.\n",
-       "    var cells = IPython.notebook.get_cells();\n",
-       "    var ncells = cells.length;\n",
-       "    for (var i=0; i<ncells; i++) {\n",
-       "        var cell = cells[i];\n",
-       "        if (cell.cell_type === 'code'){\n",
-       "            for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
-       "                var data = cell.output_area.outputs[j];\n",
-       "                if (data.data) {\n",
-       "                    // IPython >= 3 moved mimebundle to data attribute of output\n",
-       "                    data = data.data;\n",
-       "                }\n",
-       "                if (data['text/html'] == html_output) {\n",
-       "                    return [cell, data, j];\n",
-       "                }\n",
-       "            }\n",
-       "        }\n",
-       "    }\n",
-       "}\n",
-       "\n",
-       "// Register the function which deals with the matplotlib target/channel.\n",
-       "// The kernel may be null if the page has been refreshed.\n",
-       "if (IPython.notebook.kernel != null) {\n",
-       "    IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
-       "}\n"
-      ],
-      "text/plain": [
-       "<IPython.core.display.Javascript object>"
-      ]
-     },
-     "metadata": {},
-     "output_type": "display_data"
-    },
-    {
-     "data": {
-      "text/html": [
-       "<img src=\"\" width=\"1000\">"
-      ],
-      "text/plain": [
-       "<IPython.core.display.HTML object>"
-      ]
-     },
-     "metadata": {},
-     "output_type": "display_data"
-    },
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    }
-   ],
-   "source": [
-    "#df_results = %sql SELECT * FROM $results_table ORDER BY run_id;\n",
-    "df_results = %sql SELECT * FROM $results_table ORDER BY validation_metrics_final DESC LIMIT 100;\n",
-    "df_results = df_results.DataFrame()\n",
-    "\n",
-    "#set up plots\n",
-    "fig, axs = plt.subplots(nrows=1, ncols=2, figsize=(10,5))\n",
-    "fig.legend(ncol=4)\n",
-    "fig.tight_layout()\n",
-    "\n",
-    "ax_metric = axs[0]\n",
-    "ax_loss = axs[1]\n",
-    "\n",
-    "ax_metric.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
-    "ax_metric.set_xlabel('Iteration')\n",
-    "ax_metric.set_ylabel('Metric')\n",
-    "ax_metric.set_title('Validation metric curve')\n",
-    "\n",
-    "ax_loss.xaxis.set_major_locator(MaxNLocator(integer=True))\n",
-    "ax_loss.set_xlabel('Iteration')\n",
-    "ax_loss.set_ylabel('Loss')\n",
-    "ax_loss.set_title('Validation loss curve')\n",
-    "\n",
-    "for run_id in df_results['run_id']:\n",
-    "    df_output_info = %sql SELECT validation_metrics,validation_loss FROM $results_table WHERE run_id = $run_id\n",
-    "    df_output_info = df_output_info.DataFrame()\n",
-    "    validation_metrics = df_output_info['validation_metrics'][0]\n",
-    "    validation_loss = df_output_info['validation_loss'][0]\n",
-    "    X = range(len(validation_metrics))\n",
-    "    \n",
-    "    ax_metric.plot(X, validation_metrics, label=run_id, marker='o')\n",
-    "    ax_loss.plot(X, validation_loss, label=run_id, marker='o')\n",
-    "\n",
-    "# fig.savefig('./lc_keras_fit.png', dpi = 300)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"print\"></a>\n",
-    "# 6. Print run schedules (display only)"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Pretty print reg Hyperband run schedule"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 31,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "max_iter = 27\n",
-      "eta = 3\n",
-      "B = 4*max_iter = 108\n",
-      "skip_last = 0\n",
-      " \n",
-      "s=3\n",
-      "n_i      r_i\n",
-      "------------\n",
-      "27     1.0\n",
-      "9.0     3.0\n",
-      "3.0     9.0\n",
-      "1.0     27.0\n",
-      " \n",
-      "s=2\n",
-      "n_i      r_i\n",
-      "------------\n",
-      "9     3.0\n",
-      "3.0     9.0\n",
-      "1.0     27.0\n",
-      " \n",
-      "s=1\n",
-      "n_i      r_i\n",
-      "------------\n",
-      "6     9.0\n",
-      "2.0     27.0\n",
-      " \n",
-      "s=0\n",
-      "n_i      r_i\n",
-      "------------\n",
-      "4     27\n",
-      " \n",
-      "sum of configurations at leaf nodes across all s = 8.0\n",
-      "(if have more workers than this, they may not be 100% busy)\n"
-     ]
-    }
-   ],
-   "source": [
-    "import numpy as np\n",
-    "from math import log, ceil\n",
-    "\n",
-    "#input\n",
-    "max_iter = 27  # maximum iterations/epochs per configuration\n",
-    "eta = 3  # defines downsampling rate (default=3)\n",
-    "skip_last = 0 # 1 means skip last run in each bracket, 0 means run full bracket\n",
-    "\n",
-    "logeta = lambda x: log(x)/log(eta)\n",
-    "s_max = int(logeta(max_iter))  # number of unique executions of Successive Halving (minus one)\n",
-    "B = (s_max+1)*max_iter  # total number of iterations (without reuse) per execution of Succesive Halving (n,r)\n",
-    "\n",
-    "#echo output\n",
-    "print (\"max_iter = \" + str(max_iter))\n",
-    "print (\"eta = \" + str(eta))\n",
-    "print (\"B = \" + str(s_max+1) + \"*max_iter = \" + str(B))\n",
-    "print (\"skip_last = \" + str(skip_last))\n",
-    "\n",
-    "sum_leaf_n_i = 0 # count configurations at leaf nodes across all s\n",
-    "\n",
-    "#### Begin Finite Horizon Hyperband outlerloop. Repeat indefinitely.\n",
-    "for s in reversed(range(s_max+1)):\n",
-    "    \n",
-    "    print (\" \")\n",
-    "    print (\"s=\" + str(s))\n",
-    "    print (\"n_i      r_i\")\n",
-    "    print (\"------------\")\n",
-    "    counter = 0\n",
-    "    \n",
-    "    n = int(ceil(int(B/max_iter/(s+1))*eta**s)) # initial number of configurations\n",
-    "    r = max_iter*eta**(-s) # initial number of iterations to run configurations for\n",
-    "\n",
-    "    #### Begin Finite Horizon Successive Halving with (n,r)\n",
-    "    #T = [ get_random_hyperparameter_configuration() for i in range(n) ] \n",
-    "    for i in range((s+1) - int(skip_last)):\n",
-    "        # Run each of the n_i configs for r_i iterations and keep best n_i/eta\n",
-    "        n_i = n*eta**(-i)\n",
-    "        r_i = r*eta**(i)\n",
-    "        \n",
-    "        print (str(n_i) + \"     \" + str (r_i))\n",
-    "        \n",
-    "        # check if leaf node for this s\n",
-    "        if counter == (s-skip_last):\n",
-    "            sum_leaf_n_i += n_i\n",
-    "        counter += 1\n",
-    "        \n",
-    "        #val_losses = [ run_then_return_val_loss(num_iters=r_i,hyperparameters=t) for t in T ]\n",
-    "        #T = [ T[i] for i in argsort(val_losses)[0:int( n_i/eta )] ]\n",
-    "    #### End Finite Horizon Successive Halving with (n,r)\n",
-    "\n",
-    "print (\" \")\n",
-    "print (\"sum of configurations at leaf nodes across all s = \" + str(sum_leaf_n_i))\n",
-    "print (\"(if have more workers than this, they may not be 100% busy)\")"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Pretty print Hyperband diagonal run schedule"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": null,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "import numpy as np\n",
-    "from math import log, ceil\n",
-    "\n",
-    "#input\n",
-    "max_iter = 27  # maximum iterations/epochs per configuration\n",
-    "eta = 3  # defines downsampling rate (default=3)\n",
-    "skip_last = 1 # 1 means skip last run in each bracket, 0 means run full bracket\n",
-    "\n",
-    "logeta = lambda x: log(x)/log(eta)\n",
-    "s_max = int(logeta(max_iter))  # number of unique executions of Successive Halving (minus one)\n",
-    "B = (s_max+1)*max_iter  # total number of iterations (without reuse) per execution of Succesive Halving (n,r)\n",
-    "\n",
-    "#echo output\n",
-    "print (\"echo input:\")\n",
-    "print (\"max_iter = \" + str(max_iter))\n",
-    "print (\"eta = \" + str(eta))\n",
-    "print (\"s_max = \" + str(s_max))\n",
-    "print (\"B = \" + str(s_max+1) + \"*max_iter = \" + str(B))\n",
-    "\n",
-    "print (\" \")\n",
-    "print (\"initial n, r values for each s:\")\n",
-    "initial_n_vals = {}\n",
-    "initial_r_vals = {}\n",
-    "# get hyper parameter configs for each s\n",
-    "for s in reversed(range(s_max+1)):\n",
-    "    \n",
-    "    n = int(ceil(int(B/max_iter/(s+1))*eta**s)) # initial number of configurations\n",
-    "    r = max_iter*eta**(-s) # initial number of iterations to run configurations for\n",
-    "    \n",
-    "    initial_n_vals[s] = n \n",
-    "    initial_r_vals[s] = r \n",
-    "    \n",
-    "    print (\"s=\" + str(s))\n",
-    "    print (\"n=\" + str(n))\n",
-    "    print (\"r=\" + str(r))\n",
-    "    print (\" \")\n",
-    "    \n",
-    "print (\"outer loop on diagonal:\")\n",
-    "# outer loop on diagonal\n",
-    "for i in range((s_max+1) - int(skip_last)):\n",
-    "    print (\" \")\n",
-    "    print (\"i=\" + str(i))\n",
-    "    \n",
-    "    print (\"inner loop on s desc:\")\n",
-    "    # inner loop on s desc\n",
-    "    for s in range(s_max, s_max-i-1, -1):\n",
-    "        n_i = initial_n_vals[s]*eta**(-i+s_max-s)\n",
-    "        r_i = initial_r_vals[s]*eta**(i-s_max+s)\n",
-    "        \n",
-    "        print (\"s=\" + str(s))\n",
-    "        print (\"n_i=\" + str(n_i))\n",
-    "        print (\"r_i=\" + str(r_i))"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "<a id=\"predict\"></a>\n",
-    "# 7. Inference\n",
-    "\n",
-    "Use the best model from the last run.\n",
-    "\n",
-    "## 7a. Run predict on the whole validation dataset"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 93,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>mst_key</th>\n",
-       "        <th>model_id</th>\n",
-       "        <th>compile_params</th>\n",
-       "        <th>fit_params</th>\n",
-       "        <th>model_type</th>\n",
-       "        <th>model_size</th>\n",
-       "        <th>metrics_elapsed_time</th>\n",
-       "        <th>metrics_type</th>\n",
-       "        <th>training_metrics_final</th>\n",
-       "        <th>training_loss_final</th>\n",
-       "        <th>training_metrics</th>\n",
-       "        <th>training_loss</th>\n",
-       "        <th>validation_metrics_final</th>\n",
-       "        <th>validation_loss_final</th>\n",
-       "        <th>validation_metrics</th>\n",
-       "        <th>validation_loss</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>6</td>\n",
-       "        <td>2</td>\n",
-       "        <td>loss='categorical_crossentropy',optimizer='adam(lr=0.002826545217978097)',metrics=['accuracy']</td>\n",
-       "        <td>batch_size=128,epochs=5</td>\n",
-       "        <td>madlib_keras</td>\n",
-       "        <td>2159.70019531</td>\n",
-       "        <td>[156.498700857162, 314.38369679451, 471.076618909836]</td>\n",
-       "        <td>[u'accuracy']</td>\n",
-       "        <td>0.89631998539</td>\n",
-       "        <td>0.301868826151</td>\n",
-       "        <td>[0.817480027675629, 0.862479984760284, 0.896319985389709]</td>\n",
-       "        <td>[0.536632478237152, 0.400230169296265, 0.301868826150894]</td>\n",
-       "        <td>0.805899977684</td>\n",
-       "        <td>0.613121390343</td>\n",
-       "        <td>[0.764500021934509, 0.788500010967255, 0.805899977684021]</td>\n",
-       "        <td>[0.717438697814941, 0.662977695465088, 0.613121390342712]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(6, 2, u\"loss='categorical_crossentropy',optimizer='adam(lr=0.002826545217978097)',metrics=['accuracy']\", u'batch_size=128,epochs=5', u'madlib_keras', 2159.70019531, [156.498700857162, 314.38369679451, 471.076618909836], [u'accuracy'], 0.89631998539, 0.301868826151, [0.817480027675629, 0.862479984760284, 0.896319985389709], [0.536632478237152, 0.400230169296265, 0.301868826150894], 0.805899977684, 0.613121390343, [0.764500021934509, 0.788500010967255, 0.805899977684021], [0.717438697814941, 0.662977695465088, 0.613121390342712])]"
-      ]
-     },
-     "execution_count": 93,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql SELECT * FROM $best_model_info;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 94,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    }
-   ],
-   "source": [
-    "best_mst_key = %sql SELECT mst_key FROM $best_model_info; \n",
-    "best_mst_key = best_mst_key.DataFrame().to_numpy()[0][0]"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 95,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "5 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>estimated_y</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>2</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>3</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>4</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>5</td>\n",
-       "        <td>0</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1, 0), (2, 0), (3, 0), (4, 0), (5, 0)]"
-      ]
-     },
-     "execution_count": 95,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql DROP TABLE IF EXISTS cifar10_val_predict;\n",
-    "%sql SELECT madlib.madlib_keras_predict('cifar10_best_model', 'cifar10_val', 'id', 'x', 'cifar10_val_predict', 'response', True, $best_mst_key);\n",
-    "%sql SELECT * FROM cifar10_val_predict ORDER BY id LIMIT 5;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Count missclassifications"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 96,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>count</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>1941</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(1941L,)]"
-      ]
-     },
-     "execution_count": 96,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT COUNT(*) FROM cifar10_val_predict JOIN cifar10_val USING (id) \n",
-    "WHERE cifar10_val_predict.estimated_y != cifar10_val.y;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Accuracy"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 97,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>test_accuracy_percent</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>80.59</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(Decimal('80.59'),)]"
-      ]
-     },
-     "execution_count": 97,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "SELECT round(count(*)*100.0/10000.0,2) as test_accuracy_percent from\n",
-    "    (select cifar10_val.y as actual, cifar10_val_predict.estimated_y as predicted\n",
-    "     from cifar10_val_predict inner join cifar10_val\n",
-    "     on cifar10_val.id=cifar10_val_predict.id) q\n",
-    "WHERE q.actual=q.predicted;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "## 7b. Select a random image from the validation dataset and run predict\n",
-    "\n",
-    "Label map"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 98,
-   "metadata": {},
-   "outputs": [],
-   "source": [
-    "label_names = {\n",
-    "    0 :\"airplane\",\n",
-    "    1 :\"automobile\",\n",
-    "    2 :\"bird\",\n",
-    "    3:\"cat\",\n",
-    "    4 :\"deer\",\n",
-    "    5 :\"dog\",\n",
-    "    6 :\"frog\",\n",
-    "    7 :\"horse\",\n",
-    "    8 :\"ship\",\n",
-    "    9 :\"truck\"\n",
-    "}"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Pick a random image"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 99,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/plain": [
-       "[]"
-      ]
-     },
-     "execution_count": 99,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS cifar10_val_random;\n",
-    "CREATE TABLE cifar10_val_random AS\n",
-    "    SELECT * FROM cifar10_val ORDER BY random() LIMIT 1;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Predict"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 100,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>id</th>\n",
-       "        <th>prob_0</th>\n",
-       "        <th>prob_1</th>\n",
-       "        <th>prob_2</th>\n",
-       "        <th>prob_3</th>\n",
-       "        <th>prob_4</th>\n",
-       "        <th>prob_5</th>\n",
-       "        <th>prob_6</th>\n",
-       "        <th>prob_7</th>\n",
-       "        <th>prob_8</th>\n",
-       "        <th>prob_9</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>9813</td>\n",
-       "        <td>7.9166554e-08</td>\n",
-       "        <td>0.00038159246</td>\n",
-       "        <td>8.776156e-11</td>\n",
-       "        <td>1.7702625e-08</td>\n",
-       "        <td>1.2219187e-10</td>\n",
-       "        <td>8.096258e-10</td>\n",
-       "        <td>5.192042e-10</td>\n",
-       "        <td>1.5758073e-09</td>\n",
-       "        <td>4.106987e-07</td>\n",
-       "        <td>0.99961793</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[(9813, 7.9166554e-08, 0.00038159246, 8.776156e-11, 1.7702625e-08, 1.2219187e-10, 8.096258e-10, 5.192042e-10, 1.5758073e-09, 4.106987e-07, 0.99961793)]"
-      ]
-     },
-     "execution_count": 100,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%sql DROP TABLE IF EXISTS cifar10_val_random_predict;\n",
-    "%sql SELECT madlib.madlib_keras_predict('cifar10_best_model', 'cifar10_val_random', 'id', 'x', 'cifar10_val_random_predict', 'prob', True, $best_mst_key);\n",
-    "%sql SELECT * FROM cifar10_val_random_predict ;"
-   ]
-  },
-  {
-   "cell_type": "markdown",
-   "metadata": {},
-   "source": [
-    "Format output and display"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 101,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "Done.\n",
-      "1 rows affected.\n",
-      "1 rows affected.\n"
-     ]
-    },
-    {
-     "data": {
-      "text/html": [
-       "<table>\n",
-       "    <tr>\n",
-       "        <th>feature_vector</th>\n",
-       "    </tr>\n",
-       "    <tr>\n",
-       "        <td>[7.9166554e-08, 0.00038159246, 8.776156e-11, 1.7702625e-08, 1.2219187e-10, 8.096258e-10, 5.192042e-10, 1.5758073e-09, 4.106987e-07, 0.99961793]</td>\n",
-       "    </tr>\n",
-       "</table>"
-      ],
-      "text/plain": [
-       "[([7.9166554e-08, 0.00038159246, 8.776156e-11, 1.7702625e-08, 1.2219187e-10, 8.096258e-10, 5.192042e-10, 1.5758073e-09, 4.106987e-07, 0.99961793],)]"
-      ]
-     },
-     "execution_count": 101,
-     "metadata": {},
-     "output_type": "execute_result"
-    }
-   ],
-   "source": [
-    "%%sql\n",
-    "DROP TABLE IF EXISTS cifar10_val_random_predict_array, cifar10_val_random_predict_array_summary;\n",
-    "SELECT madlib.cols2vec(\n",
-    "    'cifar10_val_random_predict',\n",
-    "    'cifar10_val_random_predict_array',\n",
-    "    '*',\n",
-    "    'id'\n",
-    ");\n",
-    "select * from cifar10_val_random_predict_array;"
-   ]
-  },
-  {
-   "cell_type": "code",
-   "execution_count": 102,
-   "metadata": {},
-   "outputs": [
-    {
-     "name": "stdout",
-     "output_type": "stream",
-     "text": [
-      "1 rows affected.\n",
-      "1 rows affected.\n",
-      " \n",
-      "truck 0.99961793\n",
-      "automobile 0.00038159246\n",
-      "ship 4.106987e-07\n"
-     ]
-    },
-    {
-     "data": {
-      "image/png": "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\n",
-      "text/plain": [
-       "<Figure size 432x288 with 1 Axes>"
-      ]
-     },
-     "metadata": {
-      "needs_background": "light"
-     },
-     "output_type": "display_data"
-    }
-   ],
-   "source": [
-    "x = %sql SELECT x FROM cifar10_val_random;\n",
-    "x = x.DataFrame().to_numpy()\n",
-    "import numpy as np\n",
-    "from matplotlib.pyplot import imshow\n",
-    "%matplotlib inline\n",
-    "x_np = np.array(x[0][0], dtype=np.uint8)\n",
-    "imshow(x_np)\n",
-    "\n",
-    "x = %sql SELECT * FROM cifar10_val_random_predict_array;\n",
-    "x = x.DataFrame().to_numpy()\n",
-    "x = np.array(x[0][0])\n",
-    "top_3_prob_label_indices = x.argsort()[-3:][::-1]\n",
-    "print (\" \");\n",
-    "for index in top_3_prob_label_indices:\n",
-    "    print (label_names[index], x[index])"
-   ]
-  }
- ],
- "metadata": {
-  "kernelspec": {
-   "display_name": "Python 2",
-   "language": "python",
-   "name": "python2"
-  },
-  "language_info": {
-   "codemirror_mode": {
-    "name": "ipython",
-    "version": 2
-   },
-   "file_extension": ".py",
-   "mimetype": "text/x-python",
-   "name": "python",
-   "nbconvert_exporter": "python",
-   "pygments_lexer": "ipython2",
-   "version": "2.7.16"
-  }
- },
- "nbformat": 4,
- "nbformat_minor": 1
-}