{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Gaussian Process Latent Variable Model"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The [Gaussian Process Latent Variable Model](https://en.wikipedia.org/wiki/Nonlinear_dimensionality_reduction#Gaussian_process_latent_variable_models) (GPLVM) is a dimensionality reduction method that uses a Gaussian process to learn a low-dimensional representation of (potentially) high-dimensional data. In the typical setting of Gaussian process regression, where we are given inputs $X$ and outputs $y$, we choose a kernel and learn hyperparameters that best describe the mapping from $X$ to $y$. In the GPLVM, we are not given $X$: we are only given $y$. So we need to learn $X$ along with the kernel hyperparameters."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We do not do maximum likelihood inference on $X$. Instead, we set a Gaussian prior for $X$ and learn the mean and variance of the approximate (gaussian) posterior $q(X|y)$. In this notebook, we show how this can be done using the `pyro.contrib.gp` module. In particular we reproduce a result described in [2]."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import matplotlib.pyplot as plt\n",
    "import pandas as pd\n",
    "import torch\n",
    "from torch.nn import Parameter\n",
    "\n",
    "import pyro\n",
    "import pyro.contrib.gp as gp\n",
    "import pyro.distributions as dist\n",
    "import pyro.ops.stats as stats\n",
    "\n",
    "smoke_test = ('CI' in os.environ)  # ignore; used to check code integrity in the Pyro repo\n",
    "assert pyro.__version__.startswith('0.5.0')\n",
    "pyro.enable_validation(True)       # can help with debugging\n",
    "pyro.set_rng_seed(1)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Dataset"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The data we are going to use consists of [single-cell](https://en.wikipedia.org/wiki/Single-cell_analysis) [qPCR](https://en.wikipedia.org/wiki/Real-time_polymerase_chain_reaction) data for 48 genes obtained from mice (Guo *et al.*, [1]). This data is available at the [Open Data Science repository](https://github.com/sods/ods). The data contains 48 columns, with each column corresponding to (normalized) measurements of each gene. Cells differentiate during their development and these data were obtained at various stages of development. The various stages are labelled from the 1-cell stage to the 64-cell stage. For the 32-cell stage, the data is further differentiated into 'trophectoderm' (TE) and 'inner cell mass' (ICM). ICM further differentiates into 'epiblast' (EPI) and 'primitive endoderm' (PE) at the 64-cell stage. Each of the rows in the dataset is labelled with one of these stages."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Data shape: (437, 48)\n",
      "---------------------\n",
      "\n",
      "Data labels: ['1', '2', '4', '8', '16', '32 TE', '32 ICM', '64 PE', '64 TE', '64 EPI']\n",
      "--------------------------------------------------------------------------------------\n",
      "\n",
      "Show a small subset of the data:\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style>\n",
       "    .dataframe thead tr:only-child th {\n",
       "        text-align: right;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: left;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Actb</th>\n",
       "      <th>Ahcy</th>\n",
       "      <th>Aqp3</th>\n",
       "      <th>Atp12a</th>\n",
       "      <th>Bmp4</th>\n",
       "      <th>Cdx2</th>\n",
       "      <th>Creb312</th>\n",
       "      <th>Cebpa</th>\n",
       "      <th>Dab2</th>\n",
       "      <th>DppaI</th>\n",
       "      <th>...</th>\n",
       "      <th>Sox2</th>\n",
       "      <th>Sall4</th>\n",
       "      <th>Sox17</th>\n",
       "      <th>Snail</th>\n",
       "      <th>Sox13</th>\n",
       "      <th>Tcfap2a</th>\n",
       "      <th>Tcfap2c</th>\n",
       "      <th>Tcf23</th>\n",
       "      <th>Utf1</th>\n",
       "      <th>Tspan8</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.541050</td>\n",
       "      <td>-1.203007</td>\n",
       "      <td>1.030746</td>\n",
       "      <td>1.064808</td>\n",
       "      <td>0.494782</td>\n",
       "      <td>-0.167143</td>\n",
       "      <td>-1.369092</td>\n",
       "      <td>1.083061</td>\n",
       "      <td>0.668057</td>\n",
       "      <td>-1.553758</td>\n",
       "      <td>...</td>\n",
       "      <td>-1.351757</td>\n",
       "      <td>-1.793476</td>\n",
       "      <td>0.783185</td>\n",
       "      <td>-1.408063</td>\n",
       "      <td>-0.031991</td>\n",
       "      <td>-0.351257</td>\n",
       "      <td>-1.078982</td>\n",
       "      <td>0.942981</td>\n",
       "      <td>1.348892</td>\n",
       "      <td>-1.051999</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.680832</td>\n",
       "      <td>-1.355306</td>\n",
       "      <td>2.456375</td>\n",
       "      <td>1.234350</td>\n",
       "      <td>0.645494</td>\n",
       "      <td>1.003868</td>\n",
       "      <td>-1.207595</td>\n",
       "      <td>1.208023</td>\n",
       "      <td>0.800388</td>\n",
       "      <td>-1.435306</td>\n",
       "      <td>...</td>\n",
       "      <td>-1.363533</td>\n",
       "      <td>-1.782172</td>\n",
       "      <td>1.532477</td>\n",
       "      <td>-1.361172</td>\n",
       "      <td>-0.501715</td>\n",
       "      <td>1.082362</td>\n",
       "      <td>-0.930112</td>\n",
       "      <td>1.064399</td>\n",
       "      <td>1.469397</td>\n",
       "      <td>-0.996275</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>1.056038</td>\n",
       "      <td>-1.280447</td>\n",
       "      <td>2.046133</td>\n",
       "      <td>1.439795</td>\n",
       "      <td>0.828121</td>\n",
       "      <td>0.983404</td>\n",
       "      <td>-1.460032</td>\n",
       "      <td>1.359447</td>\n",
       "      <td>0.530701</td>\n",
       "      <td>-1.340283</td>\n",
       "      <td>...</td>\n",
       "      <td>-1.296802</td>\n",
       "      <td>-1.567402</td>\n",
       "      <td>3.194157</td>\n",
       "      <td>-1.301777</td>\n",
       "      <td>-0.445219</td>\n",
       "      <td>0.031284</td>\n",
       "      <td>-1.005767</td>\n",
       "      <td>1.211529</td>\n",
       "      <td>1.615421</td>\n",
       "      <td>-0.651393</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.732331</td>\n",
       "      <td>-1.326911</td>\n",
       "      <td>2.464234</td>\n",
       "      <td>1.244323</td>\n",
       "      <td>0.654359</td>\n",
       "      <td>0.947023</td>\n",
       "      <td>-1.265609</td>\n",
       "      <td>1.215373</td>\n",
       "      <td>0.765212</td>\n",
       "      <td>-1.431401</td>\n",
       "      <td>...</td>\n",
       "      <td>-1.684100</td>\n",
       "      <td>-1.915556</td>\n",
       "      <td>2.962515</td>\n",
       "      <td>-1.349710</td>\n",
       "      <td>1.875957</td>\n",
       "      <td>1.699892</td>\n",
       "      <td>-1.059458</td>\n",
       "      <td>1.071541</td>\n",
       "      <td>1.476485</td>\n",
       "      <td>-0.699586</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.629333</td>\n",
       "      <td>-1.244308</td>\n",
       "      <td>1.316815</td>\n",
       "      <td>1.304162</td>\n",
       "      <td>0.707552</td>\n",
       "      <td>1.429070</td>\n",
       "      <td>-0.895578</td>\n",
       "      <td>-0.007785</td>\n",
       "      <td>0.644606</td>\n",
       "      <td>-1.381937</td>\n",
       "      <td>...</td>\n",
       "      <td>-1.304653</td>\n",
       "      <td>-1.761825</td>\n",
       "      <td>1.265379</td>\n",
       "      <td>-1.320533</td>\n",
       "      <td>-0.609864</td>\n",
       "      <td>0.413826</td>\n",
       "      <td>-0.888624</td>\n",
       "      <td>1.114394</td>\n",
       "      <td>1.519017</td>\n",
       "      <td>-0.798985</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>5 rows × 48 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "       Actb      Ahcy      Aqp3    Atp12a      Bmp4      Cdx2   Creb312  \\\n",
       "1  0.541050 -1.203007  1.030746  1.064808  0.494782 -0.167143 -1.369092   \n",
       "1  0.680832 -1.355306  2.456375  1.234350  0.645494  1.003868 -1.207595   \n",
       "1  1.056038 -1.280447  2.046133  1.439795  0.828121  0.983404 -1.460032   \n",
       "1  0.732331 -1.326911  2.464234  1.244323  0.654359  0.947023 -1.265609   \n",
       "1  0.629333 -1.244308  1.316815  1.304162  0.707552  1.429070 -0.895578   \n",
       "\n",
       "      Cebpa      Dab2     DppaI    ...         Sox2     Sall4     Sox17  \\\n",
       "1  1.083061  0.668057 -1.553758    ...    -1.351757 -1.793476  0.783185   \n",
       "1  1.208023  0.800388 -1.435306    ...    -1.363533 -1.782172  1.532477   \n",
       "1  1.359447  0.530701 -1.340283    ...    -1.296802 -1.567402  3.194157   \n",
       "1  1.215373  0.765212 -1.431401    ...    -1.684100 -1.915556  2.962515   \n",
       "1 -0.007785  0.644606 -1.381937    ...    -1.304653 -1.761825  1.265379   \n",
       "\n",
       "      Snail     Sox13   Tcfap2a   Tcfap2c     Tcf23      Utf1    Tspan8  \n",
       "1 -1.408063 -0.031991 -0.351257 -1.078982  0.942981  1.348892 -1.051999  \n",
       "1 -1.361172 -0.501715  1.082362 -0.930112  1.064399  1.469397 -0.996275  \n",
       "1 -1.301777 -0.445219  0.031284 -1.005767  1.211529  1.615421 -0.651393  \n",
       "1 -1.349710  1.875957  1.699892 -1.059458  1.071541  1.476485 -0.699586  \n",
       "1 -1.320533 -0.609864  0.413826 -0.888624  1.114394  1.519017 -0.798985  \n",
       "\n",
       "[5 rows x 48 columns]"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# license: Copyright (c) 2014, the Open Data Science Initiative\n",
    "# license: https://www.elsevier.com/legal/elsevier-website-terms-and-conditions\n",
    "URL = \"https://raw.githubusercontent.com/sods/ods/master/datasets/guo_qpcr.csv\"\n",
    "\n",
    "df = pd.read_csv(URL, index_col=0)\n",
    "print(\"Data shape: {}\\n{}\\n\".format(df.shape, \"-\" * 21))\n",
    "print(\"Data labels: {}\\n{}\\n\".format(df.index.unique().tolist(), \"-\" * 86))\n",
    "print(\"Show a small subset of the data:\")\n",
    "df.head()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Modelling"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "First, we need to define the output tensor $y$. To predict values for all $48$ genes, we need $48$ Gaussian processes. So the required shape for $y$ is `num_GPs x num_data = 48 x 437`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "data = torch.tensor(df.values, dtype=torch.get_default_dtype())\n",
    "# we need to transpose data to correct its shape\n",
    "y = data.t()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now comes the most interesting part. We know that the observed data $y$ has latent structure: in particular different datapoints correspond to different cell stages. We would like our GPLVM to learn this structure in an unsupervised manner. In principle, if we do a good job of inference then we should be able to discover this structure---at least if we choose reasonable priors. First, we have to choose the dimension of our latent space $X$. We choose $dim(X)=2$, since we would like our model to disentangle 'capture time' ($1$, $2$, $4$, $8$, $16$, $32$, and $64$) from cell branching types (TE, ICM, PE, EPI). Next, when we set the mean of our prior over $X$, we set the first dimension to be equal to the observed capture time. This will help the GPLVM discover the structure we are interested in and will make it more likely that that structure will be axis-aligned in a way that is easier for us to interpret."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "capture_time = y.new_tensor([int(cell_name.split(\" \")[0]) for cell_name in df.index.values])\n",
    "# we scale the time into the interval [0, 1]\n",
    "time = capture_time.log2() / 6\n",
    "\n",
    "# we setup the mean of our prior over X\n",
    "X_prior_mean = torch.zeros(y.size(1), 2)  # shape: 437 x 2\n",
    "X_prior_mean[:, 0] = time"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We will use a sparse version of Gaussian process inference to make training faster. Remember that we also need to define $X$ as a `Parameter` so that we can set a prior and guide (variational distribution) for it."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "kernel = gp.kernels.RBF(input_dim=2, lengthscale=torch.ones(2))\n",
    "\n",
    "# we clone here so that we don't change our prior during the course of training\n",
    "X = Parameter(X_prior_mean.clone())\n",
    "\n",
    "# we will use SparseGPRegression model with num_inducing=32;\n",
    "# initial values for Xu are sampled randomly from X_prior_mean\n",
    "Xu = stats.resample(X_prior_mean.clone(), 32)\n",
    "gplvm = gp.models.SparseGPRegression(X, y, kernel, Xu, noise=torch.tensor(0.01), jitter=1e-5)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We will use the [set_prior()](http://docs.pyro.ai/en/dev/contrib.gp.html#pyro.contrib.gp.parameterized.Parameterized.set_prior) and [autoguide()](http://docs.pyro.ai/en/dev/contrib.gp.html#pyro.contrib.gp.parameterized.Parameterized.autoguide) methods from the [Parameterized](http://docs.pyro.ai/en/dev/contrib.gp.html#module-pyro.contrib.gp.parameterized) class to set a prior and guide for $X$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "# we use `.to_event()` to tell Pyro that the prior distribution for X has no batch_shape\n",
    "gplvm.set_prior(\"X\", dist.Normal(X_prior_mean, 0.1).to_event())\n",
    "gplvm.autoguide(\"X\", dist.Normal)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Inference"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "As mentioned in the [Gaussian Processes tutorial](gp.ipynb), we can use the helper function [gp.util.train](http://docs.pyro.ai/en/dev/contrib.gp.html#pyro.contrib.gp.util.train) to train a Pyro GP module. By default, this helper function uses the Adam optimizer with a learning rate of `0.01`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<matplotlib.figure.Figure at 0x151c34e610>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# note that training is expected to take a minute or so\n",
    "losses = gp.util.train(gplvm, num_steps=4000)\n",
    "\n",
    "# let's plot the loss curve after 4000 steps of training\n",
    "plt.plot(losses)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "After inference, the mean and standard deviation of the approximated posterior $q(X) \\sim p(X | y)$ will be stored in the parameters `X_loc` and `X_scale`. To get a sample from $q(X)$, we need to set the `mode` of `gplvm` to `\"guide\"`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "gplvm.mode = \"guide\"\n",
    "X = gplvm.X"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Visualizing the result"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Let’s see what we got by applying GPLVM to our dataset."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\nEpZc+LjQAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<matplotlib.figure.Figure at 0x151da85c10>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure(figsize=(8, 6))\n",
    "colors = plt.get_cmap(\"tab10\").colors[::-1]\n",
    "labels = df.index.unique()\n",
    "\n",
    "X = gplvm.X_loc.detach().numpy()\n",
    "for i, label in enumerate(labels):\n",
    "    X_i = X[df.index == label]\n",
    "    plt.scatter(X_i[:, 0], X_i[:, 1], c=colors[i], label=label)\n",
    "\n",
    "plt.legend()\n",
    "plt.xlabel(\"pseudotime\", fontsize=14)\n",
    "plt.ylabel(\"branching\", fontsize=14)\n",
    "plt.title(\"GPLVM on Single-Cell qPCR data\", fontsize=16)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We can see that the first dimension of the latent $X$ for each cell (horizontal axis) corresponds well with the observed capture time (colors). On the other hand, the 32 TE cell and 64 TE cell are clustered near each other. And the fact that ICM cells differentiate into PE and EPI can also be observed from the figure!"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Remarks\n",
    "\n",
    "+ The sparse version scales well (linearly) with the number of data points. So the GPLVM can be used with large datasets. Indeed in [2] the authors have applied GPLVM to a dataset with 68k peripheral blood mononuclear cells.\n",
    "\n",
    "+ Much of the power of Gaussian Processes lies in the function prior defined by the kernel. We recommend users try out different combinations of kernels for different types of datasets! For example, if the data contains periodicities, it might make sense to use a [Periodic kernel](http://docs.pyro.ai/en/dev/contrib.gp.html#periodic). Other kernels can also be found in the [Pyro GP docs](http://docs.pyro.ai/en/dev/contrib.gp.html#module-pyro.contrib.gp.kernels)."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### References\n",
    "\n",
    "[1] `Resolution of Cell Fate Decisions Revealed by Single-Cell Gene Expression Analysis from Zygote to Blastocyst`,<br />&nbsp;&nbsp;&nbsp;&nbsp;\n",
    "Guoji Guo, Mikael Huss, Guo Qing Tong, Chaoyang Wang, Li Li Sun, Neil D. Clarke, Paul Robson\n",
    "\n",
    "[2] `GrandPrix: Scaling up the Bayesian GPLVM for single-cell data`,<br />&nbsp;&nbsp;&nbsp;&nbsp;\n",
    "Sumon Ahmed, Magnus Rattray, Alexis Boukouvalas\n",
    "\n",
    "[3] `Bayesian Gaussian Process Latent Variable Model`,<br />&nbsp;&nbsp;&nbsp;&nbsp;\n",
    "Michalis K. Titsias, Neil D. Lawrence\n",
    "\n",
    "[4] `A novel approach for resolving differences in single-cell gene expression patterns from zygote to blastocyst`,<br />&nbsp;&nbsp;&nbsp;&nbsp;\n",
    "Florian Buettner, Fabian J. Theis"
   ]
  }
 ],
 "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.6.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
