Manage a pathway registry#
Pathways represent interconnected molecular networks of signaling cascades that govern critical cellular processes. They provide understandings cellular behavior mechanisms, insights of disease progression and treatment responses. In an R&D organization, managing pathways across different datasets are crucial for gaining insights of potential therapeutic targets and intervention strategies.
In this notebook we manage a pathway registry based on β2023 GO Biological Processβ ontology. Weβll walk you through the following steps:
Register pathways and link them to genes.
Perform a pathway enrichment analysis on an interferon-beta treated dataset and track the dataset with LaminDB.
Demonstrate how datasets are queryable by pathways and genes using LaminDB.
Setup#
Warning
Please ensure that you have created or loaded a LaminDB instance before running the remaining part of this notebook!
# A lamindb instance containing bionty schema (skip if you already loaded your own instance)
!lamin init --storage ./enrichr --schema bionty
Show code cell output
π‘ creating schemas: core==0.46.3 bionty==0.30.2
β
saved: User(id='DzTjkKse', handle='testuser1', email='testuser1@lamin.ai', name='Test User1', updated_at=2023-08-30 13:56:14)
β
saved: Storage(id='VJowFHXB', root='/home/runner/work/lamin-usecases/lamin-usecases/docs/enrichr', type='local', updated_at=2023-08-30 13:56:14, created_by_id='DzTjkKse')
β
loaded instance: testuser1/enrichr
π‘ did not register local instance on hub (if you want, call `lamin register`)
import lamindb as ln
import lnschema_bionty as lb
from lamin_usecases import datasets as ds
import gseapy as gp
import scanpy as sc
import matplotlib.pyplot as plt
lb.settings.species = "human" # globally set species
β
loaded instance: testuser1/enrichr (lamindb 0.51.2)
Fetch GO pathways annotated with human genes using Enrichr#
First we fetch the βGO_Biological_Process_2023β pathways for humans using GSEApy which wraps GSEA and Enrichr.
go_bp = gp.get_library(name="GO_Biological_Process_2023", organism="Human")
print(f"Number of pathways {len(go_bp)}")
Number of pathways 5406
go_bp["ATF6-mediated Unfolded Protein Response (GO:0036500)"]
['MBTPS1', 'MBTPS2', 'XBP1', 'ATF6B', 'DDIT3', 'CREBZF']
Parse out the ontology_id from keys, convert into the format of {ontology_id: (name, genes)}
def parse_ontology_id_from_keys(key):
"""Parse out the ontology id.
"ATF6-mediated Unfolded Protein Response (GO:0036500)" -> ("GO:0036500", "ATF6-mediated Unfolded Protein Response")
"""
id = key.split(" ")[-1].replace("(", "").replace(")", "")
name = key.replace(f" ({id})", "")
return (id, name)
go_bp_parsed = {}
for key, genes in go_bp.items():
id, name = parse_ontology_id_from_keys(key)
go_bp_parsed[id] = (name, genes)
go_bp_parsed["GO:0036500"]
('ATF6-mediated Unfolded Protein Response',
['MBTPS1', 'MBTPS2', 'XBP1', 'ATF6B', 'DDIT3', 'CREBZF'])
Register pathway ontology in LaminDB#
pathway_bionty = lb.Pathway.bionty() # equals to bionty.Pathway()
Show code cell output
pathway_bionty
Pathway
Species: all
Source: go, 2023-05-10
#terms: 47514
π Pathway.df(): ontology reference table
π Pathway.lookup(): autocompletion of terms
π― Pathway.search(): free text search of terms
β
Pathway.validate(): strictly validate values
π§ Pathway.inspect(): full inspection of values
π½ Pathway.standardize(): convert to standardized names
πͺ Pathway.diff(): difference between two versions
π Pathway.ontology: Pronto.Ontology object
Next, we register all the pathways and genes in LaminDB to finally link pathways to genes.
Register pathway terms#
To register the pathways we make use of .from_values
to directly parse the annotated GO pathway ontology IDs into LaminDB.
pathway_records = lb.Pathway.from_values(go_bp_parsed.keys(), lb.Pathway.ontology_id)
β
created 5406 Pathway records from Bionty matching ontology_id: GO:0044208, GO:0051084, GO:0006103, GO:0061158, GO:0070935, GO:0050427, GO:0042791, GO:0009452, GO:0036261, GO:0006370, GO:0015866, GO:0006167, GO:0046033, GO:0036500, GO:0006754, GO:0046034, GO:0042773, GO:0015867, GO:0086016, GO:0086067, ...
lb.Pathway.from_bionty(ontology_id="GO:0015868")
β
created 1 Pathway record from Bionty matching ontology_id: GO:0015868
Pathway(id='SMqshx3Y', name='purine ribonucleotide transport', ontology_id='GO:0015868', description='The Directed Movement Of A Purine Ribonucleotide, Any Compound Consisting Of A Purine Ribonucleoside (A Purine Organic Base Attached To A Ribose Sugar) Esterified With (Ortho)Phosphate, Into, Out Of Or Within A Cell.', bionty_source_id='VGN4', created_by_id='DzTjkKse')
ln.save(pathway_records, parents=False) # not recursing through parents
Register gene symbols#
Similarly, we use .from_values
for all Pathway associated genes to register them with LaminDB.
all_genes = {g for genes in go_bp.values() for g in genes}
gene_records = lb.Gene.from_values(all_genes, lb.Gene.symbol)
Show code cell output
β
created 14620 Gene records from Bionty matching symbol: CFAP70, KCNB2, LIX1L, TMPRSS11D, GPRC6A, FADD, ZAR1L, VEGFD, FRY, PRKD1, SGPL1, CTNS, TSEN2, CNTFR, CSRP1, NDUFV3, RAMP1, SPNS2, SPG11, CYSLTR1, ...
β
created 40 Gene records from Bionty matching synonyms: C19ORF12, C1ORF112, C9ORF72, C11ORF65, C8ORF88, C17ORF97, C1ORF131, C12ORF50, C12ORF4, PDZD3, C11ORF80, C17ORF75, C6ORF89, C3ORF70, C1ORF56, SLC9A3R2, C10ORF90, C12ORF29, C18ORF25, C20ORF173, ...
β ambiguous validation in Bionty for 1082 records: C4B, OR10AC1, OR2H2, APOB, NDUFA11, HAPLN2, DPY19L2P2, CD300H, DEFA6, WNT9B, DEFB109B, MARF1, ASB2, ACACA, SOX7, DUSP11, SPAG11B, PNMA5, TLCD2, IRF9, ...
β did not create Gene records for 37 non-validated symbols: AFD1, AZF1, CCL4L1, DGS2, DUX3, DUX5, FOXL3-OT1, IGL, LOC100653049, LOC102723475, LOC102723996, LOC102724159, LOC107984156, LOC112268384, LOC122319436, LOC122513141, LOC122539214, LOC344967, MDRV, MTRNR2L1, ...
gene_records[:3]
[Gene(id='z0rDUMrURdpw', symbol='CFAP70', ensembl_gene_id='ENSG00000156042', ncbi_gene_ids='118491', biotype='protein_coding', description='cilia and flagella associated protein 70 [Source:HGNC Symbol;Acc:HGNC:30726]', synonyms='TTC18|FLJ25765', species_id='uHJU', bionty_source_id='oX3h', created_by_id='DzTjkKse'),
Gene(id='Qp4RYlFoYnXP', symbol='KCNB2', ensembl_gene_id='ENSG00000182674', ncbi_gene_ids='9312', biotype='protein_coding', description='potassium voltage-gated channel subfamily B member 2 [Source:HGNC Symbol;Acc:HGNC:6232]', synonyms='KV2.2', species_id='uHJU', bionty_source_id='oX3h', created_by_id='DzTjkKse'),
Gene(id='Mml0lnmLnfEL', symbol='LIX1L', ensembl_gene_id='ENSG00000271601', ncbi_gene_ids='128077', biotype='protein_coding', description='limb and CNS expressed 1 like [Source:HGNC Symbol;Acc:HGNC:28715]', synonyms='MGC46719', species_id='uHJU', bionty_source_id='oX3h', created_by_id='DzTjkKse')]
ln.save(gene_records);
Link pathway to genes#
Now that we are tracking all pathways and genes records, we can link both of them to make the pathways even more queryable.
gene_records_ids = {record.symbol: record for record in gene_records}
for pathway_record in pathway_records:
pathway_genes = go_bp_parsed.get(pathway_record.ontology_id)[1]
pathway_genes_records = [gene_records_ids.get(gene) for gene in pathway_genes]
pathway_record.genes.set(pathway_genes_records)
Now genes are linked to pathways:
pathway_record.genes.list("symbol")
['XIAP', 'CAST', 'CARD8', 'CARD18', 'CST7']
A interferon-beta treated dataset#
We will now conduct a pathway enrichment analysis on a small peripheral blood mononuclear cell dataset that is split into control and stimulated groups. The stimulated group was treated with interferon beta.
Letβs load the dataset and look at the cell type annotations.
adata = ds.anndata_seurat_ifnb()
Show code cell output
/opt/hostedtoolcache/Python/3.9.17/x64/lib/python3.9/site-packages/anndata/_core/anndata.py:1113: FutureWarning: is_categorical_dtype is deprecated and will be removed in a future version. Use isinstance(dtype, CategoricalDtype) instead
if not is_categorical_dtype(df_full[k]):
adata
AnnData object with n_obs Γ n_vars = 13999 Γ 14053
obs: 'orig.ident', 'nCount_RNA', 'nFeature_RNA', 'stim', 'seurat_annotations'
var: 'features'
uns: 'log1p'
adata.obs["seurat_annotations"].value_counts()
seurat_annotations
CD14 Mono 4362
CD4 Naive T 2504
CD4 Memory T 1762
CD16 Mono 1044
B 978
CD8 T 814
NK 633
T activated 619
DC 472
B Activated 388
Mk 236
pDC 132
Eryth 55
Name: count, dtype: int64
For simplicity, we subset to βB Activatedβ cells:
adata_ba = adata[adata.obs.seurat_annotations == "B Activated"].copy()
adata_ba
/opt/hostedtoolcache/Python/3.9.17/x64/lib/python3.9/site-packages/anndata/_core/anndata.py:1113: FutureWarning: is_categorical_dtype is deprecated and will be removed in a future version. Use isinstance(dtype, CategoricalDtype) instead
if not is_categorical_dtype(df_full[k]):
AnnData object with n_obs Γ n_vars = 388 Γ 14053
obs: 'orig.ident', 'nCount_RNA', 'nFeature_RNA', 'stim', 'seurat_annotations'
var: 'features'
uns: 'log1p'
Pathway enrichment analysis using Enrichr#
This analysis is based on the GSEApy scRNA-seq Example.
First, we compute differentially expressed genes using a Wilcoxon test between stimulated and control cells.
# compute differentially expressed genes
sc.tl.rank_genes_groups(
adata_ba,
groupby="stim",
use_raw=False,
method="wilcoxon",
groups=["STIM"],
reference="CTRL",
)
rank_genes_groups_df = sc.get.rank_genes_groups_df(adata_ba, "STIM")
/opt/hostedtoolcache/Python/3.9.17/x64/lib/python3.9/site-packages/numpy/core/fromnumeric.py:84: FutureWarning: The behavior of DataFrame.sum with axis=None is deprecated, in a future version this will reduce over both axes and return a scalar. To retain the old behavior, pass axis=0 (or do not pass axis)
return reduction(axis=axis, out=out, **passkwargs)
rank_genes_groups_df.head()
names | scores | logfoldchanges | pvals | pvals_adj | |
---|---|---|---|---|---|
0 | ISG15 | 16.881584 | 5.923428 | 6.147295e-64 | 6.536230e-60 |
1 | ISG20 | 16.857113 | 4.167713 | 9.302256e-64 | 6.536230e-60 |
2 | IFIT3 | 14.587233 | 31.232290 | 3.386569e-48 | 1.586382e-44 |
3 | IFI6 | 14.128634 | 6.471180 | 2.530019e-45 | 8.888589e-42 |
4 | MX1 | 13.442097 | 6.241539 | 3.425901e-41 | 9.628837e-38 |
Next, we filter out up/down-regulated differentially expressed gene sets:
degs_up = rank_genes_groups_df[
(rank_genes_groups_df["logfoldchanges"] > 0)
& (rank_genes_groups_df["pvals_adj"] < 0.05)
]
degs_dw = rank_genes_groups_df[
(rank_genes_groups_df["logfoldchanges"] < 0)
& (rank_genes_groups_df["pvals_adj"] < 0.05)
]
degs_up.shape, degs_dw.shape
((89, 5), (47, 5))
Run pathway enrichment analysis on DEGs and plot top 10 pathways:
enr_up = gp.enrichr(degs_up.names, gene_sets="GO_Biological_Process_2023").res2d
gp.dotplot(enr_up, figsize=(2, 3), title="Up", cmap=plt.cm.autumn_r);
/opt/hostedtoolcache/Python/3.9.17/x64/lib/python3.9/site-packages/gseapy/plot.py:689: FutureWarning: The 'method' keyword in Series.replace is deprecated and will be removed in a future version.
df[self.colname].replace(
enr_dw = gp.enrichr(degs_dw.names, gene_sets="GO_Biological_Process_2023").res2d
gp.dotplot(enr_dw, figsize=(2, 3), title="Down", cmap=plt.cm.winter_r, size=10);
/opt/hostedtoolcache/Python/3.9.17/x64/lib/python3.9/site-packages/gseapy/plot.py:689: FutureWarning: The 'method' keyword in Series.replace is deprecated and will be removed in a future version.
df[self.colname].replace(
Track datasets containing annotated pathways in LaminDB#
Letβs enable tracking of the current notebook as the transform of this file:
ln.track()
π‘ notebook imports: gseapy==1.0.6 lamin_usecases==0.0.1 lamindb==0.51.2 lnschema_bionty==0.30.2 matplotlib==3.7.2 scanpy==1.9.4
β
saved: Transform(id='6oxEIEduvo6wz8', name='Manage a pathway registry', short_name='enrichr', version='0', type=notebook, updated_at=2023-08-30 13:57:12, created_by_id='DzTjkKse')
β
saved: Run(id='YGWLHw72xHevSeBG76PS', run_at=2023-08-30 13:57:12, transform_id='6oxEIEduvo6wz8', created_by_id='DzTjkKse')
We further create a File object to track the dataset.
file = ln.File.from_anndata(
adata_ba, description="seurat_ifnb_activated_Bcells", var_ref=lb.Gene.symbol
)
π‘ file will be copied to default storage upon `save()` with key `None` ('.lamindb/sEW2b2wjq3vulIpywcan.h5ad')
π‘ parsing feature names of X stored in slot 'var'
β
9324 terms (66.30%) are validated for symbol
β 4729 terms (33.70%) are not validated for symbol: AL627309.1, RP11-206L10.2, LINC00115, KLHL17, C1orf159, ACAP3, CPSF3L, GLTPD1, RP4-758J18.2, AL645728.1, RP11-345P4.9, SLC35E2B, SLC35E2, RP5-892K4.1, C1orf86, AL590822.2, MORN1, RP3-395M20.12, RP3-395M20.9, FAM213B, ...
β
linked: FeatureSet(id='Mh2bGBMGFyQLa7DCo9qW', n=10599, type='float', registry='bionty.Gene', hash='ehMHlXCXiKumXLoHU96e', created_by_id='DzTjkKse')
π‘ parsing feature names of slot 'obs'
β 5 terms (100.00%) are not validated for name: orig.ident, nCount_RNA, nFeature_RNA, stim, seurat_annotations
β no validated features, skip creating feature set
ln.save(file)
β
saved 1 feature set for slot: 'var'
β
storing file 'sEW2b2wjq3vulIpywcan' at '.lamindb/sEW2b2wjq3vulIpywcan.h5ad'
We further create two feature sets for degs_up
and degs_dw
which we can later associate with the associated pathways:
degs_up_featureset = ln.FeatureSet.from_values(degs_up.names, lb.Gene.symbol)
Show code cell output
β
76 terms (85.40%) are validated for symbol
β 13 terms (14.60%) are not validated for symbol: EPSTI1, WARS, LAP3, SAMD9L, NMI, TMEM123, CMPK2, H3F3B, PSMA2.1, PHF11, CLEC2D, DDX58, CD48
degs_dw_featureset = ln.FeatureSet.from_values(degs_dw.names, lb.Gene.symbol)
Show code cell output
β
44 terms (93.60%) are validated for symbol
β 3 terms (6.40%) are not validated for symbol: GNB2L1, TMEM66, HLA-DQB1
Link the top 10 pathways to the corresponding differentially expressed genes:
# get ontology ids for the top 10 pathways
enr_up_top10 = [
pw_id[0] for pw_id in enr_up.head(10).Term.apply(parse_ontology_id_from_keys)
]
enr_dw_top10 = [
pw_id[0] for pw_id in enr_dw.head(10).Term.apply(parse_ontology_id_from_keys)
]
# get pathway records
enr_up_top10_pathways = lb.Pathway.from_values(enr_up_top10, lb.Pathway.ontology_id)
enr_dw_top10_pathways = lb.Pathway.from_values(enr_dw_top10, lb.Pathway.ontology_id)
Link feature sets to file:
file.features.add_feature_set(degs_up_featureset, slot="up-DEGs")
file.features.add_feature_set(degs_dw_featureset, slot="down-DEGs")
Associate the pathways to the differentially expressed genes:
degs_up_featureset.pathways.set(enr_up_top10_pathways)
degs_dw_featureset.pathways.set(enr_dw_top10_pathways)
degs_up_featureset.pathways.list("name")
['defense response to virus',
'negative regulation of viral genome replication',
'interleukin-27-mediated signaling pathway',
'positive regulation of interferon-beta production',
'negative regulation of viral process',
'regulation of viral genome replication',
'defense response to symbiont',
'response to interferon-beta',
'antiviral innate immune response',
'response to type II interferon']
Querying for pathways#
Querying for pathways is now simple with .filter
:
lb.Pathway.filter(name__contains="interferon-beta").df()
name | ontology_id | abbr | synonyms | description | bionty_source_id | updated_at | created_by_id | |
---|---|---|---|---|---|---|---|---|
id | ||||||||
SGYMKD7O | positive regulation of interferon-beta production | GO:0032728 | None | positive regulation of IFN-beta production|up-... | Any Process That Activates Or Increases The Fr... | VGN4 | 2023-08-30 13:56:29 | DzTjkKse |
GD9xCHBK | regulation of interferon-beta production | GO:0032648 | None | regulation of IFN-beta production | Any Process That Modulates The Frequency, Rate... | VGN4 | 2023-08-30 13:56:29 | DzTjkKse |
uu9GYFx2 | negative regulation of interferon-beta production | GO:0032688 | None | down regulation of interferon-beta production|... | Any Process That Stops, Prevents, Or Reduces T... | VGN4 | 2023-08-30 13:56:29 | DzTjkKse |
l06ZujxW | cellular response to interferon-beta | GO:0035458 | None | cellular response to fibroblast interferon|cel... | Any Process That Results In A Change In State ... | VGN4 | 2023-08-30 13:56:29 | DzTjkKse |
mCgM7JYR | response to interferon-beta | GO:0035456 | None | response to fiblaferon|response to fibroblast ... | Any Process That Results In A Change In State ... | VGN4 | 2023-08-30 13:56:29 | DzTjkKse |
Query pathways from a gene:
lb.Pathway.filter(genes__symbol="KIR2DL1").df()
name | ontology_id | abbr | synonyms | description | bionty_source_id | updated_at | created_by_id | |
---|---|---|---|---|---|---|---|---|
id | ||||||||
TSXmNUbN | immune response-inhibiting cell surface recept... | GO:0002767 | None | immune response-inhibiting cell surface recept... | The Series Of Molecular Signals Initiated By A... | VGN4 | 2023-08-30 13:56:29 | DzTjkKse |
Query files from a pathway:
ln.File.filter(feature_sets__pathways__name__icontains="interferon-beta").first()
File(id='sEW2b2wjq3vulIpywcan', suffix='.h5ad', accessor='AnnData', description='seurat_ifnb_activated_Bcells', size=5896640, hash='vAWd5emmLj0nv0E0x5LOSA', hash_type='md5', updated_at=2023-08-30 13:57:15, storage_id='VJowFHXB', transform_id='6oxEIEduvo6wz8', run_id='YGWLHw72xHevSeBG76PS', created_by_id='DzTjkKse')
Query featuresets from a pathway to learn from which geneset this pathway was computed:
pathway = lb.Pathway.filter(ontology_id="GO:0035456").one()
pathway
Pathway(id='mCgM7JYR', name='response to interferon-beta', ontology_id='GO:0035456', synonyms='response to fiblaferon|response to fibroblast interferon|response to interferon beta', description='Any Process That Results In A Change In State Or Activity Of A Cell Or An Organism (In Terms Of Movement, Secretion, Enzyme Production, Gene Expression, Etc.) As A Result Of An Interferon-Beta Stimulus. Interferon-Beta Is A Type I Interferon.', updated_at=2023-08-30 13:56:29, bionty_source_id='VGN4', created_by_id='DzTjkKse')
degs = ln.FeatureSet.filter(pathways__ontology_id=pathway.ontology_id).one()
Now we can get the list of genes that are differentially expressed and belong to this pathway:
pathway_genes = set(pathway.genes.list("symbol"))
degs_genes = set(degs.genes.list("symbol"))
pathway_genes.intersection(degs_genes)
{'BST2',
'IFI16',
'IFITM2',
'IFITM3',
'IRF1',
'OAS1',
'PLSCR1',
'STAT1',
'XAF1'}
# clean up test instance
!lamin delete --force enrichr
!rm -r ./enrichr
Show code cell output
π‘ deleting instance testuser1/enrichr
β
deleted instance settings file: /home/runner/.lamin/instance--testuser1--enrichr.env
β
instance cache deleted
β
deleted '.lndb' sqlite file
β consider manually deleting your stored data: /home/runner/work/lamin-usecases/lamin-usecases/docs/enrichr