Hide code cell source

%load_ext autoreload
%autoreload 2

%matplotlib inline
import warnings

warnings.filterwarnings(
    "ignore",
    message="plotting functions contained within `_documentation_utils` are intended for nemos's documentation.",
    category=UserWarning,
)

warnings.filterwarnings(
    "ignore",
    message="Ignoring cached namespace 'core'",
    category=UserWarning,
)

warnings.filterwarnings(
    "ignore",
    message=(
        "invalid value encountered in div "
    ),
    category=RuntimeWarning,
)

Download

This notebook can be downloaded as place_cells-users.ipynb. See the button at the top right to download as markdown or pdf.

Model selection#

This notebook has had all its explanatory text removed and has not been run. It is intended to be downloaded and run locally (or on the provided binder) while listening to the presenter’s explanation. In order to see the fully rendered of this notebook, go here

Data for this notebook comes from recordings in the mouse hippocampus while the mouse runs on a linear track, which we explored yesterday.

Learning objectives#

  • Review how to use pynapple to analyze neuronal tuning

  • Learn how to combine NeMoS basis objects for modeling multiple predictors

import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
import pynapple as nap

import nemos as nmo

# some helper plotting functions
from nemos import _documentation_utils as doc_plots
import workshop_utils

# configure plots some
plt.style.use(nmo.styles.plot_style)

import workshop_utils

from sklearn import model_selection
from sklearn import pipeline

# shut down jax to numpy conversion warning
nap.nap_config.suppress_conversion_warnings = True

# during development, set this to a lower number so everything runs faster. 
cv_folds = 5

Pynapple#

  • Load the data using pynapple.

path = workshop_utils.fetch_data("Achilles_10252013_EEG.nwb")
data = nap.load_file(path)
data
  • Extract the spike times and mouse position.

spikes = data["units"]
position = data["position"]
  • Restrict data to when animal was traversing the linear track.

position = position.restrict(data["forward_ep"])
spikes = spikes.restrict(data["forward_ep"])
  • Restrict neurons to only excitatory neurons, discarding neurons with a low-firing rate.

spikes = spikes.getby_category("cell_type")["pE"]
spikes = spikes.getby_threshold("rate", 0.3)

Place fields#

  • Visualize the place fields: neuronal firing rate as a function of position.

place_fields = nap.compute_tuning_curves(spikes, position, bins=50, epochs=position.time_support, feature_names=["distance"])
workshop_utils.plot_place_fields(place_fields)
  • For speed, we’re only going to investigate the three neurons highlighted above.

  • Bin spikes to counts at 100 Hz.

  • Interpolate position to match spike resolution.

neurons = [82, 92, 220]
place_fields = place_fields.sel(unit=neurons)
spikes = spikes[neurons]
bin_size = .01
count = spikes.count(bin_size, ep=position.time_support)
position = position.interpolate(count, ep=count.time_support)
print(count.shape)
print(position.shape)

Speed modulation#

  • Compute animal’s speed for each epoch.

speed = []
# Analyzing each epoch separately avoids edge effects.
for s, e in position.time_support.values: 
    pos_ep = position.get(s, e)
    # Absolute difference of two consecutive points
    speed_ep = np.abs(np.diff(pos_ep)) 
    # Padding the edge so that the size is the same as the position/spike counts
    speed_ep = np.pad(speed_ep, [0, 1], mode="edge") 
    # Converting to cm/s 
    speed_ep = speed_ep * position.rate
    speed.append(speed_ep)

speed = nap.Tsd(t=position.t, d=np.hstack(speed), time_support=position.time_support)
print(speed.shape)
# enter code here
  • Visualize the position and speed tuning for these neurons.

fig = workshop_utils.plot_position_speed(position, speed, place_fields, tc_speed, neurons);

These neurons all show both position and speed tuning, and we see that the animal’s speed and position are highly correlated. We’re going to build a GLM to predict neuronal firing rate – which variable should we use? Is the speed tuning just epiphenomenal?

NeMoS#

Basis evaluation#

  • Create a separate basis object for each model input.

  • Visualize the basis objects.

# enter code here
  • Combine the two basis objects into a single “additive basis”

# enter code here
  • Create the design matrix!

  • Notice that, since we passed the basis pynapple objects, we got one back, preserving the time stamps.

  • X has the same number of time points as our input position and speed, but 25 columns. The columns come from n_basis_funcs from each basis (10 for position, 15 for speed).

# enter code here

