A universal probabilistic spike count model reveals ongoing modulation of neural variability (NeurIPS 2021)
This is the code repository for this paper. Models are implemented in Python with dependencies on libraries listed below at the end. We also include a neural data analysis library that was written for constructing scalable neural encoding models using a modern deep learning framework. The baseline models, along with the universal count model proposed in our work, are implemented in the library and can be used for analysis of other neural datasets.
Here is where all the code for fitting models is located.
This script generates the two synthetic populations and saves them into ./data/
, both generated spike counts and behaviour as well as the encoding models. Note that the population data used in the paper has been included in ./data/
, running this script will overwrite those files!
Run commands based on the following formats into the command line:
python3 validation.py --cv -1 2 5 8 --gpu 0 --modes 0 --datatype 0 --ncvx 2 --lr 1e-2 --lr_2 1e-3 --batchsize 10000
This runs a model of mode 0 on synthetic data, with --cv
indicating which cross-validation fold to leave out for validation (-1 indicates using all data) and --gpu
indicating the GPU device to run on (if available). Line 188 in validation.py gives the definition of all modes (numbered 0 to 8), in particular the likelihood (1st element of tuple) and the input space (2d element of tuple) are specified. Note there is a 10-fold split of the data, hence the cv trial integers can range from -1 to 9 (with -1 using all data for training). lr
and lr_2
indicate the learning rates, with lr_2
for toroidal kernel lengthscales and variational standard deviations of the latent state posterior (lower for latent models as described in the paper). The flag --ncvx
refers to the number of runs to do (selecting the best fit model after completion to save). One can also specify --batchsize
, which can speed up training when larger depending on the memory capacity of the hardware used. For validation.py, the flag --datatype
can be 0 (heteroscedastic Conway-Maxwell-Poisson) or 1 (modulated Poisson).
python3 HDC.py --cv -1 1 3 6 --gpu 0 --modes 0 --ncvx 2 --lr 1e-2 --lr_2 1e-3 --binsize 40
Similarly, this runs a model of mode 0 on head direction cell data, with head direction cell data binned into 40 ms bins set by --binsize
. Line 108 in HDC.py gives the definition of all modes for the head direction cell models (numbered 0 to 11). All possible flags and their default values can be seen in the validation.py and HDC.py scripts. The file models.py defines the encoding models and uses the library code (neuroprob) to implement and run these probabilistic models.
In terms of neural data, the synthetic population data used in the paper and the head direction cell data is included in the ./data/
folder. All required modes in the analysis notebooks can be seen in the code as it loads trained models. Note that there are separate notebooks for synthetic (validation) and real (HDC) datasets. All trained models are stored in the ./checkpoint/
folder.
All commands needed for real data and synthetic data experiments are put into bash files run_HDC.sh
and run_synthetic.sh
for convenience. Inside the bash files, commands are grouped by categories, such as regression models or latent variable models. If you wish to run different modes or cross-validation runs grouped together above in parallel rather than sequentially, run the respective command with only a single --modes
or --cv
argument each time and repeat while looping through the list.
Here one can find all the Jupyter notebooks for analysis and plotting.
By running the analysis notebooks, we reproduce the plotting data for figures in the paper. Intermediate files (pickled) will be stored in the ./saves/
folder.
This loads the analysis results and plots paper figures in .pdf and .svg formats, exported to the ./output/
folder.
Here we present a short description of a preliminary version of the neural data analysis library used (called neuroprob) to facilitate constructing neural encoding models. The list below shows what has implemented for use so far, see the models.py file for an example of code utilizing the library.
There are three kinds of objects that form the building blocks:
- Input group p(X,Z) and q(Z)
- Mapping p(F|X,Z)
- Likelihood p(Y|F)
The overal generative model is specified along with the variational posterior through these primitives. Input groups can contain observed and latent variables, with different priors one can put onto the latent variables.
- Linear-nonlinear and GP mappings
- LVMs
- Toroidal latent space priors (Manifold GPLVM)
- AR(1) temporal prior on latents
- GLM filters
- spike-history couplings
- spike-spike couplings
- stimulus history
- Inhomogenenous renewal point processes
- Gamma
- Inverse Gaussian
- Log Normal
- Count process likelihoods
- Poisson
- Zero-inflated Poisson
- Negative binomial
- Conway-Maxwell-Poisson
- Universal (this work)
- Gaussian likelihoods
- Univariate