{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "CrSZiwbewKr_"
   },
   "source": [
    "# Active Learning with Fully Bayesian Gaussian processes\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "import math\n",
    "import torch\n",
    "import gpytorch\n",
    "import pyro\n",
    "from pyro.infer.mcmc import NUTS, MCMC\n",
    "from matplotlib import pyplot as plt\n",
    "import seaborn as sns\n",
    "import scipy\n",
    "from scipy import stats\n",
    "from tqdm import tqdm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {
    "id": "zcepTRVUwKsA"
   },
   "outputs": [],
   "source": [
    "import torch\n",
    "from torch.distributions import constraints\n",
    "from torch.nn import Parameter\n",
    "\n",
    "import pyro \n",
    "import pyro.contrib.gp as gp\n",
    "import pyro.distributions as dist\n",
    "from pyro.contrib.gp.models.model import GPModel\n",
    "from pyro.contrib.gp.util import conditional\n",
    "from pyro.distributions.util import eye_like\n",
    "from pyro.nn.module import PyroParam, pyro_method\n",
    "\n",
    "import numpy as np\n",
    "from scipy.stats import norm, multivariate_normal\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "\n",
    "# matplotlib options\n",
    "%matplotlib inline\n",
    "plt.style.use('ggplot')\n",
    "plt.rcParams['figure.figsize'] = (12, 8)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "hly4mrkewKth"
   },
   "source": [
    "## Generate data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "id": "ZHt9E6JDwKth"
   },
   "outputs": [],
   "source": [
    "# generate some training data\n",
    "X_train = np.linspace(-0.5, 1.5, 2) #np.hstack([-0.75 + np.random.rand(1,2), 0.75 + np.random.rand(1,2)]).T\n",
    "y_train = np.sin(4.0*X_train) + 0.1 * np.random.randn(len(X_train)) #*X_train**3\n",
    "X_test = np.linspace(-0.5,1.5,101) \n",
    "y_test = np.sin(4.0*X_test) + 0.1 * np.random.randn(len(X_test)) # *X_test**3\n",
    "N_train = len(y_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "id": "wIOV7sXtwKtj",
    "outputId": "993da1df-f77c-4967-d179-fe247c4e0d62"
   },
   "outputs": [
    {
     "data": {
      "image/png": "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p70+aseOHSWlKeXem3423fZ64YUX9D//8z9auHCh/uIv/kLt7e2Tfpbrq9Tvf/97DQ8P68ILLyxuC9r1NZXR0VG1t7cXH7e1tWl0dDSQ15ZVO3bs0Gc+85mSbRs3bpQkrVixwpWzPzjl9ddf19q1azVv3jz92Z/9mbq7uy3/ragHX4flDRs26N133520/corryx5HAqFFAqFyr7G5s2b1draqsOHD2v9+vU666yzinW3fmNHe0nS0aNH9cMf/lA33XSTmprev3lx1VVXae7cuZqYmNCDDz6on//858VaJGAqO3fu1FtvvaXvfve7xW3l3psLFixw7iRd4KKLLtIf//Efa8aMGXryySf1wAMP6Dvf+Y7Tp+V6zz33nC655JLi55XE9QV7vPzyy3r66ae1fv364rYNGzaotbVVY2Nj+pu/+RvFYjEtXrzYwbN0h09+8pPavHmzZs2apb179+ruu+/WD37wA6dPS5LPw/Jdd91l+lxLS4uOHj2qefPm6ejRozrjjDPK7lcYKDN//nwtXrxYb7/9tj772c/qxIkTyuVyCofDGh0dLe7nZXa014kTJ5RMJrVq1Sqdd955xe2FXukZM2Zo2bJl+pd/+Rd7T94FWltbS261HTlyZNJ1Udinra1NuVxOJ06cUDQanfSzfrmmKplOe0nSb37zG23btk3f/e53S+5GlHtv+jnMTKe9otFo8f+XL1+un/70p8WfffXVV4vPjY6O+v6P83SvL0natWuXvva1r036eSk419dUWltbNTIyUnxcaM8gXlvT9X//93968MEHdccdd5S8NwvXVktLiy6++GLt37+fNpNKOiKXLFmihx9+WMeOHbP0Xq6XwNYs9/T06Nlnn5UkPfvss7r44osn7XP8+HGNj49Lko4dO6bXXntNXV1dCoVCuuCCC/T8889Len8kcE9PT+NO3gHTaa+JiQndc889uvTSS4v13AVHjx6VpGJ9bnd3d/1PusEWLVqkQ4cOaXh4WBMTE9q1a9ek6+Kiiy7SM888I0l6/vnndcEFFygUCqmnp0e7du3S+Pi4hoeHdejQIf3BH/yBA79F40ynvX7729/qRz/6kfr7+9XS0lLcbvbe9LPptFfhfSa9X2NaaJNPf/rTeumll3T8+HEdP35cL730ku9nW5lOe0lSOp3We++9V/LlPojX11R6enq0c+dOGYah119/XbNnz9a8efMCeW1Nx8jIiO655x7dfPPNisVixe0nT55UNpst/v9vfvOb4qwrQffuu+/K+GDpj/379yufzysajU77vVxPgV2UJJPJaNOmTRoZGSmZCu3NN9/Uk08+qRtvvFGvvfaaHnroITU1NSmfz+tLX/qSPv/5z0uSDh8+rPvuu0/Hjx/XJz/5Sd1yyy2+q8H9qOm0186dO/V3f/d3JX9UClPE/fVf/3VxsN8nPvEJ3XDDDZo1a5ZTv07d7N27Vz/5yU+Uz+e1bNkyxeNxPf7441q0aJF6enp06tQp3X///frtb3+rOXPmaM2aNcW67lQqpaefflpNTU366le/OqnGzY+maq8NGzbonXfe0dy5cyV9OIVXpfemn03VXv/4j/+oPXv2KBwOa86cObr++uvV2dkp6f26yW3btkl6f3qvZcuWOfmrNMRU7SVJ//RP/6Tx8XFdffXVxZ8L4vV133336dVXX1Umk1FLS4u+8pWvaGJiQpL0hS98QYZh6OGHH9ZLL72kmTNnavXq1Vq0aJGkYF5bU7XX3//93+uFF14o1nkXpog7fPiw7rnnHklSLpfT5z73OcXjccd+j0aaqs1+8Ytf6D/+4z8UDoc1c+ZM/fmf/7nOP/98SeXfy40U2LAMAAAATCWwZRgAAADAVAjLAAAAgAnCMgAAAGCCsAwAAACYICwDAAAAJgjLAAAAgAnCMgAAAGCCsAwAAACY+P/mM3A/+wP+7QAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 864x576 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(X_test, y_test, \"ro\")\n",
    "plt.plot(X_train, y_train, \"bo\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "id": "MJjSv8GWgHmX"
   },
   "outputs": [],
   "source": [
    "# Build torch.tensors\n",
    "train_x = torch.from_numpy(X_train).view(-1,)\n",
    "train_y = torch.from_numpy(y_train).view(-1,)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Make GP model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "from gpytorch.priors import LogNormalPrior, NormalPrior, UniformPrior, GammaPrior\n",
    "\n",
    "class ExactGPModel(gpytorch.models.ExactGP):\n",
    "    def __init__(self, train_x, train_y, likelihood):\n",
    "        super(ExactGPModel, self).__init__(train_x, train_y, likelihood)\n",
    "        self.mean_module = gpytorch.means.ConstantMean()\n",
    "        self.covar_module = gpytorch.kernels.ScaleKernel(gpytorch.kernels.RBFKernel())\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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def pyro_model(x, y):\n",
    "    model.pyro_sample_from_prior()\n",
    "    output = model(x)\n",
    "    loss = mll.pyro_factor(output, y)\n",
    "    return y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 168,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.7444)"
      ]
     },
     "execution_count": 168,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#def pyro_guide(x, y):\n",
    "from torch.nn.functional import softplus\n",
    "softplus(torch.tensor(0.1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 205,
   "metadata": {},
   "outputs": [],
   "source": [
    "? gpytorch.models.ExactGP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 176,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 400/400 [00:13<00:00, 29.25it/s]\n"
     ]
    }
   ],
   "source": [
    "### ADVI ###\n",
    "# For ADVI\n",
    "from pyro.infer import SVI, Trace_ELBO, TraceEnum_ELBO, JitTrace_ELBO, TraceMeanField_ELBO\n",
    "from pyro.contrib.autoguide import AutoDiagonalNormal, AutoGuide\n",
    "from pyro.optim import AdamW\n",
    "\n",
    "pyro.clear_param_store()  # clear global parameter cache\n",
    "pyro.set_rng_seed(1)  # set random seed\n",
    "\n",
    "class ExactGPModel(gpytorch.models.ExactGP):\n",
    "    def __init__(self, train_x, train_y, likelihood):\n",
    "        super(ExactGPModel, self).__init__(train_x, train_y, likelihood)\n",
    "        self.likelihood = likelihood\n",
    "        self.mean_module = gpytorch.means.ConstantMean()\n",
    "        self.covar_module = gpytorch.kernels.ScaleKernel(gpytorch.kernels.RBFKernel())\n",
    "        \n",
    "        length_scale_prior = GammaPrior(3, 6)\n",
    "        noise_prior = LogNormalPrior(loc=0, scale=2)\n",
    "        \n",
    "        # Variational Inference\n",
    "        self.covar_module.base_kernel.lengthscale = pyro.sample(\"lengthscale\", length_scale_prior)\n",
    "        self.likelihood.noise_covar.noise = pyro.sample(\"noise\", noise_prior)\n",
    "        \n",
    "        #para_length_scale_prior = NormalPrior(0,1)\n",
    "        #para_noise_prior = NormalPrior(0,1)\n",
    "        #self.covar_module.base_kernel.lengthscale = softplus(pyro.sample(\"lengthscale\", para_length_scale_prior))\n",
    "        #self.likelihood.noise_covar.noise = softplus(pyro.sample(\"noise\", para_noise_prior))\n",
    "        \n",
    "        # MCMC\n",
    "        #self.covar_module.base_kernel.register_prior(\"lengthscale_prior\", length_scale_prior, \"lengthscale\")\n",
    "        #self.likelihood.register_prior(\"noise_prior\", noise_prior, \"noise\")\n",
    "\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",
    "\n",
    "likelihood = gpytorch.likelihoods.GaussianLikelihood(noise_constraint=gpytorch.constraints.Positive())\n",
    "\n",
    "# Automatically define variational distribution (a mean field guide).\n",
    "guide = AutoDiagonalNormal(ExactGPModel)\n",
    "\n",
    "\n",
    "# Create SVI object for optimization.\n",
    "svi = SVI(ExactGPModel, guide, Adam({'lr': 5e-2}), TraceMeanField_ELBO(num_particles=10))\n",
    "\n",
    "\n",
    "\n",
    "# Standardize the data s.t. input is on unit cube and output has zero mean and std=1.\n",
    "min_x = torch.min(train_x)\n",
    "max_x = torch.max(train_x - min_x)\n",
    "train_x_trans = (train_x - min_x) / max_x\n",
    "\n",
    "mu_y = torch.mean(train_y)\n",
    "sigma_y = torch.std(train_y)\n",
    "train_y_trans = (train_y - mu_y) / sigma_y\n",
    "\n",
    "\n",
    "# Do 1000 gradient steps.\n",
    "advi_loss = []\n",
    "for step in tqdm(range(400)):\n",
    "    advi_loss.append(svi.step(train_x_trans.view(-1,), train_y_trans.view(-1,), likelihood))\n",
    "    \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 177,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Bijector for advi samples.\n",
    "def biject(samples):\n",
    "    return dict(lengthscale=samples[:, 0].exp().numpy(),\n",
    "                noise=samples[:, 1].exp().numpy())\n",
    "                #sigma=samples[:, 2].exp().numpy())\n",
    "    \n",
    "# Get ADVI samples in constrained space.\n",
    "advi_posterior_samples = biject(guide.get_posterior().sample((1000, )))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 178,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "AutoDiagonalNormal.loc:\n",
      "Parameter containing:\n",
      "tensor([-1.4547, -3.9433], requires_grad=True)\n",
      "\n",
      "AutoDiagonalNormal.scale:\n",
      "tensor([0.6166, 2.2731], grad_fn=<AddBackward0>)\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for key, value in pyro.get_param_store().items():    \n",
    "    print(f\"{key}:\\n{value}\\n\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 179,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x141c45160>]"
      ]
     },
     "execution_count": 179,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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REREROnGSnAqmCFYjwCREBDgfsFERERE6cQQnAqaWQmOqEowePpkIiIionRiCE4FR68QHIlASgkhRHrHRURERJSl2BOcClYlOAyEQ4DU1UciIiIiSguG4FQwK8GhqODLxXFEREREacMQnApGJVgGo/YH5l7BRERERGnDEJwKZiU4uvobYCWYiIiIKF0YglMhXghmOwQRERFR2jAEp4K5MC7AdggiIiKiTMAQnAoOY5qjWyCCDMFERERE6cIQnAoOYztmtkMQERERZQSG4FTQ+vYESy6MIyIiIkobhuBUMNshYnaHYDsEERERUbowBKdCvIVx7AkmIiIiShuG4FQweoIle4KJiIiIMgJDcCrE6QnmyTKIiIiI0ochOBXYE0xERESUURiCUyFuTzArwURERETpwhCcAkLTAKHZIVhoDMFEREREacQQnCqOqOCbnw/JdggiIiKitGEIThXNAQSD6vPcfFaCiYiIiNKIIThVHA57b+C8Ai6MIyIiIkojhuBUcTgAKdXn+QWsBBMRERGlEUNwqpg7RAAQeQzBREREROnEEJwqUSEYeflshyAiIiJKI4bgVHEYIdjpBNw5PGMcERERURoxBKeKGYIdLsDtAYIBSLNHmIiIiIhSiiE4VbSoSrDHA0gdCIfSOyYiIiKiLMUQnCrR7RCeHPU5+4KJiIiI0oIhOFU0Y6qdRjsEwL5gIiIiojRhCE4Vh9P+aIZgbpNGRERElBYMwakS1Q4hzHaIYA9kJAL91X9BhoLpGxsRERFRlnEOx41s2LABDz74IHRdx9lnn41PfOITw3GzY4vVDuEEikvV5431QIsP8vf3AOEwxJnnpm98RERERFlkyJVgXdfxwAMP4MYbb8Rdd92F1157DQcPHhyOsY0t5u4QDicwaTrgyYHcsQly23sAALnmhTQOjoiIiCi7DDkE79y5E5WVlRg/fjycTidOOukkrF27djjGNraYPcEuF4TTCcyYB7l9E+T2TYDQgD07IOv55oGIiIgoFYbcDtHc3Izy8nLr6/Lycrz//vt9rldbW4va2loAwMqVK+H1eod610lzOp1puV8AaMnNQRCAOzcPpV4vupacgM5H7gcA5H74U+h+/inkbnwTBccuSsv4+pPOORuNOF/J4Xwlj3OWHM5XcjhfyeF8JS+T5mxYeoITUVNTg5qaGutrn8+Xqru2eL3etNwv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\n",
      "text/plain": [
       "<Figure size 864x576 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(np.arange(len(advi_loss)), advi_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 202,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(0.6201)\n",
      "tensor(0.7237)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x1426b1190>"
      ]
     },
     "execution_count": 202,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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9on29vbMnj07STJ79uy0t7cnSVauXJm3vOUtqdVqOeKII9LZ2ZktW7ZUs1oAAOgHfb5SfOWVVyZJTjzxxLS2tmbr1q05+OCDkyQHHXRQtm7dmiTp6OhIS0tLz+smT56cjo6OnucCAMBQ06covvzyy9Pc3JytW7fmiiuuyNSpU1/w32u1Wmq12l69cVtbW9ra2pIkCxYseEFID2cj5Tz2VWNjY/EzqJJ5Vss8q2We1TLPaplntUqYZ5+iuLm5OUkyadKkHHvssXn88cczadKkbNmyJQcffHC2bNnSs9+4ubk5mzZt6nnt5s2be17/fK2trWltbe35+vmvGc5Gynnsq5aWluJnUCXzrJZ5Vss8q2We1TLPao2kef7vi7u/0+ue4p07d+aZZ57p+fNDDz2Uww47LDNnzsyKFSuSJCtWrMixxx6bJJk5c2buv//+1Ov1rFmzJk1NTbZOAAAwpPV6pXjr1q259tprkyRdXV05/vjj8/rXvz6HH354Fi5cmGXLlvXcki1J3vCGN2TVqlU577zzMnr06Jxzzjn9ewYAALCfeo3iQw89NNdcc82LHp8wYUIuvfTSFz1eq9Vy9tlnV7M6AAAYAD7RDgCA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACK19jXJ3Z3d+fiiy9Oc3NzLr744mzcuDE33HBDtm3blunTp+fcc89NY2Njdu3alZtuuilr167NhAkTMn/+/EyZMqU/zwEAAPZLn68U33XXXZk2bVrP11/96ldz6qmnZtGiRRk3blyWLVuWJFm2bFnGjRuXRYsW5dRTT81tt91W/aoBAKBCfYrizZs3Z9WqVZk3b16SpF6v55FHHsmsWbOSJHPmzEl7e3uSZOXKlZkzZ06SZNasWXn44YdTr9f7YekAAFCNPm2fuPXWW/Pud787zzzzTJJk27ZtaWpqSkNDQ5Kkubk5HR0dSZKOjo5Mnjw5SdLQ0JCmpqZs27YtEydOfMEx29ra0tbWliRZsGBBWlpaqjmjQTZSzmNfNTY2Fj+DKplntcyzWuZZLfOslnlWq4R59hrF//mf/5lJkyZl+vTpeeSRRyp749bW1rS2tvZ8vWnTpsqOPZhGynnsq5aWluJnUCXzrJZ5Vss8q2We1TLPao2keU6dOnWPj/caxT//+c+zcuXKrF69Os8991yeeeaZ3HrrrdmxY0e6urrS0NCQjo6ONDc3J/ntVePNmzdn8uTJ6erqyo4dOzJhwoRqzwYAACrU657id73rXfn85z+fxYsXZ/78+TnqqKNy3nnnZcaMGXnggQeSJMuXL8/MmTOTJMccc0yWL1+eJHnggQcyY8aM1Gq1/jsDAADYT/t8n+Izzzwzd9xxR84999xs3749c+fOTZLMnTs327dvz7nnnps77rgjZ555ZmWLBQCA/tDn+xQnyYwZMzJjxowkyaGHHpqrrrrqRc8ZPXp0Pvaxj1WzOgAAGAA+0Q4AgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOKJYgAAiieKAQAonigGAKB4ohgAgOI19vaE5557Lpdddll2796drq6uzJo1K+94xzuycePG3HDDDdm2bVumT5+ec889N42Njdm1a1duuummrF27NhMmTMj8+fMzZcqUgTgXAADYJ71eKT7ggANy2WWX5ZprrsnVV1+dBx98MGvWrMlXv/rVnHrqqVm0aFHGjRuXZcuWJUmWLVuWcePGZdGiRTn11FNz22239ftJAADA/ug1imu1WsaOHZsk6erqSldXV2q1Wh555JHMmjUrSTJnzpy0t7cnSVauXJk5c+YkSWbNmpWHH3449Xq9n5YPAAD7r9ftE0nS3d2dT3ziE3nyySdz0kkn5dBDD01TU1MaGhqSJM3Nzeno6EiSdHR0ZPLkyUmShoaGNDU1Zdu2bZk4cWI/nQIAAOyfPkXxqFGjcs0116SzszPXXnttNmzYsN9v3NbWlra2tiTJggUL0tLSst/HHApGynnsq8bGxuJnUCXzrJZ5Vss8q2We1TLPapUwzz5F8e+MGzcuM2bMyJo1a7Jjx450dXWloaEhHR0daW5uTvLbq8abN2/O5MmT09XVlR07dmTChAkvOlZra2taW1t7vt60adN+nsrQMFLOY1+1tLQUP4MqmWe1zLNa5lkt86yWeVZrJM1z6tSpe3y81z3FTz/9dDo7O5P89k4UDz30UKZNm5YZM2bkgQceSJIsX748M2fOTJIcc8wxWb58eZLkgQceyIwZM1Kr1ao4BwAA6Be9XinesmVLFi9enO7u7tTr9Rx33HE55phj8qpXvSo33HBDvvGNb+T3fu/3Mnfu3CTJ3Llzc9NNN+Xcc8/N+PHjM3/+/P4+BwAA2C+9RvFrXvOaXH311S96/NBDD81VV131osdHjx6dj33sY9WsDgAABoBPtAMAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiNvT1h06ZNWbx4cX7zm9+kVqultbU1p5xySrZv356FCxfmqaeeyiGHHJKPfvSjGT9+fOr1epYsWZLVq1dnzJgxOeecczJ9+vSBOBcAANgnvV4pbmhoyFlnnZWFCxfmyiuvzD333JP//u//ztKlS3P00UfnxhtvzNFHH52lS5cmSVavXp0nn3wyN954Y97//vfnlltu6e9zAACA/dJrFB988ME9V3oPPPDATJs2LR0dHWlvb8/s2bOTJLNnz057e3uSZOXKlXnLW96SWq2WI444Ip2dndmyZUs/ngIAAOyfvdpTvHHjxqxbty6///u/n61bt+bggw9Okhx00EHZunVrkqSjoyMtLS09r5k8eXI6OjoqXDIAAFSr1z3Fv7Nz585cd911+cu//Ms0NTW94L/VarXUarW9euO2tra0tbUlSRYsWPCCkB7ORsp57KvGxsbiZ1Al86yWeVbLPKtlntUyz2qVMM8+RfHu3btz3XXX5Y//+I/zpje9KUkyadKkbNmyJQcffHC2bNmSiRMnJkmam5uzadOmntdu3rw5zc3NLzpma2trWltbe75+/muGs5FyHvuqpaWl+BlUyTyrZZ7VMs9qmWe1zLNaI2meU6dO3ePjvW6fqNfr+fznP59p06blrW99a8/jM2fOzIoVK5IkK1asyLHHHtvz+P333596vZ41a9akqampZ5sFAAAMRb1eKf75z3+e+++/P4cddlguvPDCJMk73/nOnHHGGVm4cGGWLVvWc0u2JHnDG96QVatW5bzzzsvo0aNzzjnn9O8ZAADAfuo1io888sh885vf3ON/u/TSS1/0WK1Wy9lnn73/KwMAgAHiE+0AACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACKJ4oBACieKAYAoHiiGACA4oliAACK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\n",
      "text/plain": [
       "<Figure size 864x576 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for k in advi_posterior_samples.keys():\n",
    "    #plt.hist(advi_posterior_samples[k], bins=30, label=k, range=(0,1.3))\n",
    "    quan = torch.quantile(torch.tensor(advi_posterior_samples[k]), q=0.95, dim=0)\n",
    "    print(quan)\n",
    "    plt.hist(advi_posterior_samples[k], bins=30, label=k, range=(0,quan.numpy()))\n",
    "plt.legend()\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 110,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Sample: 100%|██████████| 20/20 [00:01, 13.29it/s, step size=2.00e-01, acc. prob=0.446]\n"
     ]
    }
   ],
   "source": [
    "# Standardize the data s.t. input is on unit cube and output has zero mean and std=1.\n",
    "min_x = torch.min(train_x)\n",
    "max_x = torch.max(train_x - min_x)\n",
    "train_x_trans = (train_x - min_x) / max_x\n",
    "\n",
    "mu_y = torch.mean(train_y)\n",
    "sigma_y = torch.std(train_y)\n",
    "train_y_trans = (train_y - mu_y) / sigma_y\n",
    "\n",
    "# Define likelihood, model and set priors\n",
    "length_scale_prior = GammaPrior(3, 6)\n",
    "noise_prior = LogNormalPrior(loc=0, scale=2)\n",
    "likelihood = gpytorch.likelihoods.GaussianLikelihood(noise_constraint=gpytorch.constraints.Positive())\n",
    "model = ExactGPModel(train_x_trans, train_y_trans, likelihood)\n",
    "model.covar_module.base_kernel.register_prior(\"lengthscale_prior\", length_scale_prior, \"lengthscale\")\n",
    "likelihood.register_prior(\"noise_prior\", noise_prior, \"noise\")\n",
    "mll = gpytorch.mlls.ExactMarginalLogLikelihood(likelihood, model)\n",
    "\n",
    "# Get posteriors of hyperparam\n",
    "nuts_kernel = NUTS(pyro_model, adapt_step_size=False, target_accept_prob=0.8, step_size=2e-1, max_tree_depth=10)\n",
    "mcmc_run = MCMC(nuts_kernel, num_samples=10, warmup_steps=10, num_chains=1)\n",
    "mcmc_run.run(train_x_trans, train_y_trans)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Functions for estimating new priors"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "def fit_gamma(tmp_data):\n",
    "    fit_alpha, fit_loc, fit_beta = stats.gamma.fit(tmp_data, floc=0)\n",
    "    #print(\"\\nGamma\")\n",
    "    #print(\"Parameters\", fit_alpha, fit_loc, fit_beta) #shape, location, scale\n",
    "\n",
    "\n",
    "    # calc the log likelihood \n",
    "    logLik = -np.sum( stats.gamma.logpdf(tmp_data, fit_alpha, loc=fit_loc, scale=fit_beta) )\n",
    "\n",
    "    # calc AIC\n",
    "    aic = 2 * 3 - 2 * logLik\n",
    "    #print(\"AIC:\", aic)\n",
    "    \n",
    "    return aic, fit_alpha, fit_loc, fit_beta\n",
    "\n",
    "def fit_lognorm(tmp_data):\n",
    "    fit_shape, fit_loc, fit_scale = stats.lognorm.fit(tmp_data, floc=0)\n",
    "    #print(\"\\nLogNormal\")\n",
    "    #print(\"Parameters\", fit_shape, fit_loc, fit_scale) \n",
    "\n",
    "    # calc the log likelihood \n",
    "    logLik = -np.sum( stats.lognorm.logpdf(tmp_data, fit_shape, loc=fit_loc, scale=fit_scale) )\n",
    "\n",
    "    # calc AIC\n",
    "    aic = 2 * 2 - 2 * logLik\n",
    "    #print(\"AIC:\", aic)\n",
    "    \n",
    "    return aic, fit_shape, fit_loc, fit_scale\n",
    "\n",
    "def find_my_prior(tmp_data):\n",
    "    aic_g, fit_g_alpha, fit_g_loc, fit_g_beta = fit_gamma(tmp_data)\n",
    "    aic_l, fit_l_shape, fit_l_loc, fit_l_scale = fit_lognorm(tmp_data)\n",
    "    \n",
    "    l_loc = np.log(fit_l_scale)\n",
    "    l_scale = fit_l_shape\n",
    "    concentration = fit_g_alpha\n",
    "    rate = 1/fit_g_beta\n",
    "    \n",
    "    if aic_l < aic_g:\n",
    "        print(f\"LogNormal is best: loc = {l_loc}, scale = {l_scale}\")\n",
    "        fun = LogNormalPrior(loc=l_loc, scale=l_scale)\n",
    "    else:\n",
    "        print(f\"Gamma is best: alpha = {concentration}, beta = {rate}\")\n",
    "        fun = GammaPrior(concentration, rate)\n",
    "    \n",
    "    stats = {\"logN_loc\": l_loc,\n",
    "             \"logN_scale\": l_scale,\n",
    "             \"Gamma_alpha\": concentration,\n",
    "             \"Gamma_beta\": rate\n",
    "            }\n",
    "    return fun, stats"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Plotting functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plot posteriors --> new priors\n",
    "\n",
    "def plot_posterior(ax, post_samples, stats):\n",
    "    tmp_data = post_samples\n",
    "    ax.hist(x=tmp_data.squeeze(-1).squeeze(-1).numpy(), bins=40, density=True)\n",
    "    tmp_x = np.linspace(0.001,torch.max(tmp_data),1000)\n",
    "    beta, alpha = stats['Gamma_beta'], stats['Gamma_alpha']\n",
    "    tmp_pdf_g = np.power(beta, alpha) * np.power(tmp_x, alpha-1) * np.exp(-beta * tmp_x) / scipy.special.gamma(alpha)\n",
    "    mu, sigma = stats['logN_loc'], stats['logN_scale']\n",
    "    tmp_pdf_l = np.exp(- np.power(np.log(tmp_x) - mu, 2) / (2*np.power(sigma, 2))) / (tmp_x * sigma * np.sqrt(2*np.pi))\n",
    "    ax.plot(tmp_x, tmp_pdf_g, label=\"Gamma distribution\")\n",
    "    ax.plot(tmp_x, tmp_pdf_l, label=\"LogNormal distribution\")\n",
    "    ax.legend()\n",
    "    \n",
    "    \n",
    "# Plot mean functions drawn from batch model\n",
    "def plot_mean_functions(ax, train_x, train_y, test_x, output):\n",
    "    \n",
    "    # Plot training data as black stars\n",
    "    ax.plot(train_x.numpy(), train_y.numpy(), 'k*', zorder=10)\n",
    "    \n",
    "    mean_all = torch.mean(output.mean, axis=0) * sigma_y + mu_y\n",
    "    stddev_all = torch.std(output.mean, axis=0) * sigma_y\n",
    "    mean_all, stddev_all = mean_all.detach().numpy(), stddev_all.detach().numpy()\n",
    "    ax.plot(test_x.numpy(), mean_all, c='r')\n",
    "    ax.fill_between(test_x.squeeze(-1).numpy(), mean_all - 2*stddev_all, mean_all + 2*stddev_all,\n",
    "                    alpha=0.2)\n",
    "\n",
    "    for i in range(min(num_samples, 50)):\n",
    "        # Plot predictive means as blue line\n",
    "        y_pred = output.mean[i].detach()\n",
    "        y_pred = y_pred * sigma_y + mu_y\n",
    "        ax.plot(test_x.numpy(), y_pred.numpy(), 'b', linewidth=0.3)\n",
    "        \n",
    "    ax.legend(['Observed Data', 'Mean of sampled means', 'Std. of samples means', 'Sampled Means'])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Test 1 active learning step with update of posteriors "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Run MCMC to estimate posteriors of the hyperparameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Build torch.tensors\n",
    "train_x = torch.from_numpy(X_train).view(-1,)\n",
    "train_y = torch.from_numpy(y_train).view(-1,)\n",
    "\n",
    "test_x = torch.from_numpy(X_test).view(-1,).unsqueeze(-1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Sample: 100%|██████████| 150/150 [00:10, 14.66it/s, step size=2.00e-01, acc. prob=0.576]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "                                                     mean       std    median      5.0%     95.0%     n_eff     r_hat\n",
      "                      likelihood.noise_prior[0]      0.16      0.29      0.07      0.01      0.43     39.64      1.04\n",
      "covar_module.base_kernel.lengthscale_prior[0,0]      0.42      0.17      0.39      0.17      0.69     20.02      0.99\n",
      "\n",
      "Number of divergences: 0\n",
      "Gamma is best: alpha = 6.551307187865932, beta = 15.420873205387132\n",
      "Gamma is best: alpha = 0.7401733818084482, beta = 4.50079187694946\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Sample: 100%|██████████| 150/150 [00:13, 11.06it/s, step size=2.00e-01, acc. prob=0.644]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "                                                     mean       std    median      5.0%     95.0%     n_eff     r_hat\n",
      "                      likelihood.noise_prior[0]      0.14      0.08      0.13      0.03      0.26     30.39      1.15\n",
      "covar_module.base_kernel.lengthscale_prior[0,0]      0.39      0.13      0.38      0.21      0.59     54.85      0.99\n",
      "\n",
      "Number of divergences: 0\n",
      "LogNormal is best: loc = -0.9923579729669245, scale = 0.32833985443619745\n",
      "LogNormal is best: loc = -2.1011628699436278, scale = 0.5986065491502184\n"
     ]
    }
   ],
   "source": [
    "# MCMC settings\n",
    "num_samples = 100\n",
    "warmup_steps = 50\n",
    "\n",
    "\n",
    "active_learning_steps = 2\n",
    "\n",
    "# Initial priors\n",
    "length_scale_prior = GammaPrior(3, 6)\n",
    "noise_prior = LogNormalPrior(loc=0, scale=2)\n",
    "\n",
    "\n",
    "\n",
    "for i in range(active_learning_steps):\n",
    "    \n",
    "    # Convert data to torch\n",
    "    if i > 0:\n",
    "        train_x = torch.tensor(train_x)\n",
    "        train_y = torch.tensor(train_y)\n",
    "    \n",
    "    # Standardize the data s.t. input is on unit cube and output has zero mean and std=1.\n",
    "    min_x = torch.min(train_x)\n",
    "    max_x = torch.max(train_x - min_x)\n",
    "    train_x_trans = (train_x - min_x) / max_x\n",
    "\n",
    "    mu_y = torch.mean(train_y)\n",
    "    sigma_y = torch.std(train_y)\n",
    "    train_y_trans = (train_y - mu_y) / sigma_y\n",
    "    \n",
    "    # Define likelihood, model and set priors\n",
    "    likelihood = gpytorch.likelihoods.GaussianLikelihood(noise_constraint=gpytorch.constraints.Positive())\n",
    "    model = ExactGPModel(train_x_trans, train_y_trans, likelihood)\n",
    "    model.covar_module.base_kernel.register_prior(\"lengthscale_prior\", length_scale_prior, \"lengthscale\")\n",
    "    likelihood.register_prior(\"noise_prior\", noise_prior, \"noise\")\n",
    "    mll = gpytorch.mlls.ExactMarginalLogLikelihood(likelihood, model)\n",
    "    \n",
    "    # Get posteriors of hyperparam\n",
    "    nuts_kernel = NUTS(pyro_model, adapt_step_size=False, target_accept_prob=0.8, step_size=2e-1, max_tree_depth=10)\n",
    "    mcmc_run = MCMC(nuts_kernel, num_samples=num_samples, warmup_steps=warmup_steps, num_chains=1)\n",
    "    mcmc_run.run(train_x_trans, train_y_trans)\n",
    "    \n",
    "    # Look at summary\n",
    "    mcmc_run.summary()\n",
    "    \n",
    "    # Extract samples from posterior\n",
    "    posterior_samples = mcmc_run.get_samples()\n",
    "    \n",
    "    # Get new priors\n",
    "    length_scale_prior, l_stats = find_my_prior(posterior_samples[\"covar_module.base_kernel.lengthscale_prior\"])\n",
    "    noise_prior, n_stats = find_my_prior(posterior_samples[\"likelihood.noise_prior\"])\n",
    "    \n",
    "    # Turn model into batch model in order to draw mean functions\n",
    "    model.pyro_load_from_samples(mcmc_run.get_samples())\n",
    "    \n",
    "    # Propogate search space through model\n",
    "    model.eval()\n",
    "    test_x_trans = (test_x - min_x) / max_x\n",
    "    test_y = torch.from_numpy(y_test).view(-1,)\n",
    "    expanded_test_x = test_x_trans.unsqueeze(0).repeat(num_samples, 1, 1)\n",
    "    output = model(expanded_test_x)\n",
    "    \n",
    "    # Get mean(mean) and std(mean)\n",
    "    with torch.no_grad():\n",
    "        mean_all = torch.mean(output.mean, axis=0) * sigma_y + mu_y\n",
    "        stddev_all = torch.std(output.mean, axis=0) * sigma_y\n",
    "        mean_all, stddev_all = mean_all.numpy(), stddev_all.numpy()\n",
    "    \n",
    "    # AL, where is stddev highest\n",
    "    idx = np.argmax(stddev_all)\n",
    "     \n",
    "    # Plot\n",
    "    f, ax = plt.subplots(1, 3, figsize=(20, 6))\n",
    "    plot_mean_functions(ax[0], train_x, train_y, test_x, output)\n",
    "    plot_posterior(ax[1], posterior_samples[\"covar_module.base_kernel.lengthscale_prior\"], l_stats)\n",
    "    plot_posterior(ax[2], posterior_samples[\"likelihood.noise_prior\"], n_stats)\n",
    "\n",
    "    plt.savefig(f'mbml/initial_experiment_{i}.pdf',\n",
    "                                bbox_inches=\"tight\")\n",
    "    f.clf()\n",
    "    plt.close()\n",
    "    \n",
    "    # Get new point\n",
    "    new_x = test_x[idx]\n",
    "    new_y = np.sin(4.0*new_x) + 0.1* np.random.randn(len(new_x), 1)[0] # *new_x**3 \n",
    "    \n",
    "    # Add to data set\n",
    "    train_x = np.concatenate([train_x, new_x])\n",
    "    train_y = np.concatenate([train_y, new_y])\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "MultivariateNormal(loc: torch.Size([100, 101]))"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "100"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "num_samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "from tqdm import tqdm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [00:01<00:00, 82.80it/s]\n"
     ]
    }
   ],
   "source": [
    "mlls = 0\n",
    "rmse = 0\n",
    "with torch.no_grad():\n",
    "    for j in tqdm(range(num_samples)):\n",
    "        mlls += torch.mean(-mll(output[j], test_y))\n",
    "        rmse += torch.sqrt(torch.mean(torch.pow(output.mean[j] - test_y, 2)))\n",
    "    mlls = mlls/num_samples\n",
    "    rmse = rmse/num_samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(-0.9442, dtype=torch.float64)\n",
      "tensor(0.9967, dtype=torch.float64)\n"
     ]
    }
   ],
   "source": [
    "print(mlls)\n",
    "print(rmse)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.9948, dtype=torch.float64, grad_fn=<SqrtBackward>)"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.sqrt(torch.mean(torch.pow(output.mean.mean(0) - test_y, 2)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "likelihood.noise_prior:\n",
      "posterior mode=0.03 mean=0.14 std=0.08\n",
      "\n",
      "covar_module.base_kernel.lengthscale_prior:\n",
      "posterior mode=0.17 mean=0.39 std=0.13\n"
     ]
    }
   ],
   "source": [
    "from scipy import stats\n",
    "\n",
    "for k in posterior_samples.keys():\n",
    "    print(\"\\n%s:\" % k)\n",
    "    print(\"posterior mode=%.2f mean=%.2f std=%.2f\" % (stats.mode(posterior_samples[k].numpy())[0][0], \n",
    "                                                      posterior_samples[k].mean(), posterior_samples[k].std()))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "# make a GP with these variables\n",
    "likelihood = gpytorch.likelihoods.GaussianLikelihood(noise_constraint=gpytorch.constraints.Positive())\n",
    "model = ExactGPModel(train_x_trans, train_y_trans, likelihood)\n",
    "#likelihood.register_prior(\"noise_prior\", noise_prior, \"noise\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Actual lengthscale: tensor([[0.3631]], grad_fn=<SoftplusBackward>)\n",
      "Actual mean: Parameter containing:\n",
      "tensor([0.], requires_grad=True)\n",
      "Actual outputscale: 0.6931471824645996\n",
      "Actual noise: tensor([0.2221], grad_fn=<SoftplusBackward>)\n",
      "\n",
      "Actual lengthscale: tensor([[0.1100]], grad_fn=<SoftplusBackward>)\n",
      "Actual outputscale: 1.0\n",
      "Actual noise: tensor([0.], grad_fn=<SoftplusBackward>)\n"
     ]
    }
   ],
   "source": [
    "# Convenient way of getting true outputscale\n",
    "print(f'Actual lengthscale: {model.covar_module.base_kernel.lengthscale}')\n",
    "print(f'Actual mean: {model.mean_module.constant}')\n",
    "print(f'Actual outputscale: {model.covar_module.outputscale}')\n",
    "print(f'Actual noise: {likelihood.noise}')\n",
    "\n",
    "# Convenient way of setting true outputscale\n",
    "model.covar_module.outputscale = 1.\n",
    "model.covar_module.base_kernel.lengthscale = 0.11\n",
    "likelihood.noise = 0\n",
    "print(f'\\nActual lengthscale: {model.covar_module.base_kernel.lengthscale}')\n",
    "print(f'Actual outputscale: {model.covar_module.outputscale}')\n",
    "print(f'Actual noise: {likelihood.noise}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/christofferriis/envs/mbml/lib/python3.9/site-packages/gpytorch/utils/cholesky.py:51: NumericalWarning: A not p.d., added jitter of 1.0e-08 to the diagonal\n",
      "  warnings.warn(f\"A not p.d., added jitter of {jitter_new:.1e} to the diagonal\", NumericalWarning)\n"
     ]
    }
   ],
   "source": [
    "model.eval()\n",
    "with torch.no_grad(), gpytorch.settings.fast_pred_var():\n",
    "    predictions = likelihood(model(test_x_trans))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(101,)"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_y.numpy().shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x13c317af0>]"
      ]
     },
     "execution_count": 31,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x576 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(test_x_trans.numpy(), (test_y - mu_y) / sigma_y, 'o')\n",
    "plt.plot(test_x_trans.numpy(), predictions.mean)\n",
    "plt.plot(train_x_trans, train_y_trans, 'bo')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "colab": {
   "collapsed_sections": [
    "8bnJcoggg-fi"
   ],
   "name": "12 - Gaussian processes - Pyro.ipynb",
   "provenance": []
  },
  "kernelspec": {
   "display_name": "mbml",
   "language": "python",
   "name": "mbml"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
