Skip to content

Commit

Permalink
Merge remote-tracking branch 'origin/main' into more_types_to_dummy_data
Browse files Browse the repository at this point in the history
  • Loading branch information
rcannood committed Nov 30, 2023
2 parents 8f8c8dd + 50418e1 commit 0560b0b
Show file tree
Hide file tree
Showing 12 changed files with 152 additions and 181 deletions.
113 changes: 41 additions & 72 deletions R/AbstractAnnData.R
Original file line number Diff line number Diff line change
Expand Up @@ -82,19 +82,27 @@ AbstractAnnData <- R6::R6Class("AbstractAnnData", # nolint
for (attribute in c(
"obs",
"var",
"uns",
# "uns", # TODO: remove this when uns is implemented
"obsm",
"varm",
"layers",
"obsp",
"varp"
)) {
attr_key <- paste0(attribute, "_keys")
if (!is.null(self[[attr_key]])) {
slot_keys <- self[[attr_key]]()
if (length(slot_keys) > 0) {
cat(" ", pretty_print(attribute, slot_keys), "\n", sep = "")
key_fun <- self[[paste0(attribute, "_keys")]]
keys <-
if (!is.null(key_fun)) {
key_fun()
} else {
NULL
}
if (length(keys) > 0) {
cat(
" ", attribute, ":",
paste("'", keys, "'", collapse = ", "),
"\n",
sep = ""
)
}
}
},
Expand Down Expand Up @@ -126,6 +134,22 @@ AbstractAnnData <- R6::R6Class("AbstractAnnData", # nolint
layers_keys = function() {
names(self$layers)
},
#' @description Keys (element names) of `obsm`.
obsm_keys = function() {
names(self$obsm)
},
#' @description Keys (element names) of `varm`.
varm_keys = function() {
names(self$varm)
},
#' @description Keys (element names) of `obsp`.
obsp_keys = function() {
names(self$obsp)
},
#' @description Keys (element names) of `varp`.
varp_keys = function() {
names(self$varp)
},
#' @description Convert to SingleCellExperiment
to_SingleCellExperiment = function() {
to_SingleCellExperiment(self)
Expand All @@ -145,57 +169,18 @@ AbstractAnnData <- R6::R6Class("AbstractAnnData", # nolint
}
),
private = list(
# @description `.validate_matrix()` checks that dimensions are
# consistent with `obs` and `var`, and removes dimnames if
# present.
# @param mat A matrix to validate
# @param label Must be `"X"` or `"layer[[...]]"` where `...` is
# the name of a layer.
.validate_matrix = function(mat, label) {
if (!is.null(mat)) {
if (nrow(mat) != self$n_obs()) {
stop("nrow(", label, ") should be the same as nrow(obs)")
}
if (ncol(mat) != self$n_vars()) {
stop("ncol(", label, ") should be the same as nrow(var)")
}

if (has_row_names(mat)) {
warning(wrap_message(
"rownames(", label, ") should be NULL, removing them from the matrix"
))
rownames(mat) <- NULL
}

for (i in seq(2, length(dim(mat)), by = 1)) {
if (!is.null(dimnames(mat)[[i]])) {
warning(wrap_message(
"dimnames(", label, ")[[", i, "]] should be NULL, removing them from the matrix"
))
dimnames(mat)[[i]] <- NULL
}
}
}

mat
},


# @description `.validate_aligned_array()` checks that dimensions are
# consistent with the anndata object.
# @param mat A matrix to validate
# @param label Must be `"X"` or `"layer[[...]]"` where `...` is
# the name of a layer.
# @param shape Expected dimensions of matrix
# @param expected_rownames Expected rownames of matrix
# @param excepted_colnames Expected colnames of matrix
.validate_aligned_array = function(
mat,
label,
shape,
expected_rownames = NULL,
expected_colnames = NULL
) {
# @param expected_rownames
# @param excepted_colnames
.validate_aligned_array = function(mat, label, shape, expected_rownames = NULL, expected_colnames = NULL) {
if (is.null(mat)) {
return(mat)
}
mat_dims <- dim(mat)
for (i in seq_along(shape)) {
expected_dim <- shape[i]
Expand All @@ -204,35 +189,19 @@ AbstractAnnData <- R6::R6Class("AbstractAnnData", # nolint
stop("dim(", label, ")[", i, "] should have shape: ", expected_dim, ", found: ", found_dim, ".")
}
}
if (has_row_names(mat)) {
if (!is.null(expected_rownames) & !has_row_names(mat)) {
if (!identical(rownames(mat), expected_rownames)) {
stop("rownames(", label, ") should be the same as expected_rownames")
}

warning(wrap_message(
"rownames(", label, ") should be NULL, removing them from the matrix"
))
rownames(mat) <- NULL
}

if (!is.null(expected_colnames) & !is.null(colnames(mat))) {
if (!identical(colnames(mat), expected_colnames)) {
stop("colnames(", label, ") should be the same as expected_colnames")
}
colnames(mat) <- NULL
}

# TODO: Determine whether this check makes sense. I'm thinking it does not.
# nolint start
# for (i in seq(2, length(dim(mat)), by = 1)) {
# if (!is.null(dimnames(mat)[[i]])) {
# warning(wrap_message(
# "dimnames(", label, ")[[", i, "]] should be NULL, removing them from the matrix"
# ))
# dimnames(mat)[[i]] <- NULL
# }
# }
# nolint end

mat
},
# @description `.validate_aligned_mapping()` checks for named lists and
Expand All @@ -241,8 +210,8 @@ AbstractAnnData <- R6::R6Class("AbstractAnnData", # nolint
# whose entries will be validated
# @param label The label of the collection, used for error messages
# @param shape Expected dimensions of arrays. Arrays may have more dimensions than specified here
# @param expected_rownames
# @param expected_colnames
# @param expected_rownames Expected row names
# @param expected_colnames Expected column names
.validate_aligned_mapping = function(collection, label, shape, expected_rownames = NULL, expected_colnames = NULL) {
if (is.null(collection)) {
return(collection)
Expand Down Expand Up @@ -296,7 +265,7 @@ AbstractAnnData <- R6::R6Class("AbstractAnnData", # nolint
))
}

if (.row_names_info(df) > 0) {
if (has_row_names(df)) {
warning(wrap_message(
"'", label, "' should not have any rownames, removing them from the data frame."
))
Expand Down
8 changes: 7 additions & 1 deletion R/HDF5AnnData.R
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,13 @@ HDF5AnnData <- R6::R6Class("HDF5AnnData", # nolint
read_h5ad_element(private$.h5obj, "/X")
} else {
# trackstatus: class=HDF5AnnData, feature=set_X, status=done
value <- private$.validate_matrix(value, "X")
value <- private$.validate_aligned_array(
value,
"X",
shape = c(self$n_obs(), self$n_vars()),
expected_rownames = rownames(self),
expected_colnames = colnames(self)
)
write_h5ad_element(value, private$.h5obj, "/X")
}
},
Expand Down
8 changes: 7 additions & 1 deletion R/InMemoryAnnData.R
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,13 @@ InMemoryAnnData <- R6::R6Class("InMemoryAnnData", # nolint
private$.X
} else {
# trackstatus: class=InMemoryAnnData, feature=set_X, status=done
private$.X <- private$.validate_matrix(value, "X")
private$.X <- private$.validate_aligned_array(
value,
"X",
shape = c(self$n_obs(), self$n_vars()),
expected_rownames = rownames(self),
expected_colnames = colnames(self)
)
self
}
},
Expand Down
8 changes: 0 additions & 8 deletions R/utilities.R

This file was deleted.

7 changes: 6 additions & 1 deletion R/has_row_names.R → R/utils.R
Original file line number Diff line number Diff line change
@@ -1,7 +1,12 @@
wrap_message <- function(...) {
txt <- paste0(..., collapse = "")
paste(strwrap(txt, exdent = 2L), collapse = "\n")
}

has_row_names <- function(x) {
if (is.data.frame(x)) {
.row_names_info(x) > 0
} else {
!is.null(dimnames(x)[[1]])
}
}
}
37 changes: 19 additions & 18 deletions R/write_h5ad.R
Original file line number Diff line number Diff line change
Expand Up @@ -42,24 +42,25 @@
#'
#' # Write a Seurat as a H5AD
#' if (requireNamespace("SeuratObject", quietly = TRUE)) {
#' h5ad_file <- tempfile(fileext = ".h5ad")
#' counts <- matrix(1:15, 3L, 5L)
#' dimnames(counts) <- list(
#' letters[1:3],
#' LETTERS[1:5]
#' )
#' gene.metadata <- data.frame(
#' row.names = LETTERS[1:5],
#' gene = 1:5
#' )
#' obj <- SeuratObject::CreateSeuratObject(counts, meta.data = gene.metadata)
#' cell.metadata <- data.frame(
#' row.names = letters[1:3],
#' cell = 1:3
#' )
#' obj <- SeuratObject::AddMetaData(obj, cell.metadata)
#'
#' write_h5ad(obj, h5ad_file)
#' # TODO: uncomment this code when the seurat converter is fixed
#' # h5ad_file <- tempfile(fileext = ".h5ad")
#' # counts <- matrix(1:15, 3L, 5L)
#' # dimnames(counts) <- list(
#' # letters[1:3],
#' # LETTERS[1:5]
#' # )
#' # gene.metadata <- data.frame(
#' # row.names = LETTERS[1:5],
#' # gene = 1:5
#' # )
#' # obj <- SeuratObject::CreateSeuratObject(counts, meta.data = gene.metadata)
#' # cell.metadata <- data.frame(
#' # row.names = letters[1:3],
#' # cell = 1:3
#' # )
#' # obj <- SeuratObject::AddMetaData(obj, cell.metadata)
#' #
#' # write_h5ad(obj, h5ad_file)
#' }
write_h5ad <- function(object, path) {
if (inherits(object, "SingleCellExperiment")) {
Expand Down
44 changes: 44 additions & 0 deletions man/AbstractAnnData.Rd

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

4 changes: 4 additions & 0 deletions man/HDF5AnnData.Rd

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

4 changes: 4 additions & 0 deletions man/InMemoryAnnData.Rd

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

Loading

0 comments on commit 0560b0b

Please sign in to comment.