Atlas annotation with snapseed
We developed snapseed to quickly provide marker-based annotations on large datasets. This can be very handy to provide "seed annotations" for label-dependent integration methods such as scANVI or scPoli.
Snapseed is based on the idea that a marker gene of a cell type should be specific for that cell type and should be expressed in a large fraction of the cells of that type. As a measure of specificity, snapseed computes single-feature ROC-AUC scores for each marker gene in each cluster. Together with the fraction of cells expressing the gene, we obtain a score that is used to assign clusters to cell types.
In this notebook, we'll showcase how to use snapseed to annotate cell types in the Human Neural Organoid Atlas dataset.
import os
import scvi
import scanpy as sc
import pandas as pd
import numpy as np
from flax.core.frozen_dict import FrozenDict
import hnoca.snapseed as snap
from hnoca.snapseed.utils import read_yaml
os.chdir("/home/fleckj/scratch/hnoca/")
# Read input data
hnoca_adata = sc.read("HNOCA_hv2k.h5ad")
print(hnoca_adata)
In it's most basic form, snapeed takes adata object and a dict with marker genes for different cell types. A very simple example might look like this:
marker_dict = dict(
neuron = ['STMN2', 'DCX'],
progenitor = ['SOX2', 'VIM', 'NES'],
)
assignments = snap.annotate(hnoca_adata, marker_dict, group_name="leiden_pca_rss_80")
assignments = assignments.set_index('leiden_pca_rss_80')
The output of snapseed is a pandas dataframe with annotations for each cluster. We can join this information back to the adata object to visualize the annotations on the single-cell level.
cell_assign = assignments.loc[hnoca_adata.obs['leiden_pca_rss_80'].astype(str).values, :]
hnoca_adata.obs["snap_class"] = cell_assign["class"].values
# Plot the annotations
sc.pl.scatter(hnoca_adata, color=["snap_class", "STMN2", "VIM"], basis="umap_scpoli")
This looks quite reasonable, so let's make it a bit more interesting. In practice, cell type annotations are often hierachical, so we might want to take this into account in the annotation process. For example, we can first annotate progenitors and neurons and then split them up into more fine-grained subtypes and regional identities.
Snapseed allows this by recusively annotating cells based on a hierarchy of cell types and marker genes. Such a hierarchy can be provided as a YAML file like this one:
(The FrozenDict
is just to enable better printing of a large dictionary)
marker_hierarchy = read_yaml("brain_markers.yaml")
print(FrozenDict(marker_hierarchy))
assignments = snap.annotate_hierarchy(
hnoca_adata,
marker_hierarchy,
group_name="leiden_pca_rss_80"
)
Now, snapseeds outputs a dictionary with the assignment dataframe, as well as the hierarchy. Again, we can plot the result as in a UMAP.
cell_assign = assignments["assignments"].loc[hnoca_adata.obs['leiden_pca_rss_80'].astype(str).values, :]
hnoca_adata.obs["snap_level_1"] = cell_assign["level_1"].values
hnoca_adata.obs["snap_level_2"] = cell_assign["level_2"].values
hnoca_adata.obs["snap_level_3"] = cell_assign["level_3"].values
# Plot the annotations
sc.pl.scatter(
hnoca_adata,
color=["snap_level_1", "snap_level_2", "snap_level_3"],
basis="umap_scpoli"
)
Here, snapseed gives annotations for each level of the provided hierarchy.