Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Switch from leiden to leidenbase #6792

Merged
merged 16 commits into from
Dec 19, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions DESCRIPTION
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
Package: Seurat
Version: 5.1.0.9012
Version: 5.1.0.9013
Title: Tools for Single Cell Genomics
Description: A toolkit for quality control, analysis, and exploration of single cell RNA sequencing data. 'Seurat' aims to enable users to identify and interpret sources of heterogeneity from single cell transcriptomic measurements, and to integrate diverse types of single cell data. See Satija R, Farrell J, Gennert D, et al (2015) <doi:10.1038/nbt.3192>, Macosko E, Basu A, Satija R, et al (2015) <doi:10.1016/j.cell.2015.05.002>, Stuart T, Butler A, et al (2019) <doi:10.1016/j.cell.2019.05.031>, and Hao, Hao, et al (2020) <doi:10.1101/2020.10.12.335331> for more details.
Authors@R: c(
Expand Down Expand Up @@ -56,7 +56,7 @@ Imports:
irlba,
jsonlite,
KernSmooth,
leiden (>= 0.3.1),
leidenbase,
lifecycle,
lmtest,
MASS,
Expand Down
2 changes: 1 addition & 1 deletion NAMESPACE
Original file line number Diff line number Diff line change
Expand Up @@ -385,6 +385,7 @@ export(RunCCA)
export(RunGraphLaplacian)
export(RunICA)
export(RunLDA)
export(RunLeiden)
export(RunMarkVario)
export(RunMixscape)
export(RunMoransI)
Expand Down Expand Up @@ -754,7 +755,6 @@ importFrom(igraph,plot.igraph)
importFrom(irlba,irlba)
importFrom(jsonlite,fromJSON)
importFrom(jsonlite,read_json)
importFrom(leiden,leiden)
importFrom(lifecycle,deprecate_soft)
importFrom(lifecycle,deprecate_stop)
importFrom(lifecycle,deprecate_warn)
Expand Down
1 change: 1 addition & 0 deletions NEWS.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
# Unreleased

## Changes
- Updated `RunLeiden` to use the `leidenbase` package instead of `leiden`; deprecated the `method` parameter for `RunLeiden` and `FindClusters`; updated `RunLeiden` to reset `random.seed` to 1 if the value is 0 or less ([#6792](https://github.com/satijalab/seurat/pull/6792))
- Updated `RunUMAP` to support `umap-learn` version >= 0.5.0 ([#9559](https://github.com/satijalab/seurat/pull/9559))
- Surfaced more fine-grained control over what parts of a Xenium experiment are loaded in `LoadXenium`
- Added ability to load Xenium nucleus segmentation masks
Expand Down
174 changes: 104 additions & 70 deletions R/clustering.R
Original file line number Diff line number Diff line change
Expand Up @@ -275,14 +275,16 @@ PredictAssay <- function(
#' @importFrom future nbrOfWorkers
#'
#' @param modularity.fxn Modularity function (1 = standard; 2 = alternative).
#' @param initial.membership,node.sizes Parameters to pass to the Python leidenalg function.
#' @param initial.membership Passed to the `initial_membership` parameter
#' of `leidenbase::leiden_find_partition`.
#' @param node.sizes Passed to the `node_sizes` parameter of
#' `leidenbase::leiden_find_partition`.
#' @param resolution Value of the resolution parameter, use a value above
#' (below) 1.0 if you want to obtain a larger (smaller) number of communities.
#' @param algorithm Algorithm for modularity optimization (1 = original Louvain
#' algorithm; 2 = Louvain algorithm with multilevel refinement; 3 = SLM
#' algorithm; 4 = Leiden algorithm). Leiden requires the leidenalg python.
#' @param method Method for running leiden (defaults to matrix which is fast for small datasets).
#' Enable method = "igraph" to avoid casting large data to a dense matrix.
#' algorithm; 4 = Leiden algorithm).
#' @param method DEPRECATED.
#' @param n.start Number of random starts.
#' @param n.iter Maximal number of iterations per random start.
#' @param random.seed Seed of the random number generator.
Expand All @@ -303,7 +305,7 @@ FindClusters.default <- function(
initial.membership = NULL,
node.sizes = NULL,
resolution = 0.8,
method = "matrix",
method = deprecated(),
algorithm = 1,
n.start = 10,
n.iter = 10,
Expand All @@ -315,6 +317,14 @@ FindClusters.default <- function(
...
) {
CheckDots(...)
# The `method` parameter is for `RunLeiden` but was deprecated, see
# function for more details.
if (is_present(method)) {
deprecate_soft(
when = "5.2.0",
what = "FindClusters(method)"
)
}
if (is.null(x = object)) {
stop("Please provide an SNN graph")
}
Expand Down Expand Up @@ -344,7 +354,6 @@ FindClusters.default <- function(
} else if (algorithm == 4) {
ids <- RunLeiden(
object = object,
method = method,
partition.type = "RBConfigurationVertexPartition",
initial.membership = initial.membership,
node.sizes = node.sizes,
Expand Down Expand Up @@ -418,7 +427,8 @@ FindClusters.Seurat <- function(
initial.membership = NULL,
node.sizes = NULL,
resolution = 0.8,
method = "matrix",
# ToDo: Update `LogSeuratCommand` to accommodate deprecated parameters.
method = NULL,
algorithm = 1,
n.start = 10,
n.iter = 10,
Expand All @@ -430,6 +440,15 @@ FindClusters.Seurat <- function(
...
) {
CheckDots(...)
# Since we're throwing a soft deprecation warning, it needs to be duplicated
# for each implementation of the `FindClusters` generic, see
# `FindCluster.default` for more details.
if (!is.null(method)) {
deprecate_soft(
when = "5.2.0",
what = "FindClusters(method)"
)
}
graph.name <- graph.name %||% paste0(DefaultAssay(object = object), "_snn")
if (!graph.name %in% names(x = object)) {
stop("Provided graph.name not present in Seurat object")
Expand All @@ -443,7 +462,6 @@ FindClusters.Seurat <- function(
initial.membership = initial.membership,
node.sizes = node.sizes,
resolution = resolution,
method = method,
algorithm = algorithm,
n.start = n.start,
n.iter = n.iter,
Expand Down Expand Up @@ -1629,38 +1647,35 @@ NNHelper <- function(data, query = data, k, method, cache.index = FALSE, ...) {
return(n.ob)
}

# Run Leiden clustering algorithm
#
# Implements the Leiden clustering algorithm in R using reticulate
# to run the Python version. Requires the python "leidenalg" and "igraph" modules
# to be installed. Returns a vector of partition indices.
#
# @param adj_mat An adjacency matrix or SNN matrix
# @param partition.type Type of partition to use for Leiden algorithm.
# Defaults to RBConfigurationVertexPartition. Options include: ModularityVertexPartition,
# RBERVertexPartition, CPMVertexPartition, MutableVertexPartition,
# SignificanceVertexPartition, SurpriseVertexPartition (see the Leiden python
# module documentation for more details)
# @param initial.membership,node.sizes Parameters to pass to the Python leidenalg function.
# @param resolution.parameter A parameter controlling the coarseness of the clusters
# for Leiden algorithm. Higher values lead to more clusters. (defaults to 1.0 for
# partition types that accept a resolution parameter)
# @param random.seed Seed of the random number generator
# @param n.iter Maximal number of iterations per random start
#
# @keywords graph network igraph mvtnorm simulation
#
#' @importFrom leiden leiden
#' @importFrom reticulate py_module_available
#' Run Leiden clustering algorithm
#'
#' Returns a vector of partition indices.
#'
#' @param object An adjacency matrix or adjacency list.
#' @param method DEPRECATED.
#' @param partition.type Type of partition to use for Leiden algorithm.
#' Defaults to "RBConfigurationVertexPartition", see
#' https://cran.rstudio.com/web/packages/leidenbase/leidenbase.pdf for more options.
#' @param initial.membership Passed to the `initial_membership` parameter
#' of `leidenbase::leiden_find_partition`.
#' @param node.sizes Passed to the `node_sizes` parameter of
#' `leidenbase::leiden_find_partition`.
#' @param resolution.parameter A parameter controlling the coarseness of the clusters
#' for Leiden algorithm. Higher values lead to more clusters. (defaults to 1.0 for
#' partition types that accept a resolution parameter)
#' @param random.seed Seed of the random number generator, must be greater than 0.
#' @param n.iter Maximal number of iterations per random start
#'
#' @importFrom igraph graph_from_adjacency_matrix graph_from_adj_list
#
# @author Tom Kelly
#
# @export
#
#'
#' @export
#'
#' @rdname RunLeiden
#' @concept clustering
#'
RunLeiden <- function(
object,
method = c("matrix", "igraph"),
method = deprecated(),
partition.type = c(
'RBConfigurationVertexPartition',
'ModularityVertexPartition',
Expand All @@ -1673,50 +1688,69 @@ RunLeiden <- function(
initial.membership = NULL,
node.sizes = NULL,
resolution.parameter = 1,
random.seed = 0,
random.seed = 1,
n.iter = 10
) {
if (!py_module_available(module = 'leidenalg')) {
# `leidenbase::leiden_find_partition` requires it's `seed` parameter to be
# greater than 0 (or NULL) but the default value for `FindClusters` is 0.
# If `random.seed` is 0 or less, throw a warning and reset the value to 1.
if (!is.null(random.seed) && random.seed <= 0) {
warning(
paste0(
"`random.seed` must be greater than 0 for leiden clustering, ",
"resetting `random.seed` to 1."
)
)
random.seed <- 1
}

# The `method` parameter was deprecated after switching from the `leiden`
# package to `leidenbase` to run the algorithm. Unlike `leiden`, `leidenbase`
# _requires_ an `igraph` input, so the parameter no longer makes sense. The
# good news is that `leidenbase` is much faster than `leiden` so it shouldn't
# really matter.
if (is_present(method)) {
deprecate_soft(
when = "5.2.0",
what = "RunLeiden(method)"
)
}

# Convert `object` into an `igraph`.
# If `object` is already an `igraph` no conversion is necessary.
if (inherits(object, what = "igraph")) {
input <- object
# Otherwise, if `object` is a list, assume it is an adjacency list...
} else if (inherits(object, what = "list")) {
# And convert it to an `igraph` with the appropriate method.
input <- graph_from_adj_list(object)
# Or, if `object` is a matrix...
} else if (inherits(object, what = c("dgCMatrix", "matrix", "Matrix"))) {
# Make sure the matrix is sparse.
if (inherits(object, what = "Graph")) {
object <- as.sparse(object)
}
# And then convert it to an graph.
input <- graph_from_adjacency_matrix(object, weighted = TRUE)
# Throw an error if `object` is of an unknown type.
} else {
stop(
"Cannot find Leiden algorithm, please install through pip (e.g. pip install leidenalg).",
"Method for Leiden not found for class", class(object),
call. = FALSE
)
}
switch(
EXPR = method,
"matrix" = {
input <- as(object = object, Class = "matrix")
},
"igraph" = {
input <- if (inherits(x = object, what = 'list')) {
graph_from_adj_list(adjlist = object)
} else if (inherits(x = object, what = c('dgCMatrix', 'matrix', 'Matrix'))) {
if (inherits(x = object, what = 'Graph')) {
object <- as.sparse(x = object)
}
graph_from_adjacency_matrix(adjmatrix = object, weighted = TRUE)
} else if (inherits(x = object, what = 'igraph')) {
object
} else {
stop(
"Method for Leiden not found for class", class(x = object),
call. = FALSE
)
}
},
stop("Method for Leiden must be either 'matrix' or igraph'")
)
#run leiden from CRAN package (calls python with reticulate)
partition <- leiden(
object = input,

# Run clustering with `leidenbase`.
partition <- leidenbase::leiden_find_partition(
input,
partition_type = partition.type,
initial_membership = initial.membership,
weights = NULL,
edge_weights = NULL,
node_sizes = node.sizes,
resolution_parameter = resolution.parameter,
seed = random.seed,
n_iterations = n.iter
)
num_iter = n.iter
)$membership
return(partition)
}

Expand Down
15 changes: 9 additions & 6 deletions man/FindClusters.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion man/FindSubCluster.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

46 changes: 46 additions & 0 deletions man/RunLeiden.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading
Loading