From 182b53018a29be5120f4aa9ba2ad626190265273 Mon Sep 17 00:00:00 2001 From: John Kerl Date: Wed, 4 Dec 2024 16:17:23 -0500 Subject: [PATCH 1/7] Add `check_only` flag to shape/domain upgraders/resizers, for parity with Python --- apis/r/R/RcppExports.R | 12 ++++----- apis/r/R/SOMADataFrame.R | 28 +++++++++++++++----- apis/r/R/SOMANDArrayBase.R | 34 ++++++++++++++++++++----- apis/r/src/RcppExports.cpp | 36 +++++++++++++++----------- apis/r/src/rinterface.cpp | 52 ++++++++++++++++++++++++++++++++------ 5 files changed, 120 insertions(+), 42 deletions(-) diff --git a/apis/r/R/RcppExports.R b/apis/r/R/RcppExports.R index e49188b65e..cb06831a9a 100644 --- a/apis/r/R/RcppExports.R +++ b/apis/r/R/RcppExports.R @@ -238,20 +238,20 @@ c_schema <- function(uri, ctxxp) { .Call(`_tiledbsoma_c_schema`, uri, ctxxp) } -resize <- function(uri, new_shape, function_name_for_messages, ctxxp) { - invisible(.Call(`_tiledbsoma_resize`, uri, new_shape, function_name_for_messages, ctxxp)) +resize <- function(uri, new_shape, function_name_for_messages, check_only, ctxxp) { + .Call(`_tiledbsoma_resize`, uri, new_shape, function_name_for_messages, check_only, ctxxp) } resize_soma_joinid_shape <- function(uri, new_shape, function_name_for_messages, ctxxp) { invisible(.Call(`_tiledbsoma_resize_soma_joinid_shape`, uri, new_shape, function_name_for_messages, ctxxp)) } -tiledbsoma_upgrade_shape <- function(uri, new_shape, function_name_for_messages, ctxxp) { - invisible(.Call(`_tiledbsoma_tiledbsoma_upgrade_shape`, uri, new_shape, function_name_for_messages, ctxxp)) +tiledbsoma_upgrade_shape <- function(uri, new_shape, function_name_for_messages, check_only, ctxxp) { + .Call(`_tiledbsoma_tiledbsoma_upgrade_shape`, uri, new_shape, function_name_for_messages, check_only, ctxxp) } -upgrade_or_change_domain <- function(uri, is_change_domain, nadimap, nadimsp, function_name_for_messages, ctxxp) { - invisible(.Call(`_tiledbsoma_upgrade_or_change_domain`, uri, is_change_domain, nadimap, nadimsp, function_name_for_messages, ctxxp)) +upgrade_or_change_domain <- function(uri, is_change_domain, nadimap, nadimsp, function_name_for_messages, check_only, ctxxp) { + .Call(`_tiledbsoma_upgrade_or_change_domain`, uri, is_change_domain, nadimap, nadimsp, function_name_for_messages, check_only, ctxxp) } c_update_dataframe_schema <- function(uri, ctxxp, column_names_to_drop, add_cols_types, add_cols_enum_value_types, add_cols_enum_ordered) { diff --git a/apis/r/R/SOMADataFrame.R b/apis/r/R/SOMADataFrame.R index fd35d7c700..be595bd547 100644 --- a/apis/r/R/SOMADataFrame.R +++ b/apis/r/R/SOMADataFrame.R @@ -472,9 +472,11 @@ SOMADataFrame <- R6::R6Class( #' @param new_domain A named list, keyed by index-column name, with values #' being two-element vectors containing the desired lower and upper bounds #' for the domain. - #' @return No return value - tiledbsoma_upgrade_domain = function(new_domain) { - # stopifnot("'new_domain' must be CODE ME UP PLZ" = ... + #' @param check_only If true, does not apply the operation, but only reports + #' whether it would have succeeded. + #' @return No return value if `check_only` is `FALSE`. If `check_only` is `TRUE`, + #' returns the empty string if no error is detected, else a description of the error. + tiledbsoma_upgrade_domain = function(new_domain, check_only = FALSE) { # Checking slotwise new shape >= old shape, and <= max_shape, is already # done in libtiledbsoma @@ -482,16 +484,24 @@ SOMADataFrame <- R6::R6Class( new_domain, "tiledbsoma_upgrade_domain" ) - invisible( - upgrade_or_change_domain( + reason_string = ( + upgrade_or_change_domain( self$uri, FALSE, pyarrow_domain_table$array, pyarrow_domain_table$schema, .name_of_function(), + check_only, private$.soma_context ) ) + + if (check_only) { + return(reason_string) + } else { + # Return value is always "", or it raises an error trying. + invisible(reason_string) + } }, #' @description Allows you to set the domain of a `SOMADataFrame`, when the @@ -506,8 +516,11 @@ SOMADataFrame <- R6::R6Class( #' @param new_domain A named list, keyed by index-column name, with values #' being two-element vectors containing the desired lower and upper bounds #' for the domain. - #' @return No return value - change_domain = function(new_domain) { + #' @param check_only If true, does not apply the operation, but only reports + #' whether it would have succeeded. + #' @return No return value if `check_only` is `FALSE`. If `check_only` is `TRUE`, + #' returns the empty string if no error is detected, else a description of the error. + change_domain = function(new_domain, check_only=FALSE) { # stopifnot("'new_domain' must be CODE ME UP PLZ" = ... # Checking slotwise new shape >= old shape, and <= max_shape, is already # done in libtiledbsoma @@ -523,6 +536,7 @@ SOMADataFrame <- R6::R6Class( pyarrow_domain_table$array, pyarrow_domain_table$schema, .name_of_function(), + check_only, private$.soma_context ) ) diff --git a/apis/r/R/SOMANDArrayBase.R b/apis/r/R/SOMANDArrayBase.R index 2f3d5d27a7..08d17af619 100644 --- a/apis/r/R/SOMANDArrayBase.R +++ b/apis/r/R/SOMANDArrayBase.R @@ -102,15 +102,26 @@ SOMANDArrayBase <- R6::R6Class( #' doesn't already have a shape: in that case please call #' `tiledbsoma_upgrade_shape`. #' @param new_shape A vector of integerish, of the same length as the array's `ndim`. - #' @return No return value - resize = function(new_shape) { + #' @param check_only If true, does not apply the operation, but only reports + #' whether it would have succeeded. + #' @return No return value if `check_only` is `FALSE`. If `check_only` is `TRUE`, + #' returns the empty string if no error is detected, else a description of the error. + resize = function(new_shape, check_only=FALSE) { stopifnot( "'new_shape' must be a vector of integerish values, of the same length as maxshape" = rlang::is_integerish(new_shape, n = self$ndim()) || (bit64::is.integer64(new_shape) && length(new_shape) == self$ndim()) ) # Checking slotwise new shape >= old shape, and <= max_shape, is already done in libtiledbsoma - resize(self$uri, new_shape, .name_of_function(), private$.soma_context) + + reason_string = resize(self$uri, new_shape, .name_of_function(), check_only, private$.soma_context) + + if (check_only) { + return(reason_string) + } else { + # Return value is always "", or it raises an error trying. + invisible(reason_string) + } }, #' @description Allows the array to have a resizeable shape as described in the @@ -120,15 +131,26 @@ SOMANDArrayBase <- R6::R6Class( #' call on a pre-1.15 array the first time a shape is set on it; the latter must #' be used for subsequent resizes on any array which already has upgraded shape. #' @param shape A vector of integerish, of the same length as the array's `ndim`. - #' @return No return value - tiledbsoma_upgrade_shape = function(shape) { + #' @param check_only If true, does not apply the operation, but only reports + #' whether it would have succeeded. + #' @return No return value if `check_only` is `FALSE`. If `check_only` is `TRUE`, + #' returns the empty string if no error is detected, else a description of the error. + tiledbsoma_upgrade_shape = function(shape, check_only=FALSE) { stopifnot( "'shape' must be a vector of integerish values, of the same length as maxshape" = rlang::is_integerish(shape, n = self$ndim()) || (bit64::is.integer64(shape) && length(shape) == self$ndim()) ) # Checking slotwise new shape >= old shape, and <= max_shape, is already done in libtiledbsoma - tiledbsoma_upgrade_shape(self$uri, shape, .name_of_function(), private$.soma_context) + + reason_string = tiledbsoma_upgrade_shape(self$uri, shape, .name_of_function(), check_only, private$.soma_context) + if (check_only) { + return(reason_string) + } + else { + # Return value is always "", or it raises an error trying. + invisible(reason_string) + } } ), private = list( diff --git a/apis/r/src/RcppExports.cpp b/apis/r/src/RcppExports.cpp index 7c1121cafc..5bb38b57a2 100644 --- a/apis/r/src/RcppExports.cpp +++ b/apis/r/src/RcppExports.cpp @@ -553,16 +553,18 @@ BEGIN_RCPP END_RCPP } // resize -void resize(const std::string& uri, Rcpp::NumericVector new_shape, std::string function_name_for_messages, Rcpp::XPtr ctxxp); -RcppExport SEXP _tiledbsoma_resize(SEXP uriSEXP, SEXP new_shapeSEXP, SEXP function_name_for_messagesSEXP, SEXP ctxxpSEXP) { +std::string resize(const std::string& uri, Rcpp::NumericVector new_shape, std::string function_name_for_messages, bool check_only, Rcpp::XPtr ctxxp); +RcppExport SEXP _tiledbsoma_resize(SEXP uriSEXP, SEXP new_shapeSEXP, SEXP function_name_for_messagesSEXP, SEXP check_onlySEXP, SEXP ctxxpSEXP) { BEGIN_RCPP + Rcpp::RObject rcpp_result_gen; Rcpp::RNGScope rcpp_rngScope_gen; Rcpp::traits::input_parameter< const std::string& >::type uri(uriSEXP); Rcpp::traits::input_parameter< Rcpp::NumericVector >::type new_shape(new_shapeSEXP); Rcpp::traits::input_parameter< std::string >::type function_name_for_messages(function_name_for_messagesSEXP); + Rcpp::traits::input_parameter< bool >::type check_only(check_onlySEXP); Rcpp::traits::input_parameter< Rcpp::XPtr >::type ctxxp(ctxxpSEXP); - resize(uri, new_shape, function_name_for_messages, ctxxp); - return R_NilValue; + rcpp_result_gen = Rcpp::wrap(resize(uri, new_shape, function_name_for_messages, check_only, ctxxp)); + return rcpp_result_gen; END_RCPP } // resize_soma_joinid_shape @@ -579,31 +581,35 @@ BEGIN_RCPP END_RCPP } // tiledbsoma_upgrade_shape -void tiledbsoma_upgrade_shape(const std::string& uri, Rcpp::NumericVector new_shape, std::string function_name_for_messages, Rcpp::XPtr ctxxp); -RcppExport SEXP _tiledbsoma_tiledbsoma_upgrade_shape(SEXP uriSEXP, SEXP new_shapeSEXP, SEXP function_name_for_messagesSEXP, SEXP ctxxpSEXP) { +std::string tiledbsoma_upgrade_shape(const std::string& uri, Rcpp::NumericVector new_shape, std::string function_name_for_messages, bool check_only, Rcpp::XPtr ctxxp); +RcppExport SEXP _tiledbsoma_tiledbsoma_upgrade_shape(SEXP uriSEXP, SEXP new_shapeSEXP, SEXP function_name_for_messagesSEXP, SEXP check_onlySEXP, SEXP ctxxpSEXP) { BEGIN_RCPP + Rcpp::RObject rcpp_result_gen; Rcpp::RNGScope rcpp_rngScope_gen; Rcpp::traits::input_parameter< const std::string& >::type uri(uriSEXP); Rcpp::traits::input_parameter< Rcpp::NumericVector >::type new_shape(new_shapeSEXP); Rcpp::traits::input_parameter< std::string >::type function_name_for_messages(function_name_for_messagesSEXP); + Rcpp::traits::input_parameter< bool >::type check_only(check_onlySEXP); Rcpp::traits::input_parameter< Rcpp::XPtr >::type ctxxp(ctxxpSEXP); - tiledbsoma_upgrade_shape(uri, new_shape, function_name_for_messages, ctxxp); - return R_NilValue; + rcpp_result_gen = Rcpp::wrap(tiledbsoma_upgrade_shape(uri, new_shape, function_name_for_messages, check_only, ctxxp)); + return rcpp_result_gen; END_RCPP } // upgrade_or_change_domain -void upgrade_or_change_domain(const std::string& uri, bool is_change_domain, naxpArray nadimap, naxpSchema nadimsp, std::string function_name_for_messages, Rcpp::XPtr ctxxp); -RcppExport SEXP _tiledbsoma_upgrade_or_change_domain(SEXP uriSEXP, SEXP is_change_domainSEXP, SEXP nadimapSEXP, SEXP nadimspSEXP, SEXP function_name_for_messagesSEXP, SEXP ctxxpSEXP) { +std::string upgrade_or_change_domain(const std::string& uri, bool is_change_domain, naxpArray nadimap, naxpSchema nadimsp, std::string function_name_for_messages, bool check_only, Rcpp::XPtr ctxxp); +RcppExport SEXP _tiledbsoma_upgrade_or_change_domain(SEXP uriSEXP, SEXP is_change_domainSEXP, SEXP nadimapSEXP, SEXP nadimspSEXP, SEXP function_name_for_messagesSEXP, SEXP check_onlySEXP, SEXP ctxxpSEXP) { BEGIN_RCPP + Rcpp::RObject rcpp_result_gen; Rcpp::RNGScope rcpp_rngScope_gen; Rcpp::traits::input_parameter< const std::string& >::type uri(uriSEXP); Rcpp::traits::input_parameter< bool >::type is_change_domain(is_change_domainSEXP); Rcpp::traits::input_parameter< naxpArray >::type nadimap(nadimapSEXP); Rcpp::traits::input_parameter< naxpSchema >::type nadimsp(nadimspSEXP); Rcpp::traits::input_parameter< std::string >::type function_name_for_messages(function_name_for_messagesSEXP); + Rcpp::traits::input_parameter< bool >::type check_only(check_onlySEXP); Rcpp::traits::input_parameter< Rcpp::XPtr >::type ctxxp(ctxxpSEXP); - upgrade_or_change_domain(uri, is_change_domain, nadimap, nadimsp, function_name_for_messages, ctxxp); - return R_NilValue; + rcpp_result_gen = Rcpp::wrap(upgrade_or_change_domain(uri, is_change_domain, nadimap, nadimsp, function_name_for_messages, check_only, ctxxp)); + return rcpp_result_gen; END_RCPP } // c_update_dataframe_schema @@ -834,10 +840,10 @@ static const R_CallMethodDef CallEntries[] = { {"_tiledbsoma_c_dimnames", (DL_FUNC) &_tiledbsoma_c_dimnames, 2}, {"_tiledbsoma_c_attrnames", (DL_FUNC) &_tiledbsoma_c_attrnames, 2}, {"_tiledbsoma_c_schema", (DL_FUNC) &_tiledbsoma_c_schema, 2}, - {"_tiledbsoma_resize", (DL_FUNC) &_tiledbsoma_resize, 4}, + {"_tiledbsoma_resize", (DL_FUNC) &_tiledbsoma_resize, 5}, {"_tiledbsoma_resize_soma_joinid_shape", (DL_FUNC) &_tiledbsoma_resize_soma_joinid_shape, 4}, - {"_tiledbsoma_tiledbsoma_upgrade_shape", (DL_FUNC) &_tiledbsoma_tiledbsoma_upgrade_shape, 4}, - {"_tiledbsoma_upgrade_or_change_domain", (DL_FUNC) &_tiledbsoma_upgrade_or_change_domain, 6}, + {"_tiledbsoma_tiledbsoma_upgrade_shape", (DL_FUNC) &_tiledbsoma_tiledbsoma_upgrade_shape, 5}, + {"_tiledbsoma_upgrade_or_change_domain", (DL_FUNC) &_tiledbsoma_upgrade_or_change_domain, 7}, {"_tiledbsoma_c_update_dataframe_schema", (DL_FUNC) &_tiledbsoma_c_update_dataframe_schema, 6}, {"_tiledbsoma_sr_setup", (DL_FUNC) &_tiledbsoma_sr_setup, 10}, {"_tiledbsoma_sr_complete", (DL_FUNC) &_tiledbsoma_sr_complete, 1}, diff --git a/apis/r/src/rinterface.cpp b/apis/r/src/rinterface.cpp index 4b23663a90..eeda0f9b22 100644 --- a/apis/r/src/rinterface.cpp +++ b/apis/r/src/rinterface.cpp @@ -418,18 +418,29 @@ SEXP c_schema(const std::string& uri, Rcpp::XPtr ctxxp) { } // [[Rcpp::export]] -void resize( +std::string resize( const std::string& uri, Rcpp::NumericVector new_shape, std::string function_name_for_messages, + bool check_only, Rcpp::XPtr ctxxp) { // This function is solely for SparseNDArray and DenseNDArray for which the // dims are required by the SOMA spec to be of type int64. Domain-resize for // variant-indexed dataframes is via upgrade_domain and change_domain. auto sr = tdbs::SOMAArray::open(OpenMode::write, uri, ctxxp->ctxptr); std::vector new_shape_i64 = i64_from_rcpp_numeric(new_shape); - sr->resize(new_shape_i64, function_name_for_messages); + + std::string retval = ""; + if (check_only) { + auto status_and_reason = sr->can_resize( + new_shape_i64, function_name_for_messages); + retval = status_and_reason.second; + } else { + sr->resize(new_shape_i64, function_name_for_messages); + } + sr->close(); + return retval; } // [[Rcpp::export]] @@ -446,29 +457,40 @@ void resize_soma_joinid_shape( } // [[Rcpp::export]] -void tiledbsoma_upgrade_shape( +std::string tiledbsoma_upgrade_shape( const std::string& uri, Rcpp::NumericVector new_shape, std::string function_name_for_messages, + bool check_only, Rcpp::XPtr ctxxp) { // This function is solely for SparseNDArray and DenseNDArray for which the // dims are required by the SOMA spec to be of type int64. Domain-resize for // variant-indexed dataframes is via upgrade_domain and change_domain. auto sr = tdbs::SOMAArray::open(OpenMode::write, uri, ctxxp->ctxptr); std::vector new_shape_i64 = i64_from_rcpp_numeric(new_shape); - sr->upgrade_shape(new_shape_i64, function_name_for_messages); + + std::string retval = ""; + if (check_only) { + auto status_and_reason = sr->can_upgrade_shape( + new_shape_i64, function_name_for_messages); + retval = status_and_reason.second; + } else { + sr->upgrade_shape(new_shape_i64, function_name_for_messages); + } + sr->close(); + return retval; } // [[Rcpp::export]] -void upgrade_or_change_domain( +std::string upgrade_or_change_domain( const std::string& uri, bool is_change_domain, naxpArray nadimap, naxpSchema nadimsp, std::string function_name_for_messages, + bool check_only, Rcpp::XPtr ctxxp) { - // This is pointer manipulation from R -> Rcpp SEXP -> libtiledbsoma: nanoarrow::UniqueArray apdim{nanoarrow_array_from_xptr(nadimap)}; nanoarrow::UniqueSchema spdim{nanoarrow_schema_from_xptr(nadimsp)}; @@ -482,13 +504,27 @@ void upgrade_or_change_domain( tdbs::ArrowTable arrow_table(std::move(dimarr), std::move(dimsch)); // Now call libtiledbsoma + std::string reason_string = ""; auto sr = tdbs::SOMADataFrame::open(uri, OpenMode::write, ctxxp->ctxptr); if (is_change_domain) { - sr->change_domain(arrow_table, function_name_for_messages); + if (check_only) { + auto status_and_reason = sr->can_change_domain( + arrow_table, function_name_for_messages); + reason_string = status_and_reason.second; + } else { + sr->change_domain(arrow_table, function_name_for_messages); + } } else { - sr->upgrade_domain(arrow_table, function_name_for_messages); + if (check_only) { + auto status_and_reason = sr->can_upgrade_domain( + arrow_table, function_name_for_messages); + reason_string = status_and_reason.second; + } else { + sr->upgrade_domain(arrow_table, function_name_for_messages); + } } sr->close(); + return reason_string; } // [[Rcpp::export]] From 76eea5ee171241c330b713842dd3a94b342c4424 Mon Sep 17 00:00:00 2001 From: John Kerl Date: Wed, 4 Dec 2024 16:40:02 -0500 Subject: [PATCH 2/7] unit-test coverage --- apis/r/tests/testthat/test-11-shape.R | 25 +++++++++++++++++++++++-- 1 file changed, 23 insertions(+), 2 deletions(-) diff --git a/apis/r/tests/testthat/test-11-shape.R b/apis/r/tests/testthat/test-11-shape.R index 0b510ea071..180df72fa9 100644 --- a/apis/r/tests/testthat/test-11-shape.R +++ b/apis/r/tests/testthat/test-11-shape.R @@ -360,6 +360,21 @@ test_that("SOMADataFrame domain mods", { myint = c(20, 50), myfloat = c(0.0, 6.0) ) + + # -- first check dry run + expect_no_condition(sdf$change_domain(domain_for_create, check_only=TRUE)) + sdf$close() + + check <- list( + soma_joinid = c(0, 3), + mystring = c("", ""), # this is how it reads back + myint = c(20, 50), + myfloat = c(0.0, 6.0) + ) + expect_equal(sdf$domain(), check) + sdf$close() + + sdf <- SOMADataFrameOpen(uri, "WRITE") expect_no_condition(sdf$change_domain(new_domain)) # Shrink @@ -409,10 +424,11 @@ test_that("SOMADataFrame domain mods", { sdf$close() # Check for success - sdf <- SOMADataFrameOpen(uri, "WRITE") + sdf <- SOMADataFrameOpen(uri, "READ") dom <- sdf$domain() expect_equal(sdf$domain(), new_domain) sdf$close() + }) test_that("SOMASparseNDArray shape", { @@ -555,7 +571,12 @@ test_that("SOMADenseNDArray shape", { expect_error(ndarray$write(mat)) ndarray$close() - # Test resize up + # Test resize up, dry run + new_shape <- c(501, 601) + expect_no_error(reason_string <- ndarray$resize(new_shape, check_only=TRUE)) + expect_equal(reason_string, "") + + # Test resize up, for real new_shape <- c(500, 600) expect_no_error(ndarray$resize(new_shape)) From b1e7ebd4ac0d76e524d97001cfbf7d23fabc1b7a Mon Sep 17 00:00:00 2001 From: John Kerl Date: Thu, 5 Dec 2024 11:55:17 -0500 Subject: [PATCH 3/7] Apply suggestions from code review --- apis/r/R/SOMADataFrame.R | 28 +++++++++++++-------------- apis/r/R/SOMANDArrayBase.R | 20 +++++++++---------- apis/r/tests/testthat/test-11-shape.R | 12 ++++++------ 3 files changed, 28 insertions(+), 32 deletions(-) diff --git a/apis/r/R/SOMADataFrame.R b/apis/r/R/SOMADataFrame.R index be595bd547..8c72707970 100644 --- a/apis/r/R/SOMADataFrame.R +++ b/apis/r/R/SOMADataFrame.R @@ -484,24 +484,22 @@ SOMADataFrame <- R6::R6Class( new_domain, "tiledbsoma_upgrade_domain" ) - reason_string = ( - upgrade_or_change_domain( - self$uri, - FALSE, - pyarrow_domain_table$array, - pyarrow_domain_table$schema, - .name_of_function(), - check_only, - private$.soma_context - ) + reason_string <- upgrade_or_change_domain( + self$uri, + FALSE, + pyarrow_domain_table$array, + pyarrow_domain_table$schema, + .name_of_function(), + check_only, + private$.soma_context ) - if (check_only) { + if (isTRUE(check_only)) { return(reason_string) - } else { - # Return value is always "", or it raises an error trying. - invisible(reason_string) } + + # Return value is always "", or it raises an error trying. + return(invisible(reason_string)) }, #' @description Allows you to set the domain of a `SOMADataFrame`, when the @@ -520,7 +518,7 @@ SOMADataFrame <- R6::R6Class( #' whether it would have succeeded. #' @return No return value if `check_only` is `FALSE`. If `check_only` is `TRUE`, #' returns the empty string if no error is detected, else a description of the error. - change_domain = function(new_domain, check_only=FALSE) { + change_domain = function(new_domain, check_only = FALSE) { # stopifnot("'new_domain' must be CODE ME UP PLZ" = ... # Checking slotwise new shape >= old shape, and <= max_shape, is already # done in libtiledbsoma diff --git a/apis/r/R/SOMANDArrayBase.R b/apis/r/R/SOMANDArrayBase.R index 08d17af619..bf78f8a362 100644 --- a/apis/r/R/SOMANDArrayBase.R +++ b/apis/r/R/SOMANDArrayBase.R @@ -106,7 +106,7 @@ SOMANDArrayBase <- R6::R6Class( #' whether it would have succeeded. #' @return No return value if `check_only` is `FALSE`. If `check_only` is `TRUE`, #' returns the empty string if no error is detected, else a description of the error. - resize = function(new_shape, check_only=FALSE) { + resize = function(new_shape, check_only = FALSE) { stopifnot( "'new_shape' must be a vector of integerish values, of the same length as maxshape" = rlang::is_integerish(new_shape, n = self$ndim()) || @@ -114,14 +114,14 @@ SOMANDArrayBase <- R6::R6Class( ) # Checking slotwise new shape >= old shape, and <= max_shape, is already done in libtiledbsoma - reason_string = resize(self$uri, new_shape, .name_of_function(), check_only, private$.soma_context) + reason_string <- resize(self$uri, new_shape, .name_of_function(), check_only, private$.soma_context) if (check_only) { return(reason_string) - } else { - # Return value is always "", or it raises an error trying. - invisible(reason_string) } + + # Return value is always "", or it raises an error trying. + invisible(reason_string) }, #' @description Allows the array to have a resizeable shape as described in the @@ -135,7 +135,7 @@ SOMANDArrayBase <- R6::R6Class( #' whether it would have succeeded. #' @return No return value if `check_only` is `FALSE`. If `check_only` is `TRUE`, #' returns the empty string if no error is detected, else a description of the error. - tiledbsoma_upgrade_shape = function(shape, check_only=FALSE) { + tiledbsoma_upgrade_shape = function(shape, check_only = FALSE) { stopifnot( "'shape' must be a vector of integerish values, of the same length as maxshape" = rlang::is_integerish(shape, n = self$ndim()) || @@ -143,14 +143,12 @@ SOMANDArrayBase <- R6::R6Class( ) # Checking slotwise new shape >= old shape, and <= max_shape, is already done in libtiledbsoma - reason_string = tiledbsoma_upgrade_shape(self$uri, shape, .name_of_function(), check_only, private$.soma_context) + reason_string <- tiledbsoma_upgrade_shape(self$uri, shape, .name_of_function(), check_only, private$.soma_context) if (check_only) { return(reason_string) } - else { - # Return value is always "", or it raises an error trying. - invisible(reason_string) - } + # Return value is always "", or it raises an error trying. + return(invisible(reason_string)) } ), private = list( diff --git a/apis/r/tests/testthat/test-11-shape.R b/apis/r/tests/testthat/test-11-shape.R index 180df72fa9..088e9fa73e 100644 --- a/apis/r/tests/testthat/test-11-shape.R +++ b/apis/r/tests/testthat/test-11-shape.R @@ -362,14 +362,14 @@ test_that("SOMADataFrame domain mods", { ) # -- first check dry run - expect_no_condition(sdf$change_domain(domain_for_create, check_only=TRUE)) + expect_no_condition(sdf$change_domain(domain_for_create, check_only = TRUE)) sdf$close() check <- list( - soma_joinid = c(0, 3), - mystring = c("", ""), # this is how it reads back - myint = c(20, 50), - myfloat = c(0.0, 6.0) + soma_joinid = c(0, 3), + mystring = c("", ""), # this is how it reads back + myint = c(20, 50), + myfloat = c(0.0, 6.0) ) expect_equal(sdf$domain(), check) sdf$close() @@ -573,7 +573,7 @@ test_that("SOMADenseNDArray shape", { # Test resize up, dry run new_shape <- c(501, 601) - expect_no_error(reason_string <- ndarray$resize(new_shape, check_only=TRUE)) + expect_no_error(reason_string <- ndarray$resize(new_shape, check_only = TRUE)) expect_equal(reason_string, "") # Test resize up, for real From e110646be1d120374a944801ebbabddeb76fa4e3 Mon Sep 17 00:00:00 2001 From: John Kerl Date: Thu, 5 Dec 2024 12:14:13 -0500 Subject: [PATCH 4/7] more code-review feedback --- apis/r/R/SOMADataFrame.R | 5 +++-- apis/r/R/SOMANDArrayBase.R | 11 +++++++---- 2 files changed, 10 insertions(+), 6 deletions(-) diff --git a/apis/r/R/SOMADataFrame.R b/apis/r/R/SOMADataFrame.R index 8c72707970..779c374eff 100644 --- a/apis/r/R/SOMADataFrame.R +++ b/apis/r/R/SOMADataFrame.R @@ -498,8 +498,9 @@ SOMADataFrame <- R6::R6Class( return(reason_string) } - # Return value is always "", or it raises an error trying. - return(invisible(reason_string)) + # The return value from upgrade_or_change_domain without check_only is + # always "", or it raises an error trying. + return(invisible(NULL)) }, #' @description Allows you to set the domain of a `SOMADataFrame`, when the diff --git a/apis/r/R/SOMANDArrayBase.R b/apis/r/R/SOMANDArrayBase.R index bf78f8a362..aff1ef3cad 100644 --- a/apis/r/R/SOMANDArrayBase.R +++ b/apis/r/R/SOMANDArrayBase.R @@ -120,8 +120,9 @@ SOMANDArrayBase <- R6::R6Class( return(reason_string) } - # Return value is always "", or it raises an error trying. - invisible(reason_string) + # The return value from resize without check_only is always "", or it + # raises an error trying. + return(invisible(NULL)) }, #' @description Allows the array to have a resizeable shape as described in the @@ -147,8 +148,10 @@ SOMANDArrayBase <- R6::R6Class( if (check_only) { return(reason_string) } - # Return value is always "", or it raises an error trying. - return(invisible(reason_string)) + + # The return value from tiledbsoma_upgrade_shape without check_only is + # always "", or it raises an error trying. + return(invisible(NULL)) } ), private = list( From 1cf2af64dc4a66c35478df9ee65307f8bd5de3f9 Mon Sep 17 00:00:00 2001 From: John Kerl Date: Thu, 5 Dec 2024 12:18:24 -0500 Subject: [PATCH 5/7] more code-review feedback --- apis/r/R/SOMADataFrame.R | 18 ++++++++++++------ apis/r/R/SOMANDArrayBase.R | 11 +++++++---- 2 files changed, 19 insertions(+), 10 deletions(-) diff --git a/apis/r/R/SOMADataFrame.R b/apis/r/R/SOMADataFrame.R index 8c72707970..bbd3d8cbd3 100644 --- a/apis/r/R/SOMADataFrame.R +++ b/apis/r/R/SOMADataFrame.R @@ -498,8 +498,9 @@ SOMADataFrame <- R6::R6Class( return(reason_string) } - # Return value is always "", or it raises an error trying. - return(invisible(reason_string)) + # The return value from upgrade_or_change_domain without check_only is + # always "", or it raises an error trying. + return(invisible(NULL)) }, #' @description Allows you to set the domain of a `SOMADataFrame`, when the @@ -519,7 +520,6 @@ SOMADataFrame <- R6::R6Class( #' @return No return value if `check_only` is `FALSE`. If `check_only` is `TRUE`, #' returns the empty string if no error is detected, else a description of the error. change_domain = function(new_domain, check_only = FALSE) { - # stopifnot("'new_domain' must be CODE ME UP PLZ" = ... # Checking slotwise new shape >= old shape, and <= max_shape, is already # done in libtiledbsoma @@ -527,8 +527,7 @@ SOMADataFrame <- R6::R6Class( new_domain, tiledbsoma_upgrade_domain ) - invisible( - upgrade_or_change_domain( + reason_string <- upgrade_or_change_domain( self$uri, TRUE, pyarrow_domain_table$array, @@ -536,8 +535,15 @@ SOMADataFrame <- R6::R6Class( .name_of_function(), check_only, private$.soma_context - ) ) + + if (isTRUE(check_only)) { + return(reason_string) + } + + # The return value from upgrade_or_change_domain without check_only is + # always "", or it raises an error trying. + return(invisible(NULL)) } ), private = list( diff --git a/apis/r/R/SOMANDArrayBase.R b/apis/r/R/SOMANDArrayBase.R index bf78f8a362..aff1ef3cad 100644 --- a/apis/r/R/SOMANDArrayBase.R +++ b/apis/r/R/SOMANDArrayBase.R @@ -120,8 +120,9 @@ SOMANDArrayBase <- R6::R6Class( return(reason_string) } - # Return value is always "", or it raises an error trying. - invisible(reason_string) + # The return value from resize without check_only is always "", or it + # raises an error trying. + return(invisible(NULL)) }, #' @description Allows the array to have a resizeable shape as described in the @@ -147,8 +148,10 @@ SOMANDArrayBase <- R6::R6Class( if (check_only) { return(reason_string) } - # Return value is always "", or it raises an error trying. - return(invisible(reason_string)) + + # The return value from tiledbsoma_upgrade_shape without check_only is + # always "", or it raises an error trying. + return(invisible(NULL)) } ), private = list( From d2de816eb9e594e150e8c5d25be41d740dc48aab Mon Sep 17 00:00:00 2001 From: John Kerl Date: Thu, 5 Dec 2024 12:26:26 -0500 Subject: [PATCH 6/7] code-review feedback [skip ci] Co-authored-by: Paul Hoffman --- apis/r/R/SOMANDArrayBase.R | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/apis/r/R/SOMANDArrayBase.R b/apis/r/R/SOMANDArrayBase.R index aff1ef3cad..59b48d913f 100644 --- a/apis/r/R/SOMANDArrayBase.R +++ b/apis/r/R/SOMANDArrayBase.R @@ -116,7 +116,7 @@ SOMANDArrayBase <- R6::R6Class( reason_string <- resize(self$uri, new_shape, .name_of_function(), check_only, private$.soma_context) - if (check_only) { + if (isTRUE(check_only)) { return(reason_string) } @@ -145,7 +145,7 @@ SOMANDArrayBase <- R6::R6Class( # Checking slotwise new shape >= old shape, and <= max_shape, is already done in libtiledbsoma reason_string <- tiledbsoma_upgrade_shape(self$uri, shape, .name_of_function(), check_only, private$.soma_context) - if (check_only) { + if (isTRUE(check_only)) { return(reason_string) } From 0b64bfadc0abdf11e4a25a2e084130a77b88dfd2 Mon Sep 17 00:00:00 2001 From: John Kerl Date: Thu, 5 Dec 2024 12:27:44 -0500 Subject: [PATCH 7/7] apis/r/DESCRIPTION apis/r/NEWS.md --- apis/r/DESCRIPTION | 2 +- apis/r/NEWS.md | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/apis/r/DESCRIPTION b/apis/r/DESCRIPTION index e94dc1e157..96744f3ec3 100644 --- a/apis/r/DESCRIPTION +++ b/apis/r/DESCRIPTION @@ -6,7 +6,7 @@ Description: Interface for working with 'TileDB'-based Stack of Matrices, like those commonly used for single cell data analysis. It is documented at ; a formal specification available is at . -Version: 1.15.99.19 +Version: 1.15.99.20 Authors@R: c( person(given = "Aaron", family = "Wolen", role = c("cre", "aut"), email = "aaron@tiledb.com", diff --git a/apis/r/NEWS.md b/apis/r/NEWS.md index ea6b404947..5b4bdbbd65 100644 --- a/apis/r/NEWS.md +++ b/apis/r/NEWS.md @@ -27,6 +27,7 @@ * Update docstrings for `domain` argument to `create` [#3396](https://github.com/single-cell-data/TileDB-SOMA/pull/3396) * Vignette for new-shape feature [#3302](https://github.com/single-cell-data/TileDB-SOMA/pull/3302) * Fix blockwise iterator + re-indexer to return re-indexed shape instead of full domain +* Add `check_only` support for domain/shape updates [#3400](https://github.com/single-cell-data/TileDB-SOMA/pull/3400) # tiledbsoma 1.14.5