{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# GP Regression with Uncertain Inputs\n",
    "\n",
    "## Introduction\n",
    "\n",
    "In this notebook, we're going to demonstrate one way of dealing with uncertainty in our training data. Let's say that we're collecting training data that models the following function.\n",
    "\n",
    "\\begin{align}\n",
    "y &= \\sin(2\\pi x) + \\epsilon \\\\\n",
    "  \\epsilon &\\sim \\mathcal{N}(0, 0.2) \n",
    "\\end{align}\n",
    "\n",
    "However, now assume that we're a bit uncertain about our features. In particular, we're going to assume that every `x_i` value is not a point but a distribution instead. E.g.\n",
    "\n",
    "$$ x_i \\sim \\mathcal{N}(\\mu_i, \\sigma_i). $$"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Using a distributional kernel to deal with uncertain inputs\n",
    "\n",
    "Rather than using a variational method (see the GP Regression with Uncertian Inputs tutorial in the variational examples), if we explicitly know the type of uncertainty in our inputs we can pass that into our kernel.\n",
    "\n",
    "More specifically, assuming Gaussian inputs, we will compute the symmetrized KL divergence between the Gaussian inputs.\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import math\n",
    "import torch\n",
    "import tqdm\n",
    "import gpytorch\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "%matplotlib inline\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Training data is 100 points in [0,1] inclusive regularly spaced\n",
    "train_x_mean = torch.linspace(0, 1, 20)\n",
    "# We'll assume the variance shrinks the closer we get to 1\n",
    "train_x_stdv = torch.linspace(0.03, 0.01, 20)\n",
    "\n",
    "# True function is sin(2*pi*x) with Gaussian noise\n",
    "train_y = torch.sin(train_x_mean * (2 * math.pi)) + torch.randn(train_x_mean.size()) * 0.2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "To effectively pass in the training distributional data, we will need to stack the mean and log variances."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_x_distributional = torch.stack((train_x_mean, (train_x_stdv**2).log()), dim=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7fc3069399d0>"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x216 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "f, ax = plt.subplots(1, 1, figsize=(8, 3))\n",
    "ax.errorbar(train_x_mean, train_y, xerr=(train_x_stdv * 2), fmt=\"k*\", label=\"Train Data\")\n",
    "ax.legend()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We train the hyperparameters of the resulting distributional GP via type-II gradient descent, as is standard in many settings. We could also do fully Bayesian inference."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "from gpytorch.models import ExactGP\n",
    "from gpytorch.kernels import GaussianSymmetrizedKLKernel, ScaleKernel\n",
    "from gpytorch.means import ConstantMean\n",
    "\n",
    "class ExactGPModel(ExactGP):\n",
    "    def __init__(self, train_x, train_y, likelihood):\n",
    "        super(ExactGPModel, self).__init__(train_x, train_y, likelihood)\n",
    "        self.mean_module = ConstantMean()\n",
    "        self.covar_module = ScaleKernel(GaussianSymmetrizedKLKernel())\n",
    "        \n",
    "    def forward(self, x):\n",
    "        mean_x = self.mean_module(x)\n",
    "        covar_x = self.covar_module(x)\n",
    "        return gpytorch.distributions.MultivariateNormal(mean_x, covar_x)\n",
    "\n",
    "# initialize likelihood and model\n",
    "likelihood = gpytorch.likelihoods.GaussianLikelihood()\n",
    "model = ExactGPModel(train_x_distributional, train_y, likelihood)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Iter 1/500 - Loss: 1.274   lengthscale: 0.693   noise: 0.693\n",
      "Iter 2/500 - Loss: 1.242   lengthscale: 0.826   noise: 0.576\n",
      "Iter 3/500 - Loss: 1.173   lengthscale: 0.955   noise: 0.475\n",
      "Iter 4/500 - Loss: 1.141   lengthscale: 1.102   noise: 0.389\n",
      "Iter 5/500 - Loss: 1.121   lengthscale: 1.266   noise: 0.317\n",
      "Iter 6/500 - Loss: 1.097   lengthscale: 1.442   noise: 0.261\n",
      "Iter 7/500 - Loss: 1.093   lengthscale: 1.621   noise: 0.220\n",
      "Iter 8/500 - Loss: 1.112   lengthscale: 1.801   noise: 0.192\n",
      "Iter 9/500 - Loss: 1.112   lengthscale: 1.988   noise: 0.177\n",
      "Iter 10/500 - Loss: 1.095   lengthscale: 2.188   noise: 0.172\n",
      "Iter 11/500 - Loss: 1.081   lengthscale: 2.401   noise: 0.172\n",
      "Iter 12/500 - Loss: 1.068   lengthscale: 2.624   noise: 0.177\n",
      "Iter 13/500 - Loss: 1.049   lengthscale: 2.856   noise: 0.183\n",
      "Iter 14/500 - Loss: 1.031   lengthscale: 3.095   noise: 0.188\n",
      "Iter 15/500 - Loss: 1.020   lengthscale: 3.340   noise: 0.191\n",
      "Iter 16/500 - Loss: 1.014   lengthscale: 3.589   noise: 0.189\n",
      "Iter 17/500 - Loss: 1.007   lengthscale: 3.842   noise: 0.183\n",
      "Iter 18/500 - Loss: 0.997   lengthscale: 4.098   noise: 0.173\n",
      "Iter 19/500 - Loss: 0.983   lengthscale: 4.356   noise: 0.159\n",
      "Iter 20/500 - Loss: 0.968   lengthscale: 4.616   noise: 0.144\n",
      "Iter 21/500 - Loss: 0.951   lengthscale: 4.877   noise: 0.128\n",
      "Iter 22/500 - Loss: 0.933   lengthscale: 5.139   noise: 0.112\n",
      "Iter 23/500 - Loss: 0.913   lengthscale: 5.402   noise: 0.097\n",
      "Iter 24/500 - Loss: 0.891   lengthscale: 5.667   noise: 0.083\n",
      "Iter 25/500 - Loss: 0.870   lengthscale: 5.933   noise: 0.070\n",
      "Iter 26/500 - Loss: 0.850   lengthscale: 6.202   noise: 0.059\n",
      "Iter 27/500 - Loss: 0.832   lengthscale: 6.474   noise: 0.050\n",
      "Iter 28/500 - Loss: 0.814   lengthscale: 6.748   noise: 0.042\n",
      "Iter 29/500 - Loss: 0.794   lengthscale: 7.025   noise: 0.035\n",
      "Iter 30/500 - Loss: 0.775   lengthscale: 7.304   noise: 0.029\n",
      "Iter 31/500 - Loss: 0.756   lengthscale: 7.586   noise: 0.024\n",
      "Iter 32/500 - Loss: 0.737   lengthscale: 7.868   noise: 0.020\n",
      "Iter 33/500 - Loss: 0.719   lengthscale: 8.152   noise: 0.017\n",
      "Iter 34/500 - Loss: 0.701   lengthscale: 8.437   noise: 0.014\n",
      "Iter 35/500 - Loss: 0.684   lengthscale: 8.722   noise: 0.012\n",
      "Iter 36/500 - Loss: 0.668   lengthscale: 9.007   noise: 0.010\n",
      "Iter 37/500 - Loss: 0.653   lengthscale: 9.291   noise: 0.008\n",
      "Iter 38/500 - Loss: 0.637   lengthscale: 9.575   noise: 0.007\n",
      "Iter 39/500 - Loss: 0.621   lengthscale: 9.859   noise: 0.006\n",
      "Iter 40/500 - Loss: 0.606   lengthscale: 10.142   noise: 0.005\n",
      "Iter 41/500 - Loss: 0.591   lengthscale: 10.424   noise: 0.005\n",
      "Iter 42/500 - Loss: 0.577   lengthscale: 10.705   noise: 0.004\n",
      "Iter 43/500 - Loss: 0.564   lengthscale: 10.985   noise: 0.003\n",
      "Iter 44/500 - Loss: 0.551   lengthscale: 11.263   noise: 0.003\n",
      "Iter 45/500 - Loss: 0.539   lengthscale: 11.540   noise: 0.003\n",
      "Iter 46/500 - Loss: 0.526   lengthscale: 11.815   noise: 0.002\n",
      "Iter 47/500 - Loss: 0.514   lengthscale: 12.088   noise: 0.002\n",
      "Iter 48/500 - Loss: 0.503   lengthscale: 12.359   noise: 0.002\n",
      "Iter 49/500 - Loss: 0.492   lengthscale: 12.627   noise: 0.002\n",
      "Iter 50/500 - Loss: 0.482   lengthscale: 12.892   noise: 0.001\n",
      "Iter 51/500 - Loss: 0.472   lengthscale: 13.154   noise: 0.001\n",
      "Iter 52/500 - Loss: 0.462   lengthscale: 13.412   noise: 0.001\n",
      "Iter 53/500 - Loss: 0.453   lengthscale: 13.668   noise: 0.001\n",
      "Iter 54/500 - Loss: 0.445   lengthscale: 13.919   noise: 0.001\n",
      "Iter 55/500 - Loss: 0.437   lengthscale: 14.167   noise: 0.001\n",
      "Iter 56/500 - Loss: 0.429   lengthscale: 14.410   noise: 0.001\n",
      "Iter 57/500 - Loss: 0.422   lengthscale: 14.649   noise: 0.001\n",
      "Iter 58/500 - Loss: 0.415   lengthscale: 14.883   noise: 0.001\n",
      "Iter 59/500 - Loss: 0.409   lengthscale: 15.111   noise: 0.001\n",
      "Iter 60/500 - Loss: 0.403   lengthscale: 15.335   noise: 0.001\n",
      "Iter 61/500 - Loss: 0.398   lengthscale: 15.552   noise: 0.001\n",
      "Iter 62/500 - Loss: 0.393   lengthscale: 15.764   noise: 0.001\n",
      "Iter 63/500 - Loss: 0.389   lengthscale: 15.970   noise: 0.001\n",
      "Iter 64/500 - Loss: 0.385   lengthscale: 16.170   noise: 0.000\n",
      "Iter 65/500 - Loss: 0.381   lengthscale: 16.363   noise: 0.000\n",
      "Iter 66/500 - Loss: 0.378   lengthscale: 16.551   noise: 0.000\n",
      "Iter 67/500 - Loss: 0.375   lengthscale: 16.731   noise: 0.000\n",
      "Iter 68/500 - Loss: 0.373   lengthscale: 16.905   noise: 0.000\n",
      "Iter 69/500 - Loss: 0.371   lengthscale: 17.072   noise: 0.000\n",
      "Iter 70/500 - Loss: 0.369   lengthscale: 17.231   noise: 0.000\n",
      "Iter 71/500 - Loss: 0.367   lengthscale: 17.383   noise: 0.000\n",
      "Iter 72/500 - Loss: 0.366   lengthscale: 17.527   noise: 0.000\n",
      "Iter 73/500 - Loss: 0.365   lengthscale: 17.662   noise: 0.000\n",
      "Iter 74/500 - Loss: 0.364   lengthscale: 17.791   noise: 0.000\n",
      "Iter 75/500 - Loss: 0.363   lengthscale: 17.911   noise: 0.000\n",
      "Iter 76/500 - Loss: 0.363   lengthscale: 18.024   noise: 0.000\n",
      "Iter 77/500 - Loss: 0.362   lengthscale: 18.129   noise: 0.000\n",
      "Iter 78/500 - Loss: 0.362   lengthscale: 18.227   noise: 0.000\n",
      "Iter 79/500 - Loss: 0.362   lengthscale: 18.317   noise: 0.000\n",
      "Iter 80/500 - Loss: 0.362   lengthscale: 18.400   noise: 0.000\n",
      "Iter 81/500 - Loss: 0.362   lengthscale: 18.477   noise: 0.000\n",
      "Iter 82/500 - Loss: 0.362   lengthscale: 18.546   noise: 0.000\n",
      "Iter 83/500 - Loss: 0.362   lengthscale: 18.608   noise: 0.000\n",
      "Iter 84/500 - Loss: 0.362   lengthscale: 18.664   noise: 0.000\n",
      "Iter 85/500 - Loss: 0.362   lengthscale: 18.712   noise: 0.000\n",
      "Iter 86/500 - Loss: 0.362   lengthscale: 18.755   noise: 0.000\n",
      "Iter 87/500 - Loss: 0.362   lengthscale: 18.791   noise: 0.000\n",
      "Iter 88/500 - Loss: 0.362   lengthscale: 18.822   noise: 0.000\n",
      "Iter 89/500 - Loss: 0.362   lengthscale: 18.848   noise: 0.000\n",
      "Iter 90/500 - Loss: 0.363   lengthscale: 18.868   noise: 0.000\n",
      "Iter 91/500 - Loss: 0.363   lengthscale: 18.884   noise: 0.000\n",
      "Iter 92/500 - Loss: 0.363   lengthscale: 18.896   noise: 0.000\n",
      "Iter 93/500 - Loss: 0.363   lengthscale: 18.904   noise: 0.000\n",
      "Iter 94/500 - Loss: 0.363   lengthscale: 18.909   noise: 0.000\n",
      "Iter 95/500 - Loss: 0.363   lengthscale: 18.910   noise: 0.000\n",
      "Iter 96/500 - Loss: 0.363   lengthscale: 18.907   noise: 0.000\n",
      "Iter 97/500 - Loss: 0.363   lengthscale: 18.902   noise: 0.000\n",
      "Iter 98/500 - Loss: 0.363   lengthscale: 18.894   noise: 0.000\n",
      "Iter 99/500 - Loss: 0.362   lengthscale: 18.884   noise: 0.000\n",
      "Iter 100/500 - Loss: 0.362   lengthscale: 18.873   noise: 0.000\n",
      "Iter 101/500 - Loss: 0.362   lengthscale: 18.859   noise: 0.000\n",
      "Iter 102/500 - Loss: 0.362   lengthscale: 18.844   noise: 0.000\n",
      "Iter 103/500 - Loss: 0.362   lengthscale: 18.828   noise: 0.000\n",
      "Iter 104/500 - Loss: 0.362   lengthscale: 18.811   noise: 0.000\n",
      "Iter 105/500 - Loss: 0.362   lengthscale: 18.794   noise: 0.000\n",
      "Iter 106/500 - Loss: 0.362   lengthscale: 18.776   noise: 0.000\n",
      "Iter 107/500 - Loss: 0.362   lengthscale: 18.758   noise: 0.000\n",
      "Iter 108/500 - Loss: 0.362   lengthscale: 18.740   noise: 0.000\n",
      "Iter 109/500 - Loss: 0.362   lengthscale: 18.722   noise: 0.000\n",
      "Iter 110/500 - Loss: 0.362   lengthscale: 18.704   noise: 0.000\n",
      "Iter 111/500 - Loss: 0.362   lengthscale: 18.687   noise: 0.000\n",
      "Iter 112/500 - Loss: 0.362   lengthscale: 18.670   noise: 0.000\n",
      "Iter 113/500 - Loss: 0.362   lengthscale: 18.653   noise: 0.000\n",
      "Iter 114/500 - Loss: 0.362   lengthscale: 18.638   noise: 0.000\n",
      "Iter 115/500 - Loss: 0.362   lengthscale: 18.622   noise: 0.000\n",
      "Iter 116/500 - Loss: 0.362   lengthscale: 18.608   noise: 0.000\n",
      "Iter 117/500 - Loss: 0.362   lengthscale: 18.595   noise: 0.000\n",
      "Iter 118/500 - Loss: 0.362   lengthscale: 18.583   noise: 0.000\n",
      "Iter 119/500 - Loss: 0.362   lengthscale: 18.572   noise: 0.000\n",
      "Iter 120/500 - Loss: 0.362   lengthscale: 18.561   noise: 0.000\n",
      "Iter 121/500 - Loss: 0.362   lengthscale: 18.552   noise: 0.000\n",
      "Iter 122/500 - Loss: 0.362   lengthscale: 18.543   noise: 0.000\n",
      "Iter 123/500 - Loss: 0.362   lengthscale: 18.536   noise: 0.000\n",
      "Iter 124/500 - Loss: 0.362   lengthscale: 18.529   noise: 0.000\n",
      "Iter 125/500 - Loss: 0.362   lengthscale: 18.524   noise: 0.000\n",
      "Iter 126/500 - Loss: 0.362   lengthscale: 18.519   noise: 0.000\n",
      "Iter 127/500 - Loss: 0.362   lengthscale: 18.516   noise: 0.000\n",
      "Iter 128/500 - Loss: 0.362   lengthscale: 18.513   noise: 0.000\n",
      "Iter 129/500 - Loss: 0.362   lengthscale: 18.511   noise: 0.000\n",
      "Iter 130/500 - Loss: 0.362   lengthscale: 18.510   noise: 0.000\n",
      "Iter 131/500 - Loss: 0.362   lengthscale: 18.509   noise: 0.000\n",
      "Iter 132/500 - Loss: 0.362   lengthscale: 18.510   noise: 0.000\n",
      "Iter 133/500 - Loss: 0.362   lengthscale: 18.511   noise: 0.000\n",
      "Iter 134/500 - Loss: 0.362   lengthscale: 18.512   noise: 0.000\n",
      "Iter 135/500 - Loss: 0.362   lengthscale: 18.514   noise: 0.000\n",
      "Iter 136/500 - Loss: 0.362   lengthscale: 18.517   noise: 0.000\n",
      "Iter 137/500 - Loss: 0.362   lengthscale: 18.520   noise: 0.000\n",
      "Iter 138/500 - Loss: 0.362   lengthscale: 18.524   noise: 0.000\n",
      "Iter 139/500 - Loss: 0.362   lengthscale: 18.528   noise: 0.000\n",
      "Iter 140/500 - Loss: 0.362   lengthscale: 18.532   noise: 0.000\n",
      "Iter 141/500 - Loss: 0.362   lengthscale: 18.537   noise: 0.000\n",
      "Iter 142/500 - Loss: 0.362   lengthscale: 18.542   noise: 0.000\n",
      "Iter 143/500 - Loss: 0.362   lengthscale: 18.547   noise: 0.000\n",
      "Iter 144/500 - Loss: 0.362   lengthscale: 18.552   noise: 0.000\n",
      "Iter 145/500 - Loss: 0.362   lengthscale: 18.558   noise: 0.000\n",
      "Iter 146/500 - Loss: 0.362   lengthscale: 18.563   noise: 0.000\n",
      "Iter 147/500 - Loss: 0.362   lengthscale: 18.569   noise: 0.000\n",
      "Iter 148/500 - Loss: 0.362   lengthscale: 18.574   noise: 0.000\n",
      "Iter 149/500 - Loss: 0.362   lengthscale: 18.580   noise: 0.000\n",
      "Iter 150/500 - Loss: 0.362   lengthscale: 18.586   noise: 0.000\n",
      "Iter 151/500 - Loss: 0.362   lengthscale: 18.592   noise: 0.000\n",
      "Iter 152/500 - Loss: 0.362   lengthscale: 18.597   noise: 0.000\n",
      "Iter 153/500 - Loss: 0.362   lengthscale: 18.603   noise: 0.000\n",
      "Iter 154/500 - Loss: 0.362   lengthscale: 18.608   noise: 0.000\n",
      "Iter 155/500 - Loss: 0.362   lengthscale: 18.614   noise: 0.000\n",
      "Iter 156/500 - Loss: 0.362   lengthscale: 18.619   noise: 0.000\n",
      "Iter 157/500 - Loss: 0.362   lengthscale: 18.625   noise: 0.000\n",
      "Iter 158/500 - Loss: 0.362   lengthscale: 18.630   noise: 0.000\n",
      "Iter 159/500 - Loss: 0.362   lengthscale: 18.635   noise: 0.000\n",
      "Iter 160/500 - Loss: 0.362   lengthscale: 18.640   noise: 0.000\n",
      "Iter 161/500 - Loss: 0.362   lengthscale: 18.645   noise: 0.000\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Iter 162/500 - Loss: 0.362   lengthscale: 18.650   noise: 0.000\n",
      "Iter 163/500 - Loss: 0.362   lengthscale: 18.655   noise: 0.000\n",
      "Iter 164/500 - Loss: 0.362   lengthscale: 18.659   noise: 0.000\n",
      "Iter 165/500 - Loss: 0.361   lengthscale: 18.664   noise: 0.000\n",
      "Iter 166/500 - Loss: 0.361   lengthscale: 18.669   noise: 0.000\n",
      "Iter 167/500 - Loss: 0.361   lengthscale: 18.673   noise: 0.000\n",
      "Iter 168/500 - Loss: 0.361   lengthscale: 18.678   noise: 0.000\n",
      "Iter 169/500 - Loss: 0.361   lengthscale: 18.682   noise: 0.000\n",
      "Iter 170/500 - Loss: 0.361   lengthscale: 18.686   noise: 0.000\n",
      "Iter 171/500 - Loss: 0.361   lengthscale: 18.691   noise: 0.000\n",
      "Iter 172/500 - Loss: 0.361   lengthscale: 18.695   noise: 0.000\n",
      "Iter 173/500 - Loss: 0.361   lengthscale: 18.699   noise: 0.000\n",
      "Iter 174/500 - Loss: 0.361   lengthscale: 18.704   noise: 0.000\n",
      "Iter 175/500 - Loss: 0.361   lengthscale: 18.708   noise: 0.000\n",
      "Iter 176/500 - Loss: 0.361   lengthscale: 18.713   noise: 0.000\n",
      "Iter 177/500 - Loss: 0.361   lengthscale: 18.717   noise: 0.000\n",
      "Iter 178/500 - Loss: 0.361   lengthscale: 18.721   noise: 0.000\n",
      "Iter 179/500 - Loss: 0.361   lengthscale: 18.726   noise: 0.000\n",
      "Iter 180/500 - Loss: 0.361   lengthscale: 18.731   noise: 0.000\n",
      "Iter 181/500 - Loss: 0.361   lengthscale: 18.735   noise: 0.000\n",
      "Iter 182/500 - Loss: 0.361   lengthscale: 18.740   noise: 0.000\n",
      "Iter 183/500 - Loss: 0.361   lengthscale: 18.745   noise: 0.000\n",
      "Iter 184/500 - Loss: 0.361   lengthscale: 18.750   noise: 0.000\n",
      "Iter 185/500 - Loss: 0.361   lengthscale: 18.755   noise: 0.000\n",
      "Iter 186/500 - Loss: 0.361   lengthscale: 18.760   noise: 0.000\n",
      "Iter 187/500 - Loss: 0.361   lengthscale: 18.765   noise: 0.000\n",
      "Iter 188/500 - Loss: 0.361   lengthscale: 18.771   noise: 0.000\n",
      "Iter 189/500 - Loss: 0.361   lengthscale: 18.776   noise: 0.000\n",
      "Iter 190/500 - Loss: 0.361   lengthscale: 18.782   noise: 0.000\n",
      "Iter 191/500 - Loss: 0.361   lengthscale: 18.788   noise: 0.000\n",
      "Iter 192/500 - Loss: 0.361   lengthscale: 18.794   noise: 0.000\n",
      "Iter 193/500 - Loss: 0.361   lengthscale: 18.800   noise: 0.000\n",
      "Iter 194/500 - Loss: 0.361   lengthscale: 18.806   noise: 0.000\n",
      "Iter 195/500 - Loss: 0.361   lengthscale: 18.813   noise: 0.000\n",
      "Iter 196/500 - Loss: 0.361   lengthscale: 18.820   noise: 0.000\n",
      "Iter 197/500 - Loss: 0.361   lengthscale: 18.827   noise: 0.000\n",
      "Iter 198/500 - Loss: 0.361   lengthscale: 18.834   noise: 0.000\n",
      "Iter 199/500 - Loss: 0.361   lengthscale: 18.841   noise: 0.000\n",
      "Iter 200/500 - Loss: 0.361   lengthscale: 18.848   noise: 0.000\n",
      "Iter 201/500 - Loss: 0.361   lengthscale: 18.856   noise: 0.000\n",
      "Iter 202/500 - Loss: 0.361   lengthscale: 18.863   noise: 0.000\n",
      "Iter 203/500 - Loss: 0.361   lengthscale: 18.871   noise: 0.000\n",
      "Iter 204/500 - Loss: 0.361   lengthscale: 18.880   noise: 0.000\n",
      "Iter 205/500 - Loss: 0.361   lengthscale: 18.888   noise: 0.000\n",
      "Iter 206/500 - Loss: 0.361   lengthscale: 18.897   noise: 0.001\n",
      "Iter 207/500 - Loss: 0.361   lengthscale: 18.905   noise: 0.001\n",
      "Iter 208/500 - Loss: 0.361   lengthscale: 18.914   noise: 0.001\n",
      "Iter 209/500 - Loss: 0.361   lengthscale: 18.924   noise: 0.001\n",
      "Iter 210/500 - Loss: 0.361   lengthscale: 18.933   noise: 0.001\n",
      "Iter 211/500 - Loss: 0.361   lengthscale: 18.943   noise: 0.001\n",
      "Iter 212/500 - Loss: 0.361   lengthscale: 18.953   noise: 0.001\n",
      "Iter 213/500 - Loss: 0.361   lengthscale: 18.963   noise: 0.001\n",
      "Iter 214/500 - Loss: 0.361   lengthscale: 18.974   noise: 0.001\n",
      "Iter 215/500 - Loss: 0.361   lengthscale: 18.985   noise: 0.001\n",
      "Iter 216/500 - Loss: 0.361   lengthscale: 18.996   noise: 0.001\n",
      "Iter 217/500 - Loss: 0.361   lengthscale: 19.007   noise: 0.001\n",
      "Iter 218/500 - Loss: 0.361   lengthscale: 19.019   noise: 0.001\n",
      "Iter 219/500 - Loss: 0.361   lengthscale: 19.030   noise: 0.001\n",
      "Iter 220/500 - Loss: 0.361   lengthscale: 19.043   noise: 0.001\n",
      "Iter 221/500 - Loss: 0.360   lengthscale: 19.055   noise: 0.001\n",
      "Iter 222/500 - Loss: 0.360   lengthscale: 19.068   noise: 0.001\n",
      "Iter 223/500 - Loss: 0.360   lengthscale: 19.081   noise: 0.001\n",
      "Iter 224/500 - Loss: 0.360   lengthscale: 19.095   noise: 0.001\n",
      "Iter 225/500 - Loss: 0.360   lengthscale: 19.109   noise: 0.001\n",
      "Iter 226/500 - Loss: 0.360   lengthscale: 19.123   noise: 0.001\n",
      "Iter 227/500 - Loss: 0.360   lengthscale: 19.138   noise: 0.001\n",
      "Iter 228/500 - Loss: 0.360   lengthscale: 19.153   noise: 0.001\n",
      "Iter 229/500 - Loss: 0.360   lengthscale: 19.169   noise: 0.001\n",
      "Iter 230/500 - Loss: 0.360   lengthscale: 19.185   noise: 0.001\n",
      "Iter 231/500 - Loss: 0.360   lengthscale: 19.202   noise: 0.001\n",
      "Iter 232/500 - Loss: 0.360   lengthscale: 19.219   noise: 0.001\n",
      "Iter 233/500 - Loss: 0.360   lengthscale: 19.236   noise: 0.001\n",
      "Iter 234/500 - Loss: 0.360   lengthscale: 19.254   noise: 0.001\n",
      "Iter 235/500 - Loss: 0.360   lengthscale: 19.273   noise: 0.001\n",
      "Iter 236/500 - Loss: 0.360   lengthscale: 19.292   noise: 0.001\n",
      "Iter 237/500 - Loss: 0.360   lengthscale: 19.311   noise: 0.001\n",
      "Iter 238/500 - Loss: 0.359   lengthscale: 19.332   noise: 0.001\n",
      "Iter 239/500 - Loss: 0.359   lengthscale: 19.353   noise: 0.001\n",
      "Iter 240/500 - Loss: 0.359   lengthscale: 19.374   noise: 0.001\n",
      "Iter 241/500 - Loss: 0.359   lengthscale: 19.396   noise: 0.001\n",
      "Iter 242/500 - Loss: 0.359   lengthscale: 19.419   noise: 0.001\n",
      "Iter 243/500 - Loss: 0.359   lengthscale: 19.443   noise: 0.001\n",
      "Iter 244/500 - Loss: 0.359   lengthscale: 19.467   noise: 0.001\n",
      "Iter 245/500 - Loss: 0.359   lengthscale: 19.492   noise: 0.001\n",
      "Iter 246/500 - Loss: 0.359   lengthscale: 19.518   noise: 0.001\n",
      "Iter 247/500 - Loss: 0.358   lengthscale: 19.545   noise: 0.001\n",
      "Iter 248/500 - Loss: 0.358   lengthscale: 19.572   noise: 0.001\n",
      "Iter 249/500 - Loss: 0.358   lengthscale: 19.600   noise: 0.001\n",
      "Iter 250/500 - Loss: 0.358   lengthscale: 19.630   noise: 0.001\n",
      "Iter 251/500 - Loss: 0.358   lengthscale: 19.660   noise: 0.001\n",
      "Iter 252/500 - Loss: 0.358   lengthscale: 19.690   noise: 0.001\n",
      "Iter 253/500 - Loss: 0.358   lengthscale: 19.722   noise: 0.001\n",
      "Iter 254/500 - Loss: 0.357   lengthscale: 19.755   noise: 0.001\n",
      "Iter 255/500 - Loss: 0.357   lengthscale: 19.789   noise: 0.001\n",
      "Iter 256/500 - Loss: 0.357   lengthscale: 19.823   noise: 0.001\n",
      "Iter 257/500 - Loss: 0.357   lengthscale: 19.859   noise: 0.001\n",
      "Iter 258/500 - Loss: 0.357   lengthscale: 19.896   noise: 0.001\n",
      "Iter 259/500 - Loss: 0.357   lengthscale: 19.933   noise: 0.001\n",
      "Iter 260/500 - Loss: 0.356   lengthscale: 19.972   noise: 0.001\n",
      "Iter 261/500 - Loss: 0.356   lengthscale: 20.012   noise: 0.001\n",
      "Iter 262/500 - Loss: 0.356   lengthscale: 20.052   noise: 0.001\n",
      "Iter 263/500 - Loss: 0.356   lengthscale: 20.094   noise: 0.002\n",
      "Iter 264/500 - Loss: 0.356   lengthscale: 20.136   noise: 0.002\n",
      "Iter 265/500 - Loss: 0.355   lengthscale: 20.179   noise: 0.002\n",
      "Iter 266/500 - Loss: 0.355   lengthscale: 20.224   noise: 0.002\n",
      "Iter 267/500 - Loss: 0.355   lengthscale: 20.269   noise: 0.002\n",
      "Iter 268/500 - Loss: 0.355   lengthscale: 20.315   noise: 0.002\n",
      "Iter 269/500 - Loss: 0.355   lengthscale: 20.362   noise: 0.002\n",
      "Iter 270/500 - Loss: 0.354   lengthscale: 20.409   noise: 0.002\n",
      "Iter 271/500 - Loss: 0.354   lengthscale: 20.458   noise: 0.002\n",
      "Iter 272/500 - Loss: 0.354   lengthscale: 20.507   noise: 0.002\n",
      "Iter 273/500 - Loss: 0.354   lengthscale: 20.557   noise: 0.002\n",
      "Iter 274/500 - Loss: 0.354   lengthscale: 20.607   noise: 0.002\n",
      "Iter 275/500 - Loss: 0.353   lengthscale: 20.658   noise: 0.002\n",
      "Iter 276/500 - Loss: 0.353   lengthscale: 20.709   noise: 0.002\n",
      "Iter 277/500 - Loss: 0.353   lengthscale: 20.761   noise: 0.002\n",
      "Iter 278/500 - Loss: 0.353   lengthscale: 20.813   noise: 0.002\n",
      "Iter 279/500 - Loss: 0.353   lengthscale: 20.866   noise: 0.002\n",
      "Iter 280/500 - Loss: 0.352   lengthscale: 20.919   noise: 0.002\n",
      "Iter 281/500 - Loss: 0.352   lengthscale: 20.972   noise: 0.002\n",
      "Iter 282/500 - Loss: 0.352   lengthscale: 21.025   noise: 0.002\n",
      "Iter 283/500 - Loss: 0.352   lengthscale: 21.078   noise: 0.002\n",
      "Iter 284/500 - Loss: 0.352   lengthscale: 21.132   noise: 0.002\n",
      "Iter 285/500 - Loss: 0.352   lengthscale: 21.185   noise: 0.002\n",
      "Iter 286/500 - Loss: 0.351   lengthscale: 21.239   noise: 0.003\n",
      "Iter 287/500 - Loss: 0.351   lengthscale: 21.292   noise: 0.003\n",
      "Iter 288/500 - Loss: 0.351   lengthscale: 21.345   noise: 0.003\n",
      "Iter 289/500 - Loss: 0.351   lengthscale: 21.398   noise: 0.003\n",
      "Iter 290/500 - Loss: 0.351   lengthscale: 21.451   noise: 0.003\n",
      "Iter 291/500 - Loss: 0.350   lengthscale: 21.504   noise: 0.003\n",
      "Iter 292/500 - Loss: 0.350   lengthscale: 21.556   noise: 0.003\n",
      "Iter 293/500 - Loss: 0.350   lengthscale: 21.608   noise: 0.003\n",
      "Iter 294/500 - Loss: 0.350   lengthscale: 21.661   noise: 0.003\n",
      "Iter 295/500 - Loss: 0.350   lengthscale: 21.712   noise: 0.003\n",
      "Iter 296/500 - Loss: 0.350   lengthscale: 21.764   noise: 0.003\n",
      "Iter 297/500 - Loss: 0.349   lengthscale: 21.815   noise: 0.003\n",
      "Iter 298/500 - Loss: 0.349   lengthscale: 21.867   noise: 0.003\n",
      "Iter 299/500 - Loss: 0.349   lengthscale: 21.918   noise: 0.003\n",
      "Iter 300/500 - Loss: 0.349   lengthscale: 21.968   noise: 0.003\n",
      "Iter 301/500 - Loss: 0.349   lengthscale: 22.019   noise: 0.003\n",
      "Iter 302/500 - Loss: 0.349   lengthscale: 22.070   noise: 0.003\n",
      "Iter 303/500 - Loss: 0.348   lengthscale: 22.120   noise: 0.003\n",
      "Iter 304/500 - Loss: 0.348   lengthscale: 22.170   noise: 0.003\n",
      "Iter 305/500 - Loss: 0.348   lengthscale: 22.220   noise: 0.003\n",
      "Iter 306/500 - Loss: 0.348   lengthscale: 22.270   noise: 0.003\n",
      "Iter 307/500 - Loss: 0.348   lengthscale: 22.320   noise: 0.003\n",
      "Iter 308/500 - Loss: 0.348   lengthscale: 22.370   noise: 0.003\n",
      "Iter 309/500 - Loss: 0.347   lengthscale: 22.419   noise: 0.004\n",
      "Iter 310/500 - Loss: 0.347   lengthscale: 22.469   noise: 0.004\n",
      "Iter 311/500 - Loss: 0.347   lengthscale: 22.519   noise: 0.004\n",
      "Iter 312/500 - Loss: 0.347   lengthscale: 22.568   noise: 0.004\n",
      "Iter 313/500 - Loss: 0.347   lengthscale: 22.618   noise: 0.004\n",
      "Iter 314/500 - Loss: 0.347   lengthscale: 22.668   noise: 0.004\n",
      "Iter 315/500 - Loss: 0.347   lengthscale: 22.717   noise: 0.004\n",
      "Iter 316/500 - Loss: 0.346   lengthscale: 22.767   noise: 0.004\n",
      "Iter 317/500 - Loss: 0.346   lengthscale: 22.817   noise: 0.004\n",
      "Iter 318/500 - Loss: 0.346   lengthscale: 22.867   noise: 0.004\n",
      "Iter 319/500 - Loss: 0.346   lengthscale: 22.916   noise: 0.004\n",
      "Iter 320/500 - Loss: 0.346   lengthscale: 22.966   noise: 0.004\n",
      "Iter 321/500 - Loss: 0.346   lengthscale: 23.016   noise: 0.004\n",
      "Iter 322/500 - Loss: 0.346   lengthscale: 23.066   noise: 0.004\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Iter 323/500 - Loss: 0.345   lengthscale: 23.116   noise: 0.004\n",
      "Iter 324/500 - Loss: 0.345   lengthscale: 23.166   noise: 0.004\n",
      "Iter 325/500 - Loss: 0.345   lengthscale: 23.217   noise: 0.004\n",
      "Iter 326/500 - Loss: 0.345   lengthscale: 23.267   noise: 0.004\n",
      "Iter 327/500 - Loss: 0.345   lengthscale: 23.318   noise: 0.004\n",
      "Iter 328/500 - Loss: 0.345   lengthscale: 23.369   noise: 0.004\n",
      "Iter 329/500 - Loss: 0.344   lengthscale: 23.420   noise: 0.005\n",
      "Iter 330/500 - Loss: 0.344   lengthscale: 23.471   noise: 0.005\n",
      "Iter 331/500 - Loss: 0.344   lengthscale: 23.523   noise: 0.005\n",
      "Iter 332/500 - Loss: 0.344   lengthscale: 23.574   noise: 0.005\n",
      "Iter 333/500 - Loss: 0.344   lengthscale: 23.626   noise: 0.005\n",
      "Iter 334/500 - Loss: 0.344   lengthscale: 23.679   noise: 0.005\n",
      "Iter 335/500 - Loss: 0.343   lengthscale: 23.731   noise: 0.005\n",
      "Iter 336/500 - Loss: 0.343   lengthscale: 23.784   noise: 0.005\n",
      "Iter 337/500 - Loss: 0.343   lengthscale: 23.837   noise: 0.005\n",
      "Iter 338/500 - Loss: 0.343   lengthscale: 23.891   noise: 0.005\n",
      "Iter 339/500 - Loss: 0.343   lengthscale: 23.945   noise: 0.005\n",
      "Iter 340/500 - Loss: 0.343   lengthscale: 23.999   noise: 0.005\n",
      "Iter 341/500 - Loss: 0.342   lengthscale: 24.053   noise: 0.005\n",
      "Iter 342/500 - Loss: 0.342   lengthscale: 24.108   noise: 0.005\n",
      "Iter 343/500 - Loss: 0.342   lengthscale: 24.164   noise: 0.005\n",
      "Iter 344/500 - Loss: 0.342   lengthscale: 24.219   noise: 0.005\n",
      "Iter 345/500 - Loss: 0.342   lengthscale: 24.276   noise: 0.005\n",
      "Iter 346/500 - Loss: 0.341   lengthscale: 24.332   noise: 0.005\n",
      "Iter 347/500 - Loss: 0.341   lengthscale: 24.389   noise: 0.006\n",
      "Iter 348/500 - Loss: 0.341   lengthscale: 24.447   noise: 0.006\n",
      "Iter 349/500 - Loss: 0.341   lengthscale: 24.505   noise: 0.006\n",
      "Iter 350/500 - Loss: 0.341   lengthscale: 24.563   noise: 0.006\n",
      "Iter 351/500 - Loss: 0.340   lengthscale: 24.622   noise: 0.006\n",
      "Iter 352/500 - Loss: 0.340   lengthscale: 24.681   noise: 0.006\n",
      "Iter 353/500 - Loss: 0.340   lengthscale: 24.741   noise: 0.006\n",
      "Iter 354/500 - Loss: 0.340   lengthscale: 24.801   noise: 0.006\n",
      "Iter 355/500 - Loss: 0.340   lengthscale: 24.862   noise: 0.006\n",
      "Iter 356/500 - Loss: 0.339   lengthscale: 24.924   noise: 0.006\n",
      "Iter 357/500 - Loss: 0.339   lengthscale: 24.985   noise: 0.006\n",
      "Iter 358/500 - Loss: 0.339   lengthscale: 25.048   noise: 0.006\n",
      "Iter 359/500 - Loss: 0.339   lengthscale: 25.111   noise: 0.006\n",
      "Iter 360/500 - Loss: 0.338   lengthscale: 25.174   noise: 0.006\n",
      "Iter 361/500 - Loss: 0.338   lengthscale: 25.239   noise: 0.006\n",
      "Iter 362/500 - Loss: 0.338   lengthscale: 25.303   noise: 0.006\n",
      "Iter 363/500 - Loss: 0.338   lengthscale: 25.369   noise: 0.007\n",
      "Iter 364/500 - Loss: 0.337   lengthscale: 25.435   noise: 0.007\n",
      "Iter 365/500 - Loss: 0.337   lengthscale: 25.501   noise: 0.007\n",
      "Iter 366/500 - Loss: 0.337   lengthscale: 25.568   noise: 0.007\n",
      "Iter 367/500 - Loss: 0.337   lengthscale: 25.636   noise: 0.007\n",
      "Iter 368/500 - Loss: 0.336   lengthscale: 25.704   noise: 0.007\n",
      "Iter 369/500 - Loss: 0.336   lengthscale: 25.773   noise: 0.007\n",
      "Iter 370/500 - Loss: 0.336   lengthscale: 25.842   noise: 0.007\n",
      "Iter 371/500 - Loss: 0.335   lengthscale: 25.913   noise: 0.007\n",
      "Iter 372/500 - Loss: 0.335   lengthscale: 25.983   noise: 0.007\n",
      "Iter 373/500 - Loss: 0.335   lengthscale: 26.055   noise: 0.007\n",
      "Iter 374/500 - Loss: 0.335   lengthscale: 26.127   noise: 0.007\n",
      "Iter 375/500 - Loss: 0.334   lengthscale: 26.199   noise: 0.007\n",
      "Iter 376/500 - Loss: 0.334   lengthscale: 26.273   noise: 0.007\n",
      "Iter 377/500 - Loss: 0.334   lengthscale: 26.347   noise: 0.007\n",
      "Iter 378/500 - Loss: 0.333   lengthscale: 26.421   noise: 0.008\n",
      "Iter 379/500 - Loss: 0.333   lengthscale: 26.496   noise: 0.008\n",
      "Iter 380/500 - Loss: 0.333   lengthscale: 26.572   noise: 0.008\n",
      "Iter 381/500 - Loss: 0.332   lengthscale: 26.649   noise: 0.008\n",
      "Iter 382/500 - Loss: 0.332   lengthscale: 26.726   noise: 0.008\n",
      "Iter 383/500 - Loss: 0.332   lengthscale: 26.804   noise: 0.008\n",
      "Iter 384/500 - Loss: 0.331   lengthscale: 26.882   noise: 0.008\n",
      "Iter 385/500 - Loss: 0.331   lengthscale: 26.961   noise: 0.008\n",
      "Iter 386/500 - Loss: 0.331   lengthscale: 27.041   noise: 0.008\n",
      "Iter 387/500 - Loss: 0.330   lengthscale: 27.121   noise: 0.008\n",
      "Iter 388/500 - Loss: 0.330   lengthscale: 27.202   noise: 0.008\n",
      "Iter 389/500 - Loss: 0.330   lengthscale: 27.284   noise: 0.008\n",
      "Iter 390/500 - Loss: 0.329   lengthscale: 27.366   noise: 0.008\n",
      "Iter 391/500 - Loss: 0.329   lengthscale: 27.449   noise: 0.008\n",
      "Iter 392/500 - Loss: 0.328   lengthscale: 27.532   noise: 0.008\n",
      "Iter 393/500 - Loss: 0.328   lengthscale: 27.616   noise: 0.009\n",
      "Iter 394/500 - Loss: 0.328   lengthscale: 27.701   noise: 0.009\n",
      "Iter 395/500 - Loss: 0.327   lengthscale: 27.786   noise: 0.009\n",
      "Iter 396/500 - Loss: 0.327   lengthscale: 27.872   noise: 0.009\n",
      "Iter 397/500 - Loss: 0.326   lengthscale: 27.959   noise: 0.009\n",
      "Iter 398/500 - Loss: 0.326   lengthscale: 28.046   noise: 0.009\n",
      "Iter 399/500 - Loss: 0.326   lengthscale: 28.133   noise: 0.009\n",
      "Iter 400/500 - Loss: 0.325   lengthscale: 28.222   noise: 0.009\n",
      "Iter 401/500 - Loss: 0.325   lengthscale: 28.311   noise: 0.009\n",
      "Iter 402/500 - Loss: 0.324   lengthscale: 28.400   noise: 0.009\n",
      "Iter 403/500 - Loss: 0.324   lengthscale: 28.490   noise: 0.009\n",
      "Iter 404/500 - Loss: 0.323   lengthscale: 28.580   noise: 0.009\n",
      "Iter 405/500 - Loss: 0.323   lengthscale: 28.672   noise: 0.009\n",
      "Iter 406/500 - Loss: 0.323   lengthscale: 28.763   noise: 0.009\n",
      "Iter 407/500 - Loss: 0.322   lengthscale: 28.855   noise: 0.009\n",
      "Iter 408/500 - Loss: 0.322   lengthscale: 28.948   noise: 0.010\n",
      "Iter 409/500 - Loss: 0.321   lengthscale: 29.041   noise: 0.010\n",
      "Iter 410/500 - Loss: 0.321   lengthscale: 29.135   noise: 0.010\n",
      "Iter 411/500 - Loss: 0.320   lengthscale: 29.229   noise: 0.010\n",
      "Iter 412/500 - Loss: 0.320   lengthscale: 29.324   noise: 0.010\n",
      "Iter 413/500 - Loss: 0.319   lengthscale: 29.419   noise: 0.010\n",
      "Iter 414/500 - Loss: 0.319   lengthscale: 29.515   noise: 0.010\n",
      "Iter 415/500 - Loss: 0.318   lengthscale: 29.611   noise: 0.010\n",
      "Iter 416/500 - Loss: 0.318   lengthscale: 29.708   noise: 0.010\n",
      "Iter 417/500 - Loss: 0.317   lengthscale: 29.805   noise: 0.010\n",
      "Iter 418/500 - Loss: 0.317   lengthscale: 29.903   noise: 0.010\n",
      "Iter 419/500 - Loss: 0.316   lengthscale: 30.001   noise: 0.010\n",
      "Iter 420/500 - Loss: 0.316   lengthscale: 30.099   noise: 0.010\n",
      "Iter 421/500 - Loss: 0.315   lengthscale: 30.198   noise: 0.010\n",
      "Iter 422/500 - Loss: 0.315   lengthscale: 30.297   noise: 0.010\n",
      "Iter 423/500 - Loss: 0.314   lengthscale: 30.397   noise: 0.010\n",
      "Iter 424/500 - Loss: 0.314   lengthscale: 30.497   noise: 0.010\n",
      "Iter 425/500 - Loss: 0.313   lengthscale: 30.598   noise: 0.011\n",
      "Iter 426/500 - Loss: 0.313   lengthscale: 30.699   noise: 0.011\n",
      "Iter 427/500 - Loss: 0.312   lengthscale: 30.800   noise: 0.011\n",
      "Iter 428/500 - Loss: 0.312   lengthscale: 30.901   noise: 0.011\n",
      "Iter 429/500 - Loss: 0.311   lengthscale: 31.003   noise: 0.011\n",
      "Iter 430/500 - Loss: 0.311   lengthscale: 31.106   noise: 0.011\n",
      "Iter 431/500 - Loss: 0.310   lengthscale: 31.209   noise: 0.011\n",
      "Iter 432/500 - Loss: 0.310   lengthscale: 31.312   noise: 0.011\n",
      "Iter 433/500 - Loss: 0.309   lengthscale: 31.415   noise: 0.011\n",
      "Iter 434/500 - Loss: 0.309   lengthscale: 31.519   noise: 0.011\n",
      "Iter 435/500 - Loss: 0.308   lengthscale: 31.623   noise: 0.011\n",
      "Iter 436/500 - Loss: 0.308   lengthscale: 31.727   noise: 0.011\n",
      "Iter 437/500 - Loss: 0.307   lengthscale: 31.832   noise: 0.011\n",
      "Iter 438/500 - Loss: 0.306   lengthscale: 31.936   noise: 0.011\n",
      "Iter 439/500 - Loss: 0.306   lengthscale: 32.042   noise: 0.011\n",
      "Iter 440/500 - Loss: 0.305   lengthscale: 32.147   noise: 0.011\n",
      "Iter 441/500 - Loss: 0.305   lengthscale: 32.253   noise: 0.011\n",
      "Iter 442/500 - Loss: 0.304   lengthscale: 32.359   noise: 0.011\n",
      "Iter 443/500 - Loss: 0.304   lengthscale: 32.465   noise: 0.011\n",
      "Iter 444/500 - Loss: 0.303   lengthscale: 32.571   noise: 0.012\n",
      "Iter 445/500 - Loss: 0.303   lengthscale: 32.678   noise: 0.012\n",
      "Iter 446/500 - Loss: 0.302   lengthscale: 32.785   noise: 0.012\n",
      "Iter 447/500 - Loss: 0.301   lengthscale: 32.892   noise: 0.012\n",
      "Iter 448/500 - Loss: 0.301   lengthscale: 32.999   noise: 0.012\n",
      "Iter 449/500 - Loss: 0.300   lengthscale: 33.106   noise: 0.012\n",
      "Iter 450/500 - Loss: 0.300   lengthscale: 33.214   noise: 0.012\n",
      "Iter 451/500 - Loss: 0.299   lengthscale: 33.322   noise: 0.012\n",
      "Iter 452/500 - Loss: 0.299   lengthscale: 33.430   noise: 0.012\n",
      "Iter 453/500 - Loss: 0.298   lengthscale: 33.538   noise: 0.012\n",
      "Iter 454/500 - Loss: 0.298   lengthscale: 33.646   noise: 0.012\n",
      "Iter 455/500 - Loss: 0.297   lengthscale: 33.755   noise: 0.012\n",
      "Iter 456/500 - Loss: 0.296   lengthscale: 33.863   noise: 0.012\n",
      "Iter 457/500 - Loss: 0.296   lengthscale: 33.972   noise: 0.012\n",
      "Iter 458/500 - Loss: 0.295   lengthscale: 34.081   noise: 0.012\n",
      "Iter 459/500 - Loss: 0.295   lengthscale: 34.190   noise: 0.012\n",
      "Iter 460/500 - Loss: 0.294   lengthscale: 34.299   noise: 0.012\n",
      "Iter 461/500 - Loss: 0.294   lengthscale: 34.408   noise: 0.012\n",
      "Iter 462/500 - Loss: 0.293   lengthscale: 34.517   noise: 0.012\n",
      "Iter 463/500 - Loss: 0.292   lengthscale: 34.626   noise: 0.012\n",
      "Iter 464/500 - Loss: 0.292   lengthscale: 34.736   noise: 0.012\n",
      "Iter 465/500 - Loss: 0.291   lengthscale: 34.845   noise: 0.012\n",
      "Iter 466/500 - Loss: 0.291   lengthscale: 34.955   noise: 0.012\n",
      "Iter 467/500 - Loss: 0.290   lengthscale: 35.064   noise: 0.012\n",
      "Iter 468/500 - Loss: 0.290   lengthscale: 35.174   noise: 0.012\n",
      "Iter 469/500 - Loss: 0.289   lengthscale: 35.284   noise: 0.013\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Iter 470/500 - Loss: 0.289   lengthscale: 35.394   noise: 0.013\n",
      "Iter 471/500 - Loss: 0.288   lengthscale: 35.503   noise: 0.013\n",
      "Iter 472/500 - Loss: 0.287   lengthscale: 35.613   noise: 0.013\n",
      "Iter 473/500 - Loss: 0.287   lengthscale: 35.723   noise: 0.013\n",
      "Iter 474/500 - Loss: 0.286   lengthscale: 35.833   noise: 0.013\n",
      "Iter 475/500 - Loss: 0.286   lengthscale: 35.943   noise: 0.013\n",
      "Iter 476/500 - Loss: 0.285   lengthscale: 36.052   noise: 0.013\n",
      "Iter 477/500 - Loss: 0.285   lengthscale: 36.162   noise: 0.013\n",
      "Iter 478/500 - Loss: 0.284   lengthscale: 36.272   noise: 0.013\n",
      "Iter 479/500 - Loss: 0.284   lengthscale: 36.382   noise: 0.013\n",
      "Iter 480/500 - Loss: 0.283   lengthscale: 36.491   noise: 0.013\n",
      "Iter 481/500 - Loss: 0.282   lengthscale: 36.601   noise: 0.013\n",
      "Iter 482/500 - Loss: 0.282   lengthscale: 36.711   noise: 0.013\n",
      "Iter 483/500 - Loss: 0.281   lengthscale: 36.821   noise: 0.013\n",
      "Iter 484/500 - Loss: 0.281   lengthscale: 36.930   noise: 0.013\n",
      "Iter 485/500 - Loss: 0.280   lengthscale: 37.040   noise: 0.013\n",
      "Iter 486/500 - Loss: 0.280   lengthscale: 37.149   noise: 0.013\n",
      "Iter 487/500 - Loss: 0.279   lengthscale: 37.259   noise: 0.013\n",
      "Iter 488/500 - Loss: 0.279   lengthscale: 37.368   noise: 0.013\n",
      "Iter 489/500 - Loss: 0.278   lengthscale: 37.477   noise: 0.013\n",
      "Iter 490/500 - Loss: 0.278   lengthscale: 37.586   noise: 0.013\n",
      "Iter 491/500 - Loss: 0.277   lengthscale: 37.695   noise: 0.013\n",
      "Iter 492/500 - Loss: 0.276   lengthscale: 37.804   noise: 0.013\n",
      "Iter 493/500 - Loss: 0.276   lengthscale: 37.913   noise: 0.013\n",
      "Iter 494/500 - Loss: 0.275   lengthscale: 38.022   noise: 0.013\n",
      "Iter 495/500 - Loss: 0.275   lengthscale: 38.131   noise: 0.013\n",
      "Iter 496/500 - Loss: 0.274   lengthscale: 38.239   noise: 0.013\n",
      "Iter 497/500 - Loss: 0.274   lengthscale: 38.348   noise: 0.013\n",
      "Iter 498/500 - Loss: 0.273   lengthscale: 38.456   noise: 0.013\n",
      "Iter 499/500 - Loss: 0.273   lengthscale: 38.564   noise: 0.013\n",
      "Iter 500/500 - Loss: 0.272   lengthscale: 38.672   noise: 0.013\n"
     ]
    }
   ],
   "source": [
    "# this is for running the notebook in our testing framework\n",
    "import os\n",
    "smoke_test = ('CI' in os.environ)\n",
    "training_iter = 2 if smoke_test else 500\n",
    "\n",
    "\n",
    "# Find optimal model hyperparameters\n",
    "model.train()\n",
    "likelihood.train()\n",
    "\n",
    "# Use the adam optimizer\n",
    "optimizer = torch.optim.Adam(model.parameters(), lr=0.25)  # Includes GaussianLikelihood parameters\n",
    "\n",
    "# \"Loss\" for GPs - the marginal log likelihood\n",
    "mll = gpytorch.mlls.ExactMarginalLogLikelihood(likelihood, model)\n",
    "\n",
    "for i in range(training_iter):\n",
    "    # Zero gradients from previous iteration\n",
    "    optimizer.zero_grad()\n",
    "    # Output from model\n",
    "    output = model(train_x_distributional)\n",
    "    # Calc loss and backprop gradients\n",
    "    loss = -mll(output, train_y)\n",
    "    loss.backward()\n",
    "    print('Iter %d/%d - Loss: %.3f   lengthscale: %.3f   noise: %.3f' % (\n",
    "        i + 1, training_iter, loss.item(),\n",
    "        model.covar_module.base_kernel.lengthscale.item(),\n",
    "        model.likelihood.noise.item()\n",
    "    ))\n",
    "    optimizer.step()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now, we test predictions. For simplicity, we will assume a fixed variance of $0.01.$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAeIAAADGCAYAAAAOo4M3AAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjMsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+AADFEAAAgAElEQVR4nO3deXic1Xnw/++ZfR/tsmTZeJO8yFjyisEGm50QFxogL/A2hJC2FAiUtCW/NCWJSRPStKUhbQOkNKHZCDgQUgiBF0zAgAETL9jGNt4ty7Jl7RrNvp7fHyML20iyLI00I+n+XNdzzfbMM2ceaeaes91Haa0RQgghRHYYsl0AIYQQYjyTQCyEEEJkkQRiIYQQIoskEAshhBBZJIFYCCGEyCIJxEIIIUQWDTkQK6VsSqk/KqW2KaV2KqW+lYmCCSGEEOOBGuo8YqWUApxa64BSygysB+7VWm/IRAGFEEKIscw01APodCQPdN80d2+SJUQIIYQYgIz0ESuljEqprUAzsFZr/X4mjiuEEEKMdUOuEQNorZNArVIqD/itUmqu1nrHyfsopW4HbgdwOp0LZ82alYmXFkIIIXLe5s2bW7XWxb09NuQ+4k8cUKnVQFBr/VBf+yxatEhv2rQpo68rhBBC5Cql1Gat9aLeHsvEqOni7powSik7cBmwe6jHFUIIIcaDTDRNlwE/U0oZSQf2X2utX8zAcYUQQogxLxOjprcD8zNQFiGEEGLcychgLSGEEJkVj8dpaGggEolkuyjiLNhsNioqKjCbzQN+jgRiIYTIQQ0NDbjdbqZMmUI6b5LIdVpr2traaGhoYOrUqQN+nuSaFkKIHBSJRCgsLJQgPIoopSgsLDzrVgwJxEIIkaMkCI8+g/mbSSAWQgjRq4aGBq699loqKyuZPn069957L7FYDICf/vSn3H333Vku4Se5XK5e7zcajdTW1lJdXU1NTQ3f//73SaVS/R6rrq6OX/3qV8NRzFNIIBZCCPEJWmuuu+46/vRP/5R9+/axd+9eAoEA999//7C9ZiKRGLZj2+12tm7dys6dO1m7di0vvfQS3/pW/4sFSiAWQgiRNa+//jo2m43bbrsNSNcoH374YZ544glCoRAAR44c4aqrrmLmzJk9QS0YDPLpT3+ampoa5s6dy5o1awDYvHkzK1asYOHChVx55ZU0NjYCsHLlSv7hH/6BFStW8OCDDzJlypSemmooFGLSpEnE43EOHDjAVVddxcKFC7nwwgvZvTudN+rQoUOcf/75LF68mG984xsDem8lJSU8/vjj/PCHP0RrTV1dHRdeeCELFixgwYIFvPvuuwD8/d//PW+//Ta1tbU8/PDDfe43VDJqWgghctyXvwxbt2b2mLW18IMf9P34zp07Wbhw4Sn3eTweJk+ezP79+wH44x//yI4dO3A4HCxevJhPf/rTHD58mPLycn7/+98D4PP5iMfj3HPPPTz//PMUFxezZs0a7r//fp544gkAOjs7efPNNwHYsmULb775JhdffDG/+93vuPLKKzGbzdx+++386Ec/orKykvfff5+77rqL119/nXvvvZc777yTz3/+8zzyyCMDfv/Tpk0jlUrR3NxMSUkJa9euxWazsW/fPm6++WY2bdrE9773PR566CFefDGdoyoUCvW631BJIBZCCPEJWuteBx6dfP/ll19OYWEhANdddx3r16/n6quv5r777uOrX/0qq1at4sILL2THjh3s2LGDyy+/HIBkMklZWVnPMW+88cZTrq9Zs4aLL76Yp59+mrvuuotAIMC7777LZz/72Z79otEoAO+88w6/+c1vALjlllv46le/elbvEdJztu+++262bt2K0Whk7969ve4/0P3OlgRiIYTIcf3VXIdLdXV1T4A7oauriyNHjjB9+nQ2b978iUCtlKKqqorNmzfz0ksv8bWvfY0rrriCz3zmM1RXV/Pee+/1+lpOp7Pn+jXXXMPXvvY12tvb2bx5M5dccgnBYJC8vDy29tEsMJiRygcPHsRoNFJSUsK3vvUtSktL2bZtG6lUCpvN1utzHn744QHtd7akj1gIIcQnXHrppYRCIX7+858D6Vrs3/3d3/GFL3wBh8MBwNq1a2lvbyccDvO///u/LFu2jGPHjuFwOPjc5z7Hfffdx5YtW5g5cyYtLS09gTgej7Nz585eX9flcrFkyRLuvfdeVq1ahdFoxOPxMHXqVJ555hkgXZPdtm0bAMuWLePpp58G4MknnxzQe2tpaeGOO+7g7rvvRimFz+ejrKwMg8HAL37xC5LJJAButxu/39/zvL72GyoJxEIIIT5BKcVvf/tbnnnmGSorK6mqqsJms/Hd7363Z5/ly5dzyy23UFtby/XXX8+iRYv48MMPWbJkCbW1tTz44IN8/etfx2Kx8Oyzz/LVr36Vmpoaamtr+x3odOONN/LLX/7ylCbrJ598kp/85CfU1NRQXV3N888/D8C///u/88gjj7B48WJ8Pl+fxwyHwz3Tly677DKuuOIKVq9eDcBdd93Fz372M5YuXcrevXt7aujz5s3DZDJRU1PDww8/3Od+Q5Xx9YgHQtYjFkKI/n300UfMnj0728UQg9Db325Y1yMWQgghxOBJIBZCCCGySAKxEEIIkUUSiIUQQogskkAshBBCZNGQA7FSapJS6g2l1EdKqZ1KqXszUTAhhBBiPMhEjTgB/J3WejawFPiSUmpOBo4rhBAii5RS3HLLLT23E4kExcXFrFq1KoulGnuGnOJSa90INHZf9yulPgImAruGemwhhBBpD6/NTF7jE/7m8qoz7uN0OtmxYwfhcBi73c7atWuZOHFiRsshMtxHrJSaAswH3s/kcYUQQmTHpz71qZ6VlJ566iluvvnmnseCwSBf/OIXWbx4MfPnz+/JdtXXcoHr1q1j5cqV3HDDDcyaNYs/+7M/IxtJpXJNxgKxUsoF/Ab4sta6q5fHb1dKbVJKbWppacnUywohhBhGN910E08//TSRSITt27dz3nnn9Tz24IMPcskll7Bx40beeOMNvvKVrxAMBnuWFdyyZQtr1qzhr//6r3ue88EHH/CDH/yAXbt2cfDgQd55551svK2ckpHVl5RSZtJB+Emt9XO97aO1fhx4HNIpLjPxukIIIYbXvHnzqKur46mnnuLqq68+5bFXX32VF154gYceegiASCRCfX095eXlfS4XuGTJEioqKgCora2lrq6O5cuXj9wbykFDDsQqvf7UT4CPtNbfH3qRhBBC5JJrrrmG++67j3Xr1tHW1tZzv9aa3/zmN8ycOfOU/R944IE+lwu0Wq09141GI4lEYvjfQI7LRNP0MuAW4BKl1Nbu7eozPUkIIcTo8MUvfpFvfvObnHvuuafcf+WVV/Kf//mfPf28H3zwATB8ywWOVUMOxFrr9VprpbWep7Wu7d5eykThhBBCZF9FRQX33vvJFBHf+MY3iMfjzJs3j7lz5/KNb3wD6HtZQdE7WQZRCCFykCyDOHrJMohCCCHEKCKBWAghhMgiCcRCCCFEFkkgFkIIIbJIArEQQgiRRRKIhRBCiCySQCyEEKJPx48f56abbmL69OnMmTOHq6+++pSUlQP19ttvU11dTW1tLUePHuWGG27odb+VK1cy3qa3ZiTXtBCDEUukiCaSxBIpYskUyZQmlYKU1iS1RmtNsvu21mAwgFEpDAaFQanu62A0KEwGA1azAavJgNVkzPZbEyIrGhsbuemmm1izZg0TJkwY8vG01nzmM5/h1ltv5emnnwZg69atNDU1UVV15mUUT/bkk09y3333cdtttwHw7LPPDrl8Y4UEYpExqZQmEEsQjKY3fyRBMJokEE0QiiWIxNOBN5pIEY2nSA1TMhmDUqcEZavJgNNqxGU147KZcFlNuLsvHRYj6XTpQox+3/72t1m/fj3/+I//yKOPPjrk473xxhuYzWbuuOOOnvtqa2vRWvOVr3yFl19+GaUUX//617nxxhtZt24dDzzwAEVFRezYsYOFCxfyy1/+kp/85Cf8+te/5pVXXuG1117jwQcfZNWqVT1rHd92223s2rWL2bNnEw6He17r1VdfZfXq1USjUaZPn87//M//4HK5mDJlCrfeeiu/+93viMfjPPPMM8yaNYtAIMA999zDpk2bUEqxevVqrr/++j6PkyskEIuzkkxpOkMxOkJxfOEYHcE4HaEYnaE4wViCXFhaNKU14ViScCwJxPvd12hQuKwm8hxm8p0WCp0W8h0WCpwWnFb5eIjctnLlSiDd7JtKpXruf+yxx3jssccwGAxceOGFrFu3blDHPxFMT/fcc8+xdetWtm3bRmtrK4sXL+aiiy4C0vmmd+7cSXl5OcuWLeOdd97hL/7iL1i/fj2rVq3ihhtuoK6u7pSyOhwOtm/fzvbt21mwYAEAra2tfOc73+G1117D6XTyz//8z3z/+9/nm9/8JgBFRUVs2bKFRx99lIceeogf//jHfPvb38br9fLhhx8C0NHRccbj5AL5phF98oXjtPijNPsjtPijtAVi+COJYavJZkMypfGF4/jCcQ63hU55zGo2UOCwUOiyUuK2MsFro8hlxWiQGrTILUuWLOHgwYO0traSSqUwGAwUFRUxffr0YXm99evXc/PNN2M0GiktLWXFihVs3LgRj8dz1sscvvXWWz3rFc+bN4958+YBsGHDBnbt2sWyZcsAiMVinH/++T3Pu+666wBYuHAhzz2XXn33tdde62lCB8jPz+fFF1/s9zi5QAKxAMAXitPYFaa5K0qzP0qLP0okPr5XTInGUzT6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\n",
      "text/plain": [
       "<Figure size 576x216 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Get into evaluation (predictive posterior) mode\n",
    "model.eval()\n",
    "likelihood.eval()\n",
    "\n",
    "# Test points are regularly spaced along [0,1]\n",
    "# Make predictions by feeding model through likelihood\n",
    "with torch.no_grad(), gpytorch.settings.fast_pred_var():\n",
    "    test_x = torch.linspace(0, 1, 51)\n",
    "    test_x_distributional = torch.stack((test_x, (1e-2 * torch.ones_like(test_x)).log()), dim=1)\n",
    "    observed_pred = likelihood(model(test_x_distributional))\n",
    "\n",
    "with torch.no_grad():\n",
    "    # Initialize plot\n",
    "    f, ax = plt.subplots(1, 1, figsize=(8, 3))\n",
    "\n",
    "    # Get upper and lower confidence bounds\n",
    "    lower, upper = observed_pred.confidence_region()\n",
    "    # Plot training data as black stars\n",
    "    ax.errorbar(train_x_mean.numpy(), train_y.numpy(), xerr=train_x_stdv, fmt='k*')\n",
    "    # Plot predictive means as blue line\n",
    "    ax.plot(test_x.numpy(), observed_pred.mean.numpy(), 'b')\n",
    "    # Shade between the lower and upper confidence bounds\n",
    "    ax.fill_between(test_x.numpy(), lower.numpy(), upper.numpy(), alpha=0.5)\n",
    "    ax.set_ylim([-3, 3])\n",
    "    ax.legend(['Observed Data', 'Mean', 'Confidence'])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "As a final note, we've made it very easy to extend the distributional kernel class by exposing a generic `DistributionalInputKernel` class that takes as input any distance function over probability distributions."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.7.4 64-bit ('base': conda)",
   "language": "python",
   "name": "python37464bitbaseconda52eab690427c4f7ea56588deee120c46"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
