A Python port of R limma for differential expression and abundance analysis.
pylimma is a comprehensive port of limma with demonstrated parity in output to the R package.
pylimma covers linear modelling with empirical Bayes moderation (lm_fit, contrasts_fit, e_bayes, treat, top_table), voom and RNA-seq precision weights, normalisation and background correction, batch correction and surrogate variables, duplicate correlation and array weights, competitive and self-contained gene-set testing (camera, roast, mroast, fry, romer, geneSetTest), differential splicing, diagnostics plots, and the supporting statistical utilities. Some microarray specific functionality that we do not anticipate is commonly used has not been ported.
It accepts three input idioms for any analysis:
- Flat array-like objects - a NumPy
ndarrayor a pandasDataFrameof genes x samples EList- a Python port of R limma'sEListclass, implemented as adictsubclass inpylimma.classeswith slotsE,weights,genes,targets,designAnnData- read fromadata.Xoradata.layers[layer], with shape-preserving outputs written back toadata.layers[...]and fits stored inadata.uns[...].
pip install pylimmaOptional extras:
pip install "pylimma[plot]" # matplotlib for diagnostic plots
pip install "pylimma[dev]" # pytest, ruff, mypy
pip install "pylimma[docs]" # sphinx + nbsphinxFor an editable development install from source:
git clone https://github.com/john-mulvey/pylimma.git
cd pylimma
pip install -e ".[dev,plot]"Requires Python >= 3.10. Documentation: pylimma.readthedocs.io.
The canonical limma workflow in Python mirrors the R workflow one-for-one: lm_fit, contrasts_fit, e_bayes then top_table. Parameter names and defaults follow R limma.
import numpy as np
import pandas as pd
from pylimma import lm_fit, contrasts_fit, e_bayes, top_table, make_contrasts
# 100 genes x 6 samples, two groups of three, real signal in the first 10 genes
rng = np.random.default_rng(0)
expr = rng.normal(size=(100, 6))
expr[:10, 3:] += 2.0
# design matrix - intercept plus a GroupB indicator
design = pd.DataFrame(
np.column_stack([np.ones(6), [0, 0, 0, 1, 1, 1]]),
columns=["Intercept", "GroupB"],
)
# contrasts - test GroupB against the intercept; equivalent to R's
# makeContrasts(GroupBvsA = GroupB, levels = design)
contrasts = make_contrasts(GroupBvsA="GroupB", levels=design)
fit = lm_fit(expr, design)
fit = contrasts_fit(fit, contrasts=contrasts)
fit = e_bayes(fit)
print(top_table(fit, coef="GroupBvsA").head())The result here is a pandas DataFrame with the standard limma columns (log_fc, ave_expr, t, p_value, adj_p_value, b), sorted by B-statistic (as is the case for the R package).
pylimma's output is validated against R limma on a reference corpus covering every public function. In many cases we demonstrate parity close to the precision limit of double-precision floating-point arithmetic. Agreement with the limma release ported (3.66.0) is tighter than the variation between different versions of limma itself: on voom-based pipelines, pylimma matches 3.66.0 to floating-point precision, whereas earlier limma releases (e.g. 3.54.2) diverge from 3.66.0 by up to a few percent.
Beyond the R-parity port, pylimma has been extended to operate natively on AnnData objects so that limma workflows slot into the scverse ecosystem alongside scanpy and muon. Any public function that consumes expression data (voom, lm_fit, normalize_between_arrays, array_weights, duplicate_correlation, ...) accepts an AnnData in place of a matrix. Shape-preserving outputs are written to adata.layers[...], fits to adata.uns[...], and results retrieved with the usual top_table / decide_tests calls:
import pylimma as pl
import anndata as ad
adata = ad.read_h5ad("expression.h5ad") # obs x var = samples x genes
pl.voom(adata, design) # writes adata.layers["voom_E"]
pl.lm_fit(adata, design) # stores fit in adata.uns["pylimma"]
pl.e_bayes(adata)
results = pl.top_table(adata, coef=1) # returns a DataFrameThe examples/ directory contains tutorial notebooks organised by workflow. Each tutorial is self-contained, and together they cover the three input idioms (flat array/DataFrame, EList, and AnnData) across five workflows: microarray differential expression, bulk RNA-seq with voom, bulk MS proteomics, single-cell pseudobulk differential expression, and differential splicing. Side-by-side R-vs-Python notebooks are provided for the four datasets where parity to R limma is the explicit demonstration.
The Chiaretti et al. acute lymphoblastic leukaemia microarray dataset (Chiaretti et al. 2004) demonstrates the classical limma microarray pipeline (lm_fit -> contrasts_fit -> e_bayes -> top_table) on the EList input idiom:
all_chiaretti_tutorial.ipynb- pylimma-only tutorialall_R_vs_Python.ipynb- side-by-side R limma vs pylimma comparison
The GSE60450 mouse mammary RNA-seq dataset (Fu et al. 2015) demonstrates the canonical voom pipeline (voom -> lm_fit -> e_bayes -> top_table) on a plain count matrix:
gse60450_tutorial.ipynb- pylimma-only tutorialgse60450_R_vs_Python.ipynb- side-by-side R limma vs pylimma comparison
The Yoruba HapMap RNA-seq dataset (Pickrell et al. 2010) demonstrates voom in combination with duplicate_correlation for handling technical replicates:
yoruba_tutorial.ipynb- pylimma-only tutorialyoruba_R_vs_Python.ipynb- side-by-side R limma vs pylimma comparison
The Mulvey et al. 2026 cardiac-tissue dataset (Mulvey et al. 2026) loads the data directly to an AnnData object, and then demonstrates the classical limma microarray-style pipeline (lm_fit -> contrasts_fit -> e_bayes -> top_table).
mulvey_tutorial.ipynb- pylimma-only tutorial
The Kang 2018 PBMC dataset (Kang et al. 2018) demonstrates the AnnData input idiom end-to-end - pseudobulk aggregation per cell type followed by per-cluster differential expression, with fits stored in adata.uns:
kang_pbmc_tutorial.ipynb- pylimma-only tutorial
The pasilla Drosophila exon-level dataset (Brooks et al. 2011) demonstrates the differential splicing workflow (diff_splice, top_splice, plot_splice):
pasilla_tutorial.ipynb- pylimma-only tutorialpasilla_R_vs_Python.ipynb- side-by-side R limma vs pylimma comparison
The Python implementation was written primarily by Claude Code (Opus 4.5-4.7), working under the direction and review of John Mulvey. I carry responsibility for correctness as the auditor of the port, but the large majority of the line-level Python was produced by Claude Code.
pylimma is a port, not a new method: no statistical methodology is introduced here. If you use pylimma in published work, please include two distinct citations:
- Software citation - the pylimma implementation itself.
- Methodology citation(s) - the relevant original limma paper(s) listed under limma below, for the statistical methods underlying your analysis.
pylimma is a port of the limma Bioconductor package. All statistical credit belongs to Gordon Smyth and the limma contributors listed in NOTICE.
The limma publications are:
-
Smyth, G. K. (2004). Linear models and empirical Bayes methods for assessing differential expression in microarray experiments. Statistical Applications in Genetics and Molecular Biology 3(1), Article 3. doi:10.2202/1544-6115.1027
-
Ritchie, M. E., Phipson, B., Wu, D., Hu, Y., Law, C. W., Shi, W., and Smyth, G. K. (2015). limma powers differential expression analyses for RNA-sequencing and microarray studies. Nucleic Acids Research 43(7), e47. doi:10.1093/nar/gkv007
-
Law, C. W., Chen, Y., Shi, W., and Smyth, G. K. (2014). voom: precision weights unlock linear model analysis tools for RNA-seq read counts. Genome Biology 15, R29. doi:10.1186/gb-2014-15-2-r29
-
Phipson, B., Lee, S., Majewski, I. J., Alexander, W. S., and Smyth, G. K. (2016). Robust hyperparameter estimation protects against hypervariable genes and improves power to detect differential expression. Annals of Applied Statistics 10(2), 946-963. doi:10.1214/16-AOAS920
-
Smyth, G. K., and Speed, T. P. (2003). Normalization of cDNA microarray data. Methods 31, 265-273. doi:10.1016/S1046-2023(03)00155-5
-
Ritchie, M. E., Silver, J., Oshlack, A., Holmes, M., Diyagama, D., Holloway, A., and Smyth, G. K. (2007). A comparison of background correction methods for two-colour microarrays. Bioinformatics 23(20), 2700-2707. doi:10.1093/bioinformatics/btm412
-
Silver, J. D., Ritchie, M. E., and Smyth, G. K. (2009). Microarray background correction: maximum likelihood estimation for the normal-exponential convolution. Biostatistics 10(2), 352-363. doi:10.1093/biostatistics/kxn042
-
Smyth, G. K., Michaud, J., and Scott, H. S. (2005). Use of within-array replicate spots for assessing differential expression in microarray experiments. Bioinformatics 21(9), 2067-2075. doi:10.1093/bioinformatics/bti270
-
Ritchie, M. E., Diyagama, D., Neilson, J., van Laar, R., Dobrovic, A., Holloway, A., and Smyth, G. K. (2006). Empirical array quality weights in the analysis of microarray data. BMC Bioinformatics 7, 261. doi:10.1186/1471-2105-7-261
-
Liu, R., Holik, A. Z., Su, S., Jansz, N., Chen, K., Leong, H. S., Blewitt, M. E., Asselin-Labat, M.-L., Smyth, G. K., and Ritchie, M. E. (2015). Why weight? Modelling sample and observational level variability improves power in RNA-seq analyses. Nucleic Acids Research 43(15), e97. doi:10.1093/nar/gkv412
-
McCarthy, D. J., and Smyth, G. K. (2009). Testing significance relative to a fold-change threshold is a TREAT. Bioinformatics 25(6), 765-771. doi:10.1093/bioinformatics/btp053
-
Wu, D., and Smyth, G. K. (2012). Camera: a competitive gene set test accounting for inter-gene correlation. Nucleic Acids Research 40(17), e133. doi:10.1093/nar/gks461
-
Wu, D., Lim, E., Vaillant, F., Asselin-Labat, M.-L., Visvader, J. E., and Smyth, G. K. (2010). ROAST: rotation gene set tests for complex microarray experiments. Bioinformatics 26(17), 2176-2182. doi:10.1093/bioinformatics/btq401
-
Majewski, I. J., Ritchie, M. E., Phipson, B., Corbin, J., Pakusch, M., Ebert, A., Busslinger, M., Koseki, H., Hu, Y., Smyth, G. K., Alexander, W. S., Hilton, D. J., and Blewitt, M. E. (2010). Opposing roles of polycomb repressive complexes in hematopoietic stem and progenitor cells. Blood 116(5), 731-739. doi:10.1182/blood-2009-12-260760
-
Phipson, B., and Smyth, G. K. (2010). Permutation P-values should never be zero: calculating exact P-values when permutations are randomly drawn. Statistical Applications in Genetics and Molecular Biology 9(1), Article 39. doi:10.2202/1544-6115.1585
-
Baldoni, P. L., Chen, L., Li, M., Chen, Y., and Smyth, G. K. (2025). Dividing out quantification uncertainty enables assessment of differential transcript usage with limma and edgeR. Nucleic Acids Research 53(22), gkaf1305. doi:10.1093/nar/gkaf1305
pylimma additionally ports algorithms from the R packages MASS, statmod, base R splines, and affy. Per-module attribution to upstream contributors is recorded in each ported module's SPDX header and summarised in NOTICE.
pylimma is a derivative work of limma. It is therefore licensed under the GNU General Public License v3.0 or later (GPL-3.0-or-later).
Contributions are welcome and wanted!
The kinds of contribution that are especially useful:
- Bug reports. We have taken great effort to comprehensively assess pylimma against limma. If there are differences in parity of outputs/fucntionality/interface, we would like to know about it!
- Additional R-parity fixtures for functions or parameter combinations not yet covered.
- Performance work on the hotter routines (for example
voom,camera,duplicate_correlation), where we currently are less performant than the C code contained within the limma R package.
The issue tracker at github.com/john-mulvey/pylimma/issues is the primary entry point. Please open an issue to discuss larger changes before sending a pull request.