{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Categorical variables: optimizing a neural network with TensorFlow and Emukit"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This notebook is going to illustrate how to use categorical variables with Emukit."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "### General imports\n",
    "%matplotlib inline\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "### Necessary imports\n",
    "import tensorflow as tf\n",
    "tf.logging.set_verbosity(tf.logging.ERROR)\n",
    "import GPy\n",
    "\n",
    "from emukit.core import ContinuousParameter, ParameterSpace, CategoricalParameter, OneHotEncoding\n",
    "from emukit.model_wrappers import GPyModelWrapper\n",
    "from emukit.bayesian_optimization.loops import BayesianOptimizationLoop"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "For this example we are going to optimize a very simple neural network built with TensorFlow. We will reuse the very first example of [how to use TensorFlow](https://colab.research.google.com/github/tensorflow/docs/blob/master/site/en/tutorials/_index.ipynb) with number of epochs reduced to 1 in the interest of saving time. We are going to consider two hyper parameters that are going to be subjects of optimization:\n",
    "* Type of optimizer. We are going to choose from Adam, AdaGrad and SGD. This is clearly a categorical parameter.\n",
    "* Dropout rate, ranging between 0.05 and 0.95. This parameter is continuous.  "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Here is the definition of the function we are going to optimize:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "mnist = tf.keras.datasets.mnist\n",
    "\n",
    "(x_train, y_train),(x_test, y_test) = mnist.load_data()\n",
    "x_train, x_test = x_train / 255.0, x_test / 255.0 \n",
    "\n",
    "def eval_model(dropout_rate, optimizer):\n",
    "    model = tf.keras.models.Sequential([\n",
    "      tf.keras.layers.Flatten(),\n",
    "      tf.keras.layers.Dense(512, activation=tf.nn.relu),\n",
    "      tf.keras.layers.Dropout(dropout_rate),\n",
    "      tf.keras.layers.Dense(10, activation=tf.nn.softmax)\n",
    "    ])\n",
    "    model.compile(optimizer=optimizer,\n",
    "                  loss='sparse_categorical_crossentropy',\n",
    "                  metrics=['accuracy'])\n",
    "\n",
    "    model.fit(x_train, y_train, epochs=1)\n",
    "    loss, accuracy = model.evaluate(x_test, y_test)\n",
    "    \n",
    "    return accuracy"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Emukit takes both objective and model as inputs. So when users come to Emukit it is expected that they have already done necessary conversions to turn categorical parameters into numerical values, known as encodings.\n",
    "\n",
    "Encodings in Emukit are represented with `Encoding` class and its subclasses. Emukit provides implementations for a few standard encodings (at the time of writing there are two: one hot and ordinal). If your encoding is not included in one of these, you can either `Encoding` class directly, or subclass it. We should leave discussion of these options to a different tutorial.\n",
    "\n",
    "For this example we will use one hot encoding already included in Emukit."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "optimizers = ['adam', 'adagrad', 'sgd']\n",
    "encoding = OneHotEncoding(optimizers)\n",
    "space = ParameterSpace([\n",
    "    ContinuousParameter('dropout_rate', 0.05, 0.95),\n",
    "    CategoricalParameter('optimizer', encoding)\n",
    "])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Emukit requires objective function to accept 2d array as an input, and it also expects model and objective function to accept input of the same form. It is not the case with our objective yet: it declares input parameters explicitly, and expects optimizer name instead of encoding value. Let's fix it. Note how we can use our encoding object here:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def emukit_friendly_objective_function(input_rows):\n",
    "    output = []\n",
    "    for row in input_rows:\n",
    "        dropout_rate = row[0]\n",
    "        optimizer = encoding.get_category(row[1:])\n",
    "        eval_result = eval_model(dropout_rate, optimizer)\n",
    "        \n",
    "        # Emukit minimizes, so we need to revert accuracy\n",
    "        output.append([-1 * eval_result])\n",
    "    \n",
    "    return np.array(output)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We can use random forest as our model, which comes in Emukit's examples package. Notice that you need Scikit-learn installed to be able to use it. We will use a couple of random values to initialize it:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "60000/60000 [==============================] - 9s 147us/sample - loss: 0.5335 - acc: 0.8344\n",
      "10000/10000 [==============================] - 0s 41us/sample - loss: 0.1861 - acc: 0.9469\n",
      "60000/60000 [==============================] - 9s 143us/sample - loss: 0.2328 - acc: 0.9310\n",
      "10000/10000 [==============================] - 0s 42us/sample - loss: 0.1264 - acc: 0.9613\n",
      "60000/60000 [==============================] - 6s 94us/sample - loss: 0.9496 - acc: 0.6998\n",
      "10000/10000 [==============================] - 0s 44us/sample - loss: 0.3953 - acc: 0.8991\n",
      "60000/60000 [==============================] - 9s 145us/sample - loss: 0.2192 - acc: 0.9351\n",
      "10000/10000 [==============================] - 0s 44us/sample - loss: 0.1057 - acc: 0.9686\n",
      "60000/60000 [==============================] - 7s 117us/sample - loss: 0.2727 - acc: 0.9197\n",
      "10000/10000 [==============================] - 0s 45us/sample - loss: 0.1492 - acc: 0.9561\n"
     ]
    }
   ],
   "source": [
    "from emukit.examples.models.random_forest import RandomForest\n",
    "from emukit.experimental_design import RandomDesign\n",
    "\n",
    "random_design = RandomDesign(space)\n",
    "initial_points_count = 5\n",
    "X_init = random_design.get_samples(initial_points_count)\n",
    "Y_init = emukit_friendly_objective_function(X_init)\n",
    "\n",
    "rf_model = RandomForest(X_init, Y_init)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Let's examine the returned data:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[-0.94690001],\n",
       "       [-0.96130002],\n",
       "       [-0.89910001],\n",
       "       [-0.96859998],\n",
       "       [-0.95609999]])"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Y_init"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "At this point everything is ready to run the optimization loop."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "60000/60000 [==============================] - 7s 113us/sample - loss: 0.2211 - acc: 0.9347\n",
      "10000/10000 [==============================] - 0s 45us/sample - loss: 0.1269 - acc: 0.9631\n",
      "60000/60000 [==============================] - 9s 145us/sample - loss: 0.2100 - acc: 0.9377\n",
      "10000/10000 [==============================] - 0s 46us/sample - loss: 0.1063 - acc: 0.9689\n",
      "60000/60000 [==============================] - 7s 113us/sample - loss: 0.2281 - acc: 0.9339\n",
      "10000/10000 [==============================] - 0s 47us/sample - loss: 0.1268 - acc: 0.9637\n",
      "60000/60000 [==============================] - 6s 96us/sample - loss: 0.6212 - acc: 0.8449\n",
      "10000/10000 [==============================] - 0s 49us/sample - loss: 0.3445 - acc: 0.9089\n",
      "60000/60000 [==============================] - 9s 144us/sample - loss: 0.2899 - acc: 0.9130\n",
      "10000/10000 [==============================] - 0s 48us/sample - loss: 0.1289 - acc: 0.9591\n",
      "60000/60000 [==============================] - 9s 146us/sample - loss: 0.2107 - acc: 0.9373\n",
      "10000/10000 [==============================] - 0s 49us/sample - loss: 0.1028 - acc: 0.9675\n",
      "60000/60000 [==============================] - 7s 118us/sample - loss: 0.3568 - acc: 0.8930\n",
      "10000/10000 [==============================] - 1s 50us/sample - loss: 0.1733 - acc: 0.9482\n",
      "60000/60000 [==============================] - 9s 147us/sample - loss: 0.2080 - acc: 0.9392\n",
      "10000/10000 [==============================] - 1s 51us/sample - loss: 0.1032 - acc: 0.9682\n",
      "60000/60000 [==============================] - 9s 149us/sample - loss: 0.2142 - acc: 0.9377\n",
      "10000/10000 [==============================] - 1s 52us/sample - loss: 0.0929 - acc: 0.9707\n",
      "60000/60000 [==============================] - 9s 150us/sample - loss: 0.2175 - acc: 0.9352\n",
      "10000/10000 [==============================] - 1s 54us/sample - loss: 0.1113 - acc: 0.9652\n"
     ]
    }
   ],
   "source": [
    "from emukit.core.optimization import LocalSearchAcquisitionOptimizer\n",
    "\n",
    "ls = LocalSearchAcquisitionOptimizer(space, num_steps=10, num_init_points=5)\n",
    "loop = BayesianOptimizationLoop(model=rf_model, space=space, acquisition_optimizer=ls)\n",
    "loop.run_loop(emukit_friendly_objective_function, 10)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Let's plot the results of the optimization run."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "%matplotlib inline\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "accuracies = -1 * loop.loop_state.Y\n",
    "\n",
    "plt.title('Executions of objective')\n",
    "plt.xlabel('Iterations')\n",
    "plt.ylabel('Accuracy')\n",
    "plt.plot(range(accuracies.shape[0]), accuracies[:, 0], c='k')\n",
    "plt.scatter(range(initial_points_count), accuracies[:initial_points_count, 0], c='r', linewidths=5, label=\"Initial values\")\n",
    "plt.scatter(range(initial_points_count, accuracies.shape[0]), accuracies[initial_points_count:, 0], c='b', linewidths=5, label=\"Optimizaion executions\")\n",
    "plt.legend();"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.5.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