Model learning#

  • Initialize PopulationGLM

  • Use the “LBFGS” solver and pass {"tol": 1e-12} to solver_kwargs.

  • Fit the data, passing the design matrix and spike counts to the glm object.

# enter code here

Prediction#

  • Use predict to check whether our GLM has captured each neuron’s speed and position tuning.

  • Remember to convert the predicted firing rate to spikes per second!

# enter code here
  • Compare model and data tuning curves together. The model did a pretty good job!

workshop_utils.plot_position_speed_tuning(place_fields, tc_speed, glm_pos, glm_speed);

We can see that this model does a good job capturing both the position and the speed. In the rest of this notebook, we’re going to investigate all the scientific decisions that we swept under the rug: should we regularize the model? what basis should we use? do we need both inputs?

To make our lives easier, let’s create a helper function that wraps the above lines, because we’re going to be visualizing our model predictions a lot.

# enter code here

In our previous analysis of the place field hyppocampal dataset we compared multiple encoding models and tried to figure out which predictor (position, speed or phase) had more explanatory power. In this notebook we will keep going on that effort and learn more principled (and convenient) approaches to model comparison combining NeMoS and scikit-learn.

Learning Objectives#

  • Learn how to use NeMoS objects with scikit-learn for cross-validation

  • Learn how to use NeMoS objects with scikit-learn pipelines

  • Learn how to use cross-validation to perform model and feature selection

Scikit-learn#

How to know when to regularize?#

  • How do we decide when to use regularization?

  • Cross-validation allows you to fairly compare different models on the same dataset.

  • NeMoS makes use of scikit-learn, the standard machine learning library in python.

  • Define parameter grid to search over.

  • Anything not specified in grid will be kept constant.

# enter code here
# enter code here
  • We interact with this in a very similar way to the glm object.

  • In particular, call fit with same arguments:

# enter code here
  • We got a warning because we didn’t specify the regularizer strength, so we just fell back on default value.

  • Let’s investigate results:

# enter code here

Select basis#

  • You can (and should) do something similar to determine how many basis functions you need for each input.

  • NeMoS basis objects are not scikit-learn-compatible right out of the box.

  • But we have provided a simple method to make them so:

# enter code here
  • This gives the basis object the transform method, which is equivalent to compute_features.

  • However, transformers have some limits:

# enter code here
  • Transformers only accept 2d inputs, whereas nemos basis objects can accept inputs of any dimensionality.

# enter code here
  • If the basis is composite (for example, the addition of two 1D bases), the transformer will expect a shape of (n_sampels, 1) each 1D component. If that’s not the case, you need to call set_input_shape:

# enter code here
  • Then you can call transform on the 2d input as expected.

# enter code here
  • You can, equivalently, call compute_features before turning the basis into a transformer. Then we cache the shape for future use:

# enter code here
  • Create a single TsdFrame to hold all our inputs:

transformer_input = nap.TsdFrame(
    t=position.t,
    d=np.stack([position, speed], 1),
    time_support=position.time_support,
    columns=["position", "speed"],
)
  • Pass this input to our transformed additive basis:

# enter code here

Pipelines#

  • If we want to cross-validate over the basis, we need more one more step: combining the basis and the GLM into a single scikit-learn estimator.

  • Pipelines to the rescue!

# enter code here
  • Pipeline runs basis.transform, then passes that output to glm, so we can do everything in a single line:

# enter code here
  • Visualize model predictions!

# enter code here

Cross-validating on the basis#

Now that we have our pipeline estimator, we can cross-validate on any of its parameters!

# enter code here

Let’s cross-validate on:

  • The number of the basis functions of the position basis

  • The functional form of the basis for speed

# enter code here
  • Construct param_grid, using __ to stand in for .

  • In sklearn pipelines, we access nested parameters using double underscores:

    • pipe["basis"]["position"].n_basis_funcs ← normal Python syntax

    • "basis__position__n_basis_funcs" ← sklearn parameter grid syntax

# enter code here
  • Cross-validate as before:

# enter code here
  • Investigate results:

# enter code here
  • Can easily grab the best estimator, the pipeline that did the best:

# enter code here
  • Visualize model predictions!

visualize_model_predictions(best_estim, transformer_input)

Feature selection#

# enter code here

Conclusion#

References#

The data in this tutorial comes from Grosmark, Andres D., and György Buzsáki. “Diversity in neural firing dynamics supports both rigid and learned hippocampal sequences.” Science 351.6280 (2016): 1440-1443.

# enter code here