From 59a103d1e9e4a612ed2ce998a5d90e90f91b1fed Mon Sep 17 00:00:00 2001 From: m7pr Date: Wed, 16 Oct 2024 12:04:47 +0200 Subject: [PATCH 01/24] deprecated datanames for teal.data::get_code --- NEWS.md | 5 +++++ R/teal_data-get_code.R | 3 ++- man/get_code.Rd | 2 +- 3 files changed, 8 insertions(+), 2 deletions(-) diff --git a/NEWS.md b/NEWS.md index 08d441316..b7fb3b300 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,5 +1,10 @@ # teal.data 0.6.0.9012 +### Deprecations + +- `get_code(datanames)` `S4` method parameter for `teal_data()` object was soft deprecated. Use `get_code(names)`. + + ### Enhancements - `datanames()` diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index d74435bd6..ebeda7f6a 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -1,6 +1,6 @@ #' Get code from `teal_data` object #' -#' Retrieve code from `teal_data` object. +#' `r lifecycle::badge("deprecated")` Retrieve code from `teal_data` object. #' #' Retrieve code stored in `@code`, which (in principle) can be used to recreate all objects found in `@env`. #' Use `datanames` to limit the code to one or more of the datasets enumerated in `@datanames`. @@ -108,6 +108,7 @@ setMethod("get_code", signature = "teal_data", definition = function(object, dep checkmate::assert_flag(deparse) code <- if (!is.null(datanames)) { + warning("get_code(datanames) was deprecated in teal.data 0.6.1, use get_code(names) instead.") get_code_dependency(object@code, datanames, ...) } else { object@code diff --git a/man/get_code.Rd b/man/get_code.Rd index 29b3a4134..09f31a755 100644 --- a/man/get_code.Rd +++ b/man/get_code.Rd @@ -24,7 +24,7 @@ Either a character string or an expression. If \code{datanames} is used to reque only code that \emph{creates} that dataset (not code that uses it) is returned. Otherwise, all contents of \verb{@code}. } \description{ -Retrieve code from \code{teal_data} object. +\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} Retrieve code from \code{teal_data} object. } \details{ Retrieve code stored in \verb{@code}, which (in principle) can be used to recreate all objects found in \verb{@env}. From 8d61ecc6a760b967cf4e3bc2ee81463049d4a0f0 Mon Sep 17 00:00:00 2001 From: m7pr Date: Wed, 16 Oct 2024 14:47:13 +0200 Subject: [PATCH 02/24] do not add a call to the warning message --- R/teal_data-get_code.R | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index ebeda7f6a..71b9701e5 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -108,7 +108,7 @@ setMethod("get_code", signature = "teal_data", definition = function(object, dep checkmate::assert_flag(deparse) code <- if (!is.null(datanames)) { - warning("get_code(datanames) was deprecated in teal.data 0.6.1, use get_code(names) instead.") + warning("get_code(datanames) was deprecated in teal.data 0.6.1, use get_code(names) instead.", call. = FALSE) get_code_dependency(object@code, datanames, ...) } else { object@code From 7a695826d5c5cb9f5717634b622a48b8e3fd88fc Mon Sep 17 00:00:00 2001 From: m7pr Date: Wed, 16 Oct 2024 15:25:30 +0200 Subject: [PATCH 03/24] update vignettes --- vignettes/teal-data-reproducibility.Rmd | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/vignettes/teal-data-reproducibility.Rmd b/vignettes/teal-data-reproducibility.Rmd index 10f08d583..f39d0a81b 100644 --- a/vignettes/teal-data-reproducibility.Rmd +++ b/vignettes/teal-data-reproducibility.Rmd @@ -87,7 +87,7 @@ verify(data_wrong) # fails verification, raises error ## Retrieving code The `get_code` function is used to retrieve the code stored in a `teal_data` object. -A simple `get_code()` will return the entirety of the code but using the `datanames` argument allows for obtaining a subset of the code that only deals with some of the objects stored in `teal_data`. +A simple `get_code()` will return the entirety of the code but using the `names` argument allows for obtaining a subset of the code that only deals with some of the objects stored in `teal_data`. ```{r} library(teal.data) @@ -98,7 +98,7 @@ data <- within(teal_data(), { head(i) }) cat(get_code(data)) # retrieve all code -cat(get_code(data, datanames = "i")) # retrieve code for `i` +cat(get_code(data, names = "i")) # retrieve code for `i` ``` Note that in when retrieving code for a specific dataset, the result is only the code used to _create_ that dataset, not code that _uses_ is. From 3942fecb9027fc6689cae15e6e0d02aa113951d8 Mon Sep 17 00:00:00 2001 From: m7pr Date: Thu, 17 Oct 2024 11:23:05 +0200 Subject: [PATCH 04/24] use lifecyclfe::deprecate_warn instead of warning --- R/teal_data-get_code.R | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index 71b9701e5..9e8d88dec 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -108,7 +108,12 @@ setMethod("get_code", signature = "teal_data", definition = function(object, dep checkmate::assert_flag(deparse) code <- if (!is.null(datanames)) { - warning("get_code(datanames) was deprecated in teal.data 0.6.1, use get_code(names) instead.", call. = FALSE) + lifecycle::deprecate_warn( + when = "0.6.1", + what = "teal.data::get_code(datanames)", + with = "teal.code::get_code(names)", + always = TRUE + ) get_code_dependency(object@code, datanames, ...) } else { object@code From 2e756e320b0fafa74870757caff13119da28fbe2 Mon Sep 17 00:00:00 2001 From: m7pr Date: Thu, 17 Oct 2024 13:54:15 +0200 Subject: [PATCH 05/24] update documentation --- R/teal_data-get_code.R | 18 +++++++++--------- man/get_code.Rd | 18 +++++++++--------- 2 files changed, 18 insertions(+), 18 deletions(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index 9e8d88dec..22dc2c74c 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -3,10 +3,10 @@ #' `r lifecycle::badge("deprecated")` Retrieve code from `teal_data` object. #' #' Retrieve code stored in `@code`, which (in principle) can be used to recreate all objects found in `@env`. -#' Use `datanames` to limit the code to one or more of the datasets enumerated in `@datanames`. +#' Use `names` to limit the code to one or more of the datasets enumerated in `@datanames`. #' #' @section Extracting dataset-specific code: -#' When `datanames` is specified, the code returned will be limited to the lines needed to _create_ +#' When `names` is specified, the code returned will be limited to the lines needed to _create_ #' the requested datasets. The code stored in the `@code` slot is analyzed statically to determine #' which lines the datasets of interest depend upon. The analysis works well when objects are created #' with standard infix assignment operators (see `?assignOps`) but it can fail in some situations. @@ -23,10 +23,10 @@ #' x <- 0 #' y <- foo(x) #' }) -#' get_code(data, datanames = "y") +#' teal.code::get_code(data, names = "y") #' ``` -#' `x` has no dependencies, so `get_code(data, datanames = "x")` will return only the second call.\cr -#' `y` depends on `x` and `foo`, so `get_code(data, datanames = "y")` will contain all three calls. +#' `x` has no dependencies, so `teal.code::get_code(data, names = "x")` will return only the second call.\cr +#' `y` depends on `x` and `foo`, so `teal.code::get_code(data, names = "y")` will contain all three calls. #' #' _Case 2: Some objects are created by a function's side effects._ #' ```r @@ -39,10 +39,10 @@ #' foo() #' y <- x #' }) -#' get_code(data, datanames = "y") +#' teal.code::get_code(data, names = "y") #' ``` #' Here, `y` depends on `x` but `x` is modified by `foo` as a side effect (not by reassignment) -#' and so `get_code(data, datanames = "y")` will not return the `foo()` call.\cr +#' and so `teal.code::get_code(data, names = "y")` will not return the `foo()` call.\cr #' To overcome this limitation, code dependencies can be specified manually. #' Lines where side effects occur can be flagged by adding "`# @linksto `" at the end.\cr #' Note that `within` evaluates code passed to `expr` as is and comments are ignored. @@ -92,8 +92,8 @@ #' c <- list(x = 2) #' }) #' get_code(tdata1) -#' get_code(tdata1, datanames = "a") -#' get_code(tdata1, datanames = "b") +#' teal.code::get_code(tdata1, names = "a") +#' teal.code::get_code(tdata1, names = "b") #' #' tdata2 <- teal_data(x1 = iris, code = "x1 <- iris") #' get_code(tdata2) diff --git a/man/get_code.Rd b/man/get_code.Rd index 09f31a755..b3f0398c5 100644 --- a/man/get_code.Rd +++ b/man/get_code.Rd @@ -28,11 +28,11 @@ only code that \emph{creates} that dataset (not code that uses it) is returned. } \details{ Retrieve code stored in \verb{@code}, which (in principle) can be used to recreate all objects found in \verb{@env}. -Use \code{datanames} to limit the code to one or more of the datasets enumerated in \verb{@datanames}. +Use \code{names} to limit the code to one or more of the datasets enumerated in \verb{@datanames}. } \section{Extracting dataset-specific code}{ -When \code{datanames} is specified, the code returned will be limited to the lines needed to \emph{create} +When \code{names} is specified, the code returned will be limited to the lines needed to \emph{create} the requested datasets. The code stored in the \verb{@code} slot is analyzed statically to determine which lines the datasets of interest depend upon. The analysis works well when objects are created with standard infix assignment operators (see \code{?assignOps}) but it can fail in some situations. @@ -49,11 +49,11 @@ Consider the following examples: x <- 0 y <- foo(x) \}) -get_code(data, datanames = "y") +teal.code::get_code(data, names = "y") }\if{html}{\out{}} -\code{x} has no dependencies, so \code{get_code(data, datanames = "x")} will return only the second call.\cr -\code{y} depends on \code{x} and \code{foo}, so \code{get_code(data, datanames = "y")} will contain all three calls. +\code{x} has no dependencies, so \code{teal.code::get_code(data, names = "x")} will return only the second call.\cr +\code{y} depends on \code{x} and \code{foo}, so \code{teal.code::get_code(data, names = "y")} will contain all three calls. \emph{Case 2: Some objects are created by a function's side effects.} @@ -66,11 +66,11 @@ get_code(data, datanames = "y") foo() y <- x \}) -get_code(data, datanames = "y") +teal.code::get_code(data, names = "y") }\if{html}{\out{}} Here, \code{y} depends on \code{x} but \code{x} is modified by \code{foo} as a side effect (not by reassignment) -and so \code{get_code(data, datanames = "y")} will not return the \code{foo()} call.\cr +and so \code{teal.code::get_code(data, names = "y")} will not return the \code{foo()} call.\cr To overcome this limitation, code dependencies can be specified manually. Lines where side effects occur can be flagged by adding "\verb{# @linksto }" at the end.\cr Note that \code{within} evaluates code passed to \code{expr} as is and comments are ignored. @@ -109,8 +109,8 @@ tdata1 <- within(tdata1, { c <- list(x = 2) }) get_code(tdata1) -get_code(tdata1, datanames = "a") -get_code(tdata1, datanames = "b") +teal.code::get_code(tdata1, names = "a") +teal.code::get_code(tdata1, names = "b") tdata2 <- teal_data(x1 = iris, code = "x1 <- iris") get_code(tdata2) From 50d02d9be4479bc9df3efb46b21168d4dc515635 Mon Sep 17 00:00:00 2001 From: m7pr Date: Thu, 17 Oct 2024 15:26:28 +0200 Subject: [PATCH 06/24] remove teal.data::get_code bones (remove get_code_dependency0 --- R/utils-get_code_dependency.R | 436 --------------------- man/get_code_dependency.Rd | 37 -- tests/testthat/test-get_code.R | 697 --------------------------------- 3 files changed, 1170 deletions(-) delete mode 100644 R/utils-get_code_dependency.R delete mode 100644 man/get_code_dependency.Rd delete mode 100644 tests/testthat/test-get_code.R diff --git a/R/utils-get_code_dependency.R b/R/utils-get_code_dependency.R deleted file mode 100644 index 48fe6b122..000000000 --- a/R/utils-get_code_dependency.R +++ /dev/null @@ -1,436 +0,0 @@ -# get_code_dependency ---- - -#' Get code dependency of an object -#' -#' Extract subset of code required to reproduce specific object(s), including code producing side-effects. -#' -#' Given a character vector with code, this function will extract the part of the code responsible for creating -#' the variables specified by `names`. -#' This includes the final call that creates the variable(s) in question as well as all _parent calls_, -#' _i.e._ calls that create variables used in the final call and their parents, etc. -#' Also included are calls that create side-effects like establishing connections. -#' -#' It is assumed that object dependency is established by using three assignment operators: `<-`, `=`, and `->` . -#' Other assignment methods (`assign`, `<<-`) or non-standard-evaluation methods are not supported. -#' -#' Side-effects are not detected automatically and must be marked in the code. -#' Add `# @linksto object` at the end of a line where a side-effect occurs to specify that this line is required -#' to reproduce a variable called `object`. -#' -#' @param code `character` with the code. -#' @param names `character` vector of object names. -#' @param check_names `logical(1)` flag specifying if a warning for non-existing names should be displayed. -#' -#' @return Character vector, a subset of `code`. -#' Note that subsetting is actually done on the calls `code`, not necessarily on the elements of the vector. -#' -#' @keywords internal -get_code_dependency <- function(code, names, check_names = TRUE) { - checkmate::assert_character(code) - checkmate::assert_character(names, any.missing = FALSE) - - if (identical(code, character(0)) || identical(trimws(code), "")) { - return(code) - } - - # If code is bound in curly brackets, remove them. - tcode <- trimws(code) - if (any(grepl("^\\{.*\\}$", tcode))) { - code <- sub("^\\{(.*)\\}$", "\\1", tcode) - } - - - code <- parse(text = code, keep.source = TRUE) - pd <- utils::getParseData(code) - calls_pd <- extract_calls(pd) - - if (check_names) { - # Detect if names are actually in code. - symbols <- unlist(lapply(calls_pd, function(call) call[call$token == "SYMBOL", "text"])) - if (any(pd$text == "assign")) { - assign_calls <- Filter(function(call) find_call(call, "assign"), calls_pd) - ass_str <- unlist(lapply(assign_calls, function(call) call[call$token == "STR_CONST", "text"])) - ass_str <- gsub("^['\"]|['\"]$", "", ass_str) - symbols <- c(ass_str, symbols) - } - if (!all(names %in% unique(symbols))) { - warning("Object(s) not found in code: ", toString(setdiff(names, symbols))) - } - } - - graph <- code_graph(calls_pd) - ind <- unlist(lapply(names, function(x) graph_parser(x, graph))) - - lib_ind <- detect_libraries(calls_pd) - - as.character(code[sort(unique(c(lib_ind, ind)))]) -} - -#' Locate function call token -#' -#' Determine which row of parsed data is specific `SYMBOL_FUNCTION_CALL` token. -#' -#' Useful for determining occurrence of `assign` or `data` functions in an input call. -#' -#' @param call_pd `data.frame` as returned by `extract_calls()` -#' @param text `character(1)` to look for in `text` column of `call_pd` -#' -#' @return -#' Single integer specifying row in `call_pd` where `token` is `SYMBOL_FUNCTION_CALL` and `text` is `text`. -#' 0 if not found. -#' -#' @keywords internal -#' @noRd -find_call <- function(call_pd, text) { - checkmate::check_data_frame(call_pd) - checkmate::check_names(call_pd, must.include = c("token", "text")) - checkmate::check_string(text) - - ans <- which(call_pd$token == "SYMBOL_FUNCTION_CALL" & call_pd$text == text) - if (length(ans)) { - ans - } else { - 0L - } -} - -#' Split the result of `utils::getParseData()` into separate calls -#' -#' @param pd (`data.frame`) A result of `utils::getParseData()`. -#' -#' @return -#' A `list` of `data.frame`s. -#' Each element is a subset of `pd` corresponding to one call in the original code from which `pd` was obtained. -#' Only four columns (`"token"`, `"text"`, `"id"`, `"parent"`) are kept, the rest is discarded. -#' -#' @keywords internal -#' @noRd -extract_calls <- function(pd) { - calls <- lapply( - pd[pd$parent == 0, "id"], - function(parent) { - rbind( - pd[pd$id == parent, c("token", "text", "id", "parent")], - get_children(pd = pd, parent = parent) - ) - } - ) - calls <- Filter(function(call) !(nrow(call) == 1 && call$token == "';'"), calls) - calls <- Filter(Negate(is.null), calls) - calls <- fix_shifted_comments(calls) - fix_arrows(calls) -} - -#' @keywords internal -#' @noRd -get_children <- function(pd, parent) { - idx_children <- abs(pd$parent) == parent - children <- pd[idx_children, c("token", "text", "id", "parent")] - if (nrow(children) == 0) { - return(NULL) - } - - if (parent > 0) { - do.call(rbind, c(list(children), lapply(children$id, get_children, pd = pd))) - } -} - -#' Fixes edge case of comments being shifted to the next call. -#' @keywords internal -#' @noRd -fix_shifted_comments <- function(calls) { - # If the first or the second token is a @linksto COMMENT, - # then it belongs to the previous call. - if (length(calls) >= 2) { - for (i in 2:length(calls)) { - comment_idx <- grep("@linksto", calls[[i]][, "text"]) - if (isTRUE(comment_idx[1] <= 2)) { - calls[[i - 1]] <- rbind( - calls[[i - 1]], - calls[[i]][seq_len(comment_idx[1]), ] - ) - calls[[i]] <- calls[[i]][-seq_len(comment_idx[1]), ] - } - } - } - Filter(nrow, calls) -} - -#' Fixes edge case of `<-` assignment operator being called as function, -#' which is \code{`<-`(y,x)} instead of traditional `y <- x`. -#' @keywords internal -#' @noRd -fix_arrows <- function(calls) { - checkmate::assert_list(calls) - lapply(calls, function(call) { - sym_fun <- call$token == "SYMBOL_FUNCTION_CALL" - call[sym_fun, ] <- sub_arrows(call[sym_fun, ]) - call - }) -} - -#' Execution of assignment operator substitutions for a call. -#' @keywords internal -#' @noRd -sub_arrows <- function(call) { - checkmate::assert_data_frame(call) - map <- data.frame( - row.names = c("`<-`", "`<<-`", "`=`"), - token = rep("LEFT_ASSIGN", 3), - text = rep("<-", 3) - ) - sub_ids <- call$text %in% rownames(map) - call[sub_ids, c("token", "text")] <- map[call$text[sub_ids], ] - call -} - -# code_graph ---- - -#' Create object dependencies graph within parsed code -#' -#' Builds dependency graph that identifies dependencies between objects in parsed code. -#' Helps understand which objects depend on which. -#' -#' @param calls_pd `list` of `data.frame`s; -#' result of `utils::getParseData()` split into subsets representing individual calls; -#' created by `extract_calls()` function -#' -#' @return -#' A list (of length of input `calls_pd`) where each element represents one call. -#' Each element is a character vector listing names of objects that depend on this call -#' and names of objects that this call depends on. -#' Dependencies are listed after the `"<-"` string, e.g. `c("a", "<-", "b", "c")` means that in this call object `a` -#' depends on objects `b` and `c`. -#' If a call is tagged with `@linksto a`, then object `a` is understood to depend on that call. -#' -#' @keywords internal -#' @noRd -code_graph <- function(calls_pd) { - cooccurrence <- extract_occurrence(calls_pd) - - side_effects <- extract_side_effects(calls_pd) - - mapply(c, side_effects, cooccurrence, SIMPLIFY = FALSE) -} - -#' Extract object occurrence -#' -#' Extracts objects occurrence within calls passed by `calls_pd`. -#' Also detects which objects depend on which within a call. -#' -#' @param calls_pd `list` of `data.frame`s; -#' result of `utils::getParseData()` split into subsets representing individual calls; -#' created by `extract_calls()` function -#' -#' @return -#' A list (of length of input `calls_pd`) where each element represents one call. -#' Each element is a character vector listing names of objects that depend on this call -#' and names of objects that this call depends on. -#' Dependencies are listed after the `"<-"` string, e.g. `c("a", "<-", "b", "c")` means that in this call object `a` -#' depends on objects `b` and `c`. -#' If a call is tagged with `@linksto a`, then object `a` is understood to depend on that call. -#' -#' @keywords internal -#' @noRd -extract_occurrence <- function(calls_pd) { - is_in_function <- function(x) { - # If an object is a function parameter, - # then in calls_pd there is a `SYMBOL_FORMALS` entry for that object. - function_id <- x[x$token == "FUNCTION", "parent"] - if (length(function_id)) { - x$id %in% get_children(x, function_id[1])$id - } else { - rep(FALSE, nrow(x)) - } - } - in_parenthesis <- function(x) { - if (any(x$token %in% c("LBB", "'['"))) { - id_start <- min(x$id[x$token %in% c("LBB", "'['")]) - id_end <- min(x$id[x$token == "']'"]) - x$text[x$token == "SYMBOL" & x$id > id_start & x$id < id_end] - } - } - lapply( - calls_pd, - function(call_pd) { - # Handle data(object)/data("object")/data(object, envir = ) independently. - data_call <- find_call(call_pd, "data") - if (data_call) { - sym <- call_pd[data_call + 1, "text"] - return(c(gsub("^['\"]|['\"]$", "", sym), "<-")) - } - # Handle assign(x = ). - assign_call <- find_call(call_pd, "assign") - if (assign_call) { - # Check if parameters were named. - # "','" is for unnamed parameters, where "SYMBOL_SUB" is for named. - # "EQ_SUB" is for `=` appearing after the name of the named parameter. - if (any(call_pd$token == "SYMBOL_SUB")) { - params <- call_pd[call_pd$token %in% c("SYMBOL_SUB", "','", "EQ_SUB"), "text"] - # Remove sequence of "=", ",". - if (length(params > 1)) { - remove <- integer(0) - for (i in 2:length(params)) { - if (params[i - 1] == "=" & params[i] == ",") { - remove <- c(remove, i - 1, i) - } - } - if (length(remove)) params <- params[-remove] - } - pos <- match("x", setdiff(params, ","), nomatch = match(",", params, nomatch = 0)) - if (!pos) { - return(character(0L)) - } - # pos is indicator of the place of 'x' - # 1. All parameters are named, but none is 'x' - return(character(0L)) - # 2. Some parameters are named, 'x' is in named parameters: match("x", setdiff(params, ",")) - # - check "x" in params being just a vector of named parameters. - # 3. Some parameters are named, 'x' is not in named parameters - # - check first appearance of "," (unnamed parameter) in vector parameters. - } else { - # Object is the first entry after 'assign'. - pos <- 1 - } - sym <- call_pd[assign_call + pos, "text"] - return(c(gsub("^['\"]|['\"]$", "", sym), "<-")) - } - - # What occurs in a function body is not tracked. - x <- call_pd[!is_in_function(call_pd), ] - sym_cond <- which(x$token %in% c("SYMBOL", "SYMBOL_FUNCTION_CALL")) - - if (length(sym_cond) == 0) { - return(character(0L)) - } - # Watch out for SYMBOLS after $ and @. For x$a x@a: x is object, a is not. - # For x$a, a's ID is $'s ID-2 so we need to remove all IDs that have ID = $ID - 2. - dollar_ids <- x[x$token %in% c("'$'", "'@'"), "id"] - if (length(dollar_ids)) { - object_ids <- x[sym_cond, "id"] - after_dollar <- object_ids[(object_ids - 2) %in% dollar_ids] - sym_cond <- setdiff(sym_cond, which(x$id %in% after_dollar)) - } - - ass_cond <- grep("ASSIGN", x$token) - if (!length(ass_cond)) { - return(c("<-", unique(x[sym_cond, "text"]))) - } - - sym_cond <- sym_cond[sym_cond > ass_cond] # NOTE 1 - # If there was an assignment operation detect direction of it. - if (unique(x$text[ass_cond]) == "->") { # NOTE 2 - sym_cond <- rev(sym_cond) - } - - after <- match(min(x$id[ass_cond]), sort(x$id[c(min(ass_cond), sym_cond)])) - 1 - ans <- append(x[sym_cond, "text"], "<-", after = max(1, after)) - roll <- in_parenthesis(call_pd) - if (length(roll)) { - c(setdiff(ans, roll), roll) - } else { - ans - } - - ### NOTE 2: What if there are 2 assignments: e.g. a <- b -> c. - ### NOTE 1: For cases like 'eval(expression(b <- b + 2))' removes 'eval(expression('. - } - ) -} - -#' Extract side effects -#' -#' Extracts all object names from the code that are marked with `@linksto` tag. -#' -#' The code may contain functions calls that create side effects, e.g. modify the environment. -#' Static code analysis may be insufficient to determine which objects are created or modified by such a function call. -#' The `@linksto` comment tag is introduced to mark a call as having a (side) effect on one or more objects. -#' With this tag a complete object dependency structure can be established. -#' Read more about side effects and the usage of `@linksto` tag in [`get_code_dependencies()`] function. -#' -#' @param calls_pd `list` of `data.frame`s; -#' result of `utils::getParseData()` split into subsets representing individual calls; -#' created by `extract_calls()` function -#' -#' @return -#' A list of length equal to that of `calls_pd`, where each element is a character vector of names of objects -#' depending a call tagged with `@linksto` in a corresponding element of `calls_pd`. -#' -#' @keywords internal -#' @noRd -extract_side_effects <- function(calls_pd) { - lapply( - calls_pd, - function(x) { - linksto <- grep("@linksto", x[x$token == "COMMENT", "text"], value = TRUE) - unlist(strsplit(sub("\\s*#\\s*@linksto\\s+", "", linksto), "\\s+")) - } - ) -} - -# graph_parser ---- - -#' Return the indices of calls needed to reproduce an object -#' -#' @param x The name of the object to return code for. -#' @param graph A result of `code_graph()`. -#' -#' @return -#' Integer vector of indices that can be applied to `graph` to obtain all calls required to reproduce object `x`. -#' -#' @keywords internal -#' @noRd -graph_parser <- function(x, graph) { - occurrence <- vapply( - graph, function(call) { - ind <- match("<-", call, nomatch = length(call) + 1L) - x %in% call[seq_len(ind - 1L)] - }, - logical(1) - ) - - dependencies <- lapply(graph[occurrence], function(call) { - ind <- match("<-", call, nomatch = 0L) - call[(ind + 1L):length(call)] - }) - dependencies <- setdiff(unlist(dependencies), x) - - if (length(dependencies) && any(occurrence)) { - dependency_ids <- lapply(dependencies, function(dependency) { - graph_parser(dependency, graph[1:max(which(occurrence))]) - }) - sort(unique(c(which(occurrence), unlist(dependency_ids)))) - } else { - which(occurrence) - } -} - - -# default_side_effects -------------------------------------------------------------------------------------------- - -#' Detect library calls -#' -#' Detects `library()` and `require()` function calls. -#' -#' @param calls_pd `list` of `data.frame`s; -#' result of `utils::getParseData()` split into subsets representing individual calls; -#' created by `extract_calls()` function -#' -#' @return -#' Integer vector of indices that can be applied to `graph` (result of `code_graph()`) to obtain all calls containing -#' `library()` or `require()` calls that are always returned for reproducibility. -#' -#' @keywords internal -#' @noRd -detect_libraries <- function(calls_pd) { - defaults <- c("library", "require") - - which( - vapply( - calls_pd, - function(call) { - any(call$token == "SYMBOL_FUNCTION_CALL" & call$text %in% defaults) - }, - logical(1) - ) - ) -} diff --git a/man/get_code_dependency.Rd b/man/get_code_dependency.Rd deleted file mode 100644 index 8db40903d..000000000 --- a/man/get_code_dependency.Rd +++ /dev/null @@ -1,37 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/utils-get_code_dependency.R -\name{get_code_dependency} -\alias{get_code_dependency} -\title{Get code dependency of an object} -\usage{ -get_code_dependency(code, names, check_names = TRUE) -} -\arguments{ -\item{code}{\code{character} with the code.} - -\item{names}{\code{character} vector of object names.} - -\item{check_names}{\code{logical(1)} flag specifying if a warning for non-existing names should be displayed.} -} -\value{ -Character vector, a subset of \code{code}. -Note that subsetting is actually done on the calls \code{code}, not necessarily on the elements of the vector. -} -\description{ -Extract subset of code required to reproduce specific object(s), including code producing side-effects. -} -\details{ -Given a character vector with code, this function will extract the part of the code responsible for creating -the variables specified by \code{names}. -This includes the final call that creates the variable(s) in question as well as all \emph{parent calls}, -\emph{i.e.} calls that create variables used in the final call and their parents, etc. -Also included are calls that create side-effects like establishing connections. - -It is assumed that object dependency is established by using three assignment operators: \verb{<-}, \code{=}, and \verb{->} . -Other assignment methods (\code{assign}, \verb{<<-}) or non-standard-evaluation methods are not supported. - -Side-effects are not detected automatically and must be marked in the code. -Add \verb{# @linksto object} at the end of a line where a side-effect occurs to specify that this line is required -to reproduce a variable called \code{object}. -} -\keyword{internal} diff --git a/tests/testthat/test-get_code.R b/tests/testthat/test-get_code.R deleted file mode 100644 index 40095f1bc..000000000 --- a/tests/testthat/test-get_code.R +++ /dev/null @@ -1,697 +0,0 @@ -testthat::test_that("handles empty @code slot", { - testthat::expect_identical( - get_code(teal_data(a = 1, code = character(0)), datanames = "a"), - character(0) - ) - testthat::expect_identical( - get_code(teal_data(a = 1, code = ""), datanames = "a"), - "" - ) -}) - -testthat::test_that("handles the code without symbols on rhs", { - code <- c( - "1 + 1", - "a <- 5", - "501" - ) - - testthat::expect_identical( - get_code(teal_data(a = 5, code = code), datanames = "a"), - "a <- 5" - ) -}) - -testthat::test_that("handles the code included in curly brackets", { - code <- "{1 + 1;a <- 5}" - - testthat::expect_identical( - get_code(teal_data(a = 5, code = code), datanames = "a"), - "a <- 5" - ) -}) - -testthat::test_that("handles the code of length > 1 when at least one is enclosed in curly brackets", { - code <- c("{a<-5}", "1+1") - tdata <- eval_code(eval_code(teal_data(), code[1]), code[2]) - - testthat::expect_identical( - get_code(tdata, datanames = "a"), - "a <- 5" - ) -}) - - -testthat::test_that("extracts the code of a binding from character vector containing simple code", { - code <- c( - "a <- 1", - "b <- 2" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "a"), - "a <- 1" - ) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - "b <- 2" - ) -}) - -testthat::test_that("extracts the code without downstream usage", { - code <- c( - "a <- 1", - "head(a)" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "a"), - "a <- 1" - ) -}) - -testthat::test_that("works for datanames of length > 1", { - code <- c( - "a <- 1", - "b <- 2" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = c("a", "b")), - paste(code, collapse = "\n") - ) -}) - -testthat::test_that("warns if binding doesn't exist in code", { - code <- c("a <- 1") - tdata <- eval_code(teal_data(), code) - testthat::expect_warning( - get_code(tdata, datanames = "c"), - "Object\\(s\\) not found in code: c" - ) -}) - -testthat::test_that("does not fall into a loop", { - code <- c( - "a <- 1", - "b <- a", - "c <- b", - "a <- c" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "a"), - paste(code, collapse = "\n") - ) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste(code[1:2], collapse = "\n") - ) - testthat::expect_identical( - get_code(tdata, datanames = "c"), - paste(code[1:3], collapse = "\n") - ) -}) - - -testthat::test_that("extracts code of a parent binding but only those evaluated before coocurence", { - code <- c( - "a <- 1", - "b <- a", - "a <- 2" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste("a <- 1", "b <- a", sep = "\n") - ) -}) - -testthat::test_that("extracts the code of a parent binding if used as an arg in a function call", { - code <- c( - "a <- 1", - "b <- identity(x = a)", - "a <- 2" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste("a <- 1", "b <- identity(x = a)", sep = "\n") - ) -}) - -testthat::test_that("extracts the code when using <<-", { - code <- c( - "a <- 1", - "b <- a", - "b <<- b + 2" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste("a <- 1", "b <- a", "b <<- b + 2", sep = "\n") - ) -}) - -testthat::test_that("detects every assign calls even if not evaluated, if there is only one assignment in this line", { - code <- c( - "a <- 1", - "b <- 2", - "eval(expression({b <- b + 2}))" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste("b <- 2", "eval(expression({\n b <- b + 2\n}))", sep = "\n") - ) -}) - -testthat::test_that("returns result of length 1 for non-empty input", { - tdata1 <- teal_data() - tdata1 <- within(tdata1, { - a <- 1 - b <- a^5 - c <- list(x = 2) - }) - - testthat::expect_length(get_code(tdata1, deparse = FALSE), 1) - testthat::expect_length(get_code(tdata1, deparse = TRUE), 1) -}) - -testthat::test_that("does not break if code is separated by ;", { - code <- c( - "a <- 1;a <- a + 1" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "a"), - gsub(";", "\n", code, fixed = TRUE) - ) -}) - -testthat::test_that("does not break if code uses quote()", { - code <- c( - "expr <- quote(x <- x + 1)", - "x <- 0", - "eval(expr)" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "x"), - code[2] - ) -}) - -testthat::test_that("does not break if object is used in a function on lhs", { - code <- c( - "data(iris)", - "iris2 <- iris", - "names(iris) <- letters[1:5]" - ) - tdata <- eval_code(teal_data(), code = code) - testthat::expect_identical( - get_code(tdata, datanames = "iris"), - paste(code[c(1, 3)], collapse = "\n") - ) -}) - -testthat::test_that( - "does not break if object is used in a function on lhs and influencers are both on lhs and rhs", - { - code <- c( - "x <- 5", - "y <- length(x)", - "names(x)[y] <- y" - ) - tdata <- eval_code(teal_data(), code = code) - testthat::expect_identical( - get_code(tdata, datanames = "x"), - paste(code, collapse = "\n") - ) - } -) - -# assign ---------------------------------------------------------------------------------------------------------- - -testthat::test_that("extracts the code for assign() where \"x\" is a literal string", { - code <- c( - "a <- 1", - "assign('b', 5)", - "assign(value = 7, x = 'c')", - "assign(value = 15, x = \"d\")", - "b <- b + 2", - "c <- b", - "d <- d * 2" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste("assign(\"b\", 5)", "b <- b + 2", sep = "\n") - ) - testthat::expect_identical( - get_code(tdata, datanames = "c"), - paste( - "assign(\"b\", 5)", - "assign(value = 7, x = \"c\")", - "b <- b + 2", - "c <- b", - sep = "\n" - ) - ) - testthat::expect_identical( - get_code(tdata, datanames = "d"), - paste("assign(value = 15, x = \"d\")", "d <- d * 2", sep = "\n") - ) -}) - -testthat::test_that("extracts the code for assign() where \"x\" is variable", { - testthat::skip("We will not resolve this, as this requires code evaluation.") - code <- c( - "x <- \"a\"", - "assign(x, 5)", - "b <- a" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste(code, collapse = "\n") - ) -}) - -testthat::test_that("works for assign() detection no matter how many parametrers were provided in assignq()", { - code <- c( - "x <- 1", - "assign(\"x\", 0, envir = environment())", - "assign(inherits = FALSE, immediate = TRUE, \"z\", 5, envir = environment())", - "y <- x + z", - "y <- x" - ) - - tdata <- eval_code(teal_data(), code) - - testthat::expect_identical( - get_code(tdata, datanames = "y"), - paste(code, collapse = "\n") - ) -}) - -testthat::test_that("detects function usage of the assignment operator", { - code <- c( - "x <- 1", - "`<-`(y,x)" - ) - code2 <- "`<-`(y, `<-`(x, 2))" - - tdata <- eval_code(teal_data(), code) - tdata2 <- eval_code(teal_data(), code2) - - testthat::expect_identical( - get_code(tdata, datanames = "y"), - paste(c(code[1], "y <- x"), collapse = "\n") - ) - testthat::expect_identical( - get_code(tdata2, datanames = "y"), - "y <- x <- 2" - ) -}) - - -# @linksto --------------------------------------------------------------------------------------------------------- - -testthat::test_that("get_code does not break if @linksto is put in the last line", { - # In some cases R parses comment as a separate expression so the comment is not - # directly associated with this line of code. This situation occurs when `eval` is in the last - # line of the code. Other cases are not known but are highly probable. - code <- c( - "expr <- quote(x <- x + 1)", - "x <- 0", - "eval(expr) #@linksto x" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "x"), - paste(gsub(" #@linksto x", "", code, fixed = TRUE), collapse = "\n") - ) -}) - -testthat::test_that("@linksto makes a line being returned for an affected binding", { - code <- " - a <- 1 # @linksto b - b <- 2 - " - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste("a <- 1", "b <- 2", sep = "\n") - ) -}) - -testthat::test_that( - "@linksto returns the line for an affected binding - even if the object did not exist in the same iteration of eval_code", - { - code <- c( - "a <- 1 # @linksto b", - "b <- 2" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste("a <- 1", "b <- 2", sep = "\n") - ) - } -) - -testthat::test_that( - "lines affecting parent evaluated after co-occurrence are not included in output when using @linksto", - { - code <- c( - "a <- 1 ", - "b <- 2 # @linksto a", - "a <- a + 1", - "b <- b + 1" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "a"), - paste("a <- 1", "b <- 2", "a <- a + 1", sep = "\n") - ) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste("b <- 2", "b <- b + 1", sep = "\n") - ) - } -) - -testthat::test_that( - "@linksto gets extracted if it's a side-effect on a dependent object (even of a dependent object)", - { - code <- " - iris[1:5, ] -> iris2 - iris_head <- head(iris) # @linksto iris3 - iris3 <- iris_head[1, ] # @linksto iris2 - classes <- lapply(iris2, class) - " - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "classes"), - paste("iris2 <- iris[1:5, ]", - "iris_head <- head(iris)", - "iris3 <- iris_head[1, ]", - "classes <- lapply(iris2, class)", - sep = "\n" - ) - ) - } -) - -# functions ------------------------------------------------------------------------------------------------------- - -testthat::test_that("ignores occurrence in a function definition", { - code <- c( - "b <- 2", - "foo <- function(b) { b <- b + 2 }" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - "b <- 2" - ) - testthat::expect_identical( - get_code(tdata, datanames = "foo"), - "foo <- function(b) {\n b <- b + 2\n}" - ) -}) - -testthat::test_that("ignores occurrence in a function definition that has function in it", { - code <- c( - "b <- 2", - "foo <- function(b) { function(c) {b <- c + 2 }}" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - "b <- 2" - ) - testthat::expect_identical( - get_code(tdata, datanames = "foo"), - "foo <- function(b) {\n function(c) {\n b <- c + 2\n }\n}" - ) -}) - -testthat::test_that("ignores occurrence in a function definition if there is multiple function definitions", { - code <- c( - "b <- 2", - "foo <- function(b) { function(c) {b <- c + 2 }}", - "b <- b + 1", - "bar <- function(b) print(b)" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - "b <- 2\nb <- b + 1" - ) - testthat::expect_identical( - get_code(tdata, datanames = "foo"), - "foo <- function(b) {\n function(c) {\n b <- c + 2\n }\n}" - ) -}) - -testthat::test_that("ignores occurrence in a function definition in lapply", { - code <- c( - "a <- list(a = 1, b = 2, c = 3)", - "b <- lapply(a, FUN = function(x) { x <- x + 1 })", - "b <- Filter(function(x) x > 2, b)", - "x <- 1", - "identity(x)" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "x"), - "x <- 1" - ) -}) - -testthat::test_that("does not ignore occurrence in function body if object exsits in env", { - skip("This is not urgent and can be ommitted with @linksto tag.") - code <- c( - "a <- list(a = 1, b = 2, c = 3)", - "p <- 5", # This is not extracted, even though is used in the next line. - "b <- lapply(a, FUN = function(x) { x <- x + p })", - "b <- Filter(function(x) x > 2, b)" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste(code, sep = "\n") - ) -}) - -testthat::test_that("ignores occurrence in function definition without { curly brackets", { - code <- c( - "b <- 2", - "foo <- function(b) b <- b + 2 " - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "foo"), - "foo <- function(b) b <- b + 2" - ) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - "b <- 2" - ) -}) - -testthat::test_that("detects occurrence of the function object", { - code <- c( - "a <- 1", - "b <- 2", - "foo <- function(b) { b <- b + 2 }", - "b <- foo(a)" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste("a <- 1", "b <- 2", "foo <- function(b) {\n b <- b + 2\n}", "b <- foo(a)", sep = "\n") - ) -}) - -testthat::test_that("detects occurrence of a function definition when a formal is named the same as a function", { - code <- c( - "x <- 1", - "foo <- function(foo = 1) 'text'", - "a <- foo(x)" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "a"), - paste("x <- 1", "foo <- function(foo = 1) \"text\"", "a <- foo(x)", sep = "\n") - ) -}) - -testthat::test_that("detects occurrence of a function definition with a @linksto usage", { - code <- c( - " - foo <- function() { - env <- parent.frame() - env$x <- 0 - }", - "foo() # @linksto x", - "y <- x" - ) - tdata <- teal_data(code = code) - testthat::expect_identical( - get_code(tdata, datanames = "x"), - "foo <- function() {\n env <- parent.frame()\n env$x <- 0\n}\nfoo()" - ) -}) -# $ --------------------------------------------------------------------------------------------------------------- - -testthat::test_that("understands $ usage and do not treat rhs of $ as objects (only lhs)", { - code <- c( - "x <- data.frame(a = 1:3)", - "a <- data.frame(y = 1:3)", - "a$x <- a$y", - "a$x <- a$x + 2", - "a$x <- x$a" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "x"), - "x <- data.frame(a = 1:3)" - ) - testthat::expect_identical( - get_code(tdata, datanames = "a"), - paste("x <- data.frame(a = 1:3)", - "a <- data.frame(y = 1:3)", - "a$x <- a$y", - "a$x <- a$x + 2", - "a$x <- x$a", - sep = "\n" - ) - ) -}) - -testthat::test_that("detects cooccurrence properly even if all objects are on lhs", { - code <- c( - "a <- 1", - "b <- list(c = 2)", - "b[[a]] <- 3" - ) - tdata <- eval_code(teal_data(), code) - testthat::expect_identical( - get_code(tdata, datanames = "b"), - paste(code, collapse = "\n") - ) -}) - - -# @ --------------------------------------------------------------------------------------------------------------- - -testthat::test_that("understands @ usage and do not treat rhs of @ as objects (only lhs)", { - code <- c( - "setClass('aclass', slots = c(a = 'numeric', x = 'numeric', y = 'numeric')) # @linksto a x", - "x <- new('aclass', a = 1:3, x = 1:3, y = 1:3)", - "a <- new('aclass', a = 1:3, x = 1:3, y = 1:3)", - "a@x <- a@y", - "a@x <- a@x + 2", - "a@x <- x@a" - ) - tdata <- teal_data(x = 1, a = 1, code = code) - testthat::expect_identical( - get_code(tdata, datanames = "x"), - paste( - 'setClass("aclass", slots = c(a = "numeric", x = "numeric", y = "numeric"))', - 'x <- new("aclass", a = 1:3, x = 1:3, y = 1:3)', - sep = "\n" - ) - ) - testthat::expect_identical( - get_code(tdata, datanames = "a"), - paste( - 'setClass("aclass", slots = c(a = "numeric", x = "numeric", y = "numeric"))', - 'x <- new("aclass", a = 1:3, x = 1:3, y = 1:3)', - 'a <- new("aclass", a = 1:3, x = 1:3, y = 1:3)', - "a@x <- a@y", - "a@x <- a@x + 2", - "a@x <- x@a", - sep = "\n" - ) - ) -}) - - - -# libraries ------------------------------------------------------------------------------------------------------- - -testthat::test_that("library() and require() are always returned", { - code <- c( - "set.seed(1)", - "library(random.cdisc.data)", - "require(dplyr)", - "library(MultiAssayExperiment)", - "x <- 5", - "y <- 6" - ) - tdata <- teal_data(x = 5, y = 6, code = code) - testthat::expect_identical( - get_code(tdata, datanames = "x"), - paste( - "library(random.cdisc.data)", - "require(dplyr)", - "library(MultiAssayExperiment)", - "x <- 5", - sep = "\n" - ) - ) -}) - - -# data() ---------------------------------------------------------------------------------------------------------- - -testthat::test_that("data() call is returned when data name is provided as is", { - code <- c( - "set.seed(1)", - "library(random.cdisc.data)", - "require(dplyr)", - "library(MultiAssayExperiment)", - "data(miniACC, envir = environment())", - "x <- miniACC" - ) - tdata <- teal_data(x = 1, code = code) - testthat::expect_identical( - get_code(tdata, datanames = "x"), - paste( - "library(random.cdisc.data)", - "require(dplyr)", - "library(MultiAssayExperiment)", - "data(miniACC, envir = environment())", - "x <- miniACC", - sep = "\n" - ) - ) -}) - -testthat::test_that("data() call is returned when data name is provided as a character", { - code <- c( - "set.seed(1)", - "library(random.cdisc.data)", - "require(dplyr)", - "library(MultiAssayExperiment)", - "data('mtcars')", - "z <- mtcars" - ) - tdata <- teal_data(z = 1, code = code) - testthat::expect_identical( - get_code(tdata, datanames = "z"), - paste( - "library(random.cdisc.data)", - "require(dplyr)", - "library(MultiAssayExperiment)", - "data(\"mtcars\")", - "z <- mtcars", - sep = "\n" - ) - ) -}) From 5ca6190cc3ec1a5e403f654fc4b718db449d233d Mon Sep 17 00:00:00 2001 From: m7pr Date: Thu, 17 Oct 2024 15:27:00 +0200 Subject: [PATCH 07/24] update documentation to use `names` and no prefixes for teal.code --- R/teal_data-get_code.R | 45 +++++++++++++++++------------------------- man/get_code.Rd | 25 ++++++++++++----------- 2 files changed, 32 insertions(+), 38 deletions(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index 22dc2c74c..0d55dbb22 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -1,6 +1,6 @@ #' Get code from `teal_data` object #' -#' `r lifecycle::badge("deprecated")` Retrieve code from `teal_data` object. +#' `r lifecycle::badge("deprecated")` Use [`teal.code::get_code()`]. Retrieve code from `teal_data` object. #' #' Retrieve code stored in `@code`, which (in principle) can be used to recreate all objects found in `@env`. #' Use `names` to limit the code to one or more of the datasets enumerated in `@datanames`. @@ -23,10 +23,10 @@ #' x <- 0 #' y <- foo(x) #' }) -#' teal.code::get_code(data, names = "y") +#' get_code(data, names = "y") #' ``` -#' `x` has no dependencies, so `teal.code::get_code(data, names = "x")` will return only the second call.\cr -#' `y` depends on `x` and `foo`, so `teal.code::get_code(data, names = "y")` will contain all three calls. +#' `x` has no dependencies, so `get_code(data, names = "x")` will return only the second call.\cr +#' `y` depends on `x` and `foo`, so `get_code(data, names = "y")` will contain all three calls. #' #' _Case 2: Some objects are created by a function's side effects._ #' ```r @@ -39,10 +39,10 @@ #' foo() #' y <- x #' }) -#' teal.code::get_code(data, names = "y") +#' get_code(data, names = "y") #' ``` #' Here, `y` depends on `x` but `x` is modified by `foo` as a side effect (not by reassignment) -#' and so `teal.code::get_code(data, names = "y")` will not return the `foo()` call.\cr +#' and so `get_code(data, names = "y")` will not return the `foo()` call.\cr #' To overcome this limitation, code dependencies can be specified manually. #' Lines where side effects occur can be flagged by adding "`# @linksto `" at the end.\cr #' Note that `within` evaluates code passed to `expr` as is and comments are ignored. @@ -58,7 +58,7 @@ #' foo() # @linksto x #' y <- x #' ") -#' get_code(data, datanames = "y") +#' get_code(data, names = "y") #' ``` #' Now the `foo()` call will be properly included in the code required to recreate `y`. #' @@ -74,6 +74,8 @@ #' @param object (`teal_data`) #' @param datanames `r lifecycle::badge("experimental")` (`character`) vector of dataset names to return the code for. #' For more details see the "Extracting dataset-specific code" section. +#' @param names (`character`) Successor of `datanames`. Vector of dataset names to return the code for. +#' For more details see the "Extracting dataset-specific code" section. #' @param deparse (`logical`) flag specifying whether to return code as `character` (`deparse = TRUE`) or as #' `expression` (`deparse = FALSE`). #' @param ... Parameters passed to internal methods. Currently, the only supported parameter is `check_names` @@ -81,7 +83,7 @@ #' `code` but are passed in `datanames`. To remove the warning, set `check_names = FALSE`. #' #' @return -#' Either a character string or an expression. If `datanames` is used to request a specific dataset, +#' Either a character string or an expression. If `names` is used to request a specific dataset, #' only code that _creates_ that dataset (not code that uses it) is returned. Otherwise, all contents of `@code`. #' #' @examples @@ -92,8 +94,8 @@ #' c <- list(x = 2) #' }) #' get_code(tdata1) -#' teal.code::get_code(tdata1, names = "a") -#' teal.code::get_code(tdata1, names = "b") +#' get_code(tdata1, names = "a") +#' get_code(tdata1, names = "b") #' #' tdata2 <- teal_data(x1 = iris, code = "x1 <- iris") #' get_code(tdata2) @@ -103,29 +105,18 @@ #' @aliases get_code,teal_data-method #' #' @export -setMethod("get_code", signature = "teal_data", definition = function(object, deparse = TRUE, datanames = NULL, ...) { +setMethod("get_code", signature = "teal_data", definition = function(object, deparse = TRUE, datanames = NULL, names = datanames, ...) { checkmate::assert_character(datanames, min.len = 1L, null.ok = TRUE) + checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) checkmate::assert_flag(deparse) - code <- if (!is.null(datanames)) { + if (!is.null(datanames)) { lifecycle::deprecate_warn( when = "0.6.1", - what = "teal.data::get_code(datanames)", - with = "teal.code::get_code(names)", + what = "get_code()", + with = "teal.code::get_code()", always = TRUE ) - get_code_dependency(object@code, datanames, ...) - } else { - object@code - } - - if (deparse) { - if (length(code) == 0) { - code - } else { - paste(code, collapse = "\n") - } - } else { - parse(text = paste(c("{", code, "}"), collapse = "\n"), keep.source = TRUE) } + teal.code::get_code(object = object, deparse = deparse, names = names, ...) }) diff --git a/man/get_code.Rd b/man/get_code.Rd index b3f0398c5..9bfdf14f6 100644 --- a/man/get_code.Rd +++ b/man/get_code.Rd @@ -4,7 +4,7 @@ \alias{get_code,teal_data-method} \title{Get code from \code{teal_data} object} \usage{ -\S4method{get_code}{teal_data}(object, deparse = TRUE, datanames = NULL, ...) +\S4method{get_code}{teal_data}(object, deparse = TRUE, datanames = NULL, names = datanames, ...) } \arguments{ \item{object}{(\code{teal_data})} @@ -15,16 +15,19 @@ \item{datanames}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#experimental}{\figure{lifecycle-experimental.svg}{options: alt='[Experimental]'}}}{\strong{[Experimental]}} (\code{character}) vector of dataset names to return the code for. For more details see the "Extracting dataset-specific code" section.} +\item{names}{(\code{character}) Successor of \code{datanames}. Vector of dataset names to return the code for. +For more details see the "Extracting dataset-specific code" section.} + \item{...}{Parameters passed to internal methods. Currently, the only supported parameter is \code{check_names} (\code{logical(1)}) flag, which is \code{TRUE} by default. Function warns about missing objects, if they do not exist in \code{code} but are passed in \code{datanames}. To remove the warning, set \code{check_names = FALSE}.} } \value{ -Either a character string or an expression. If \code{datanames} is used to request a specific dataset, +Either a character string or an expression. If \code{names} is used to request a specific dataset, only code that \emph{creates} that dataset (not code that uses it) is returned. Otherwise, all contents of \verb{@code}. } \description{ -\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} Retrieve code from \code{teal_data} object. +\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} Use \code{\link[teal.code:qenv]{teal.code::get_code()}}. Retrieve code from \code{teal_data} object. } \details{ Retrieve code stored in \verb{@code}, which (in principle) can be used to recreate all objects found in \verb{@env}. @@ -49,11 +52,11 @@ Consider the following examples: x <- 0 y <- foo(x) \}) -teal.code::get_code(data, names = "y") +get_code(data, names = "y") }\if{html}{\out{}} -\code{x} has no dependencies, so \code{teal.code::get_code(data, names = "x")} will return only the second call.\cr -\code{y} depends on \code{x} and \code{foo}, so \code{teal.code::get_code(data, names = "y")} will contain all three calls. +\code{x} has no dependencies, so \code{get_code(data, names = "x")} will return only the second call.\cr +\code{y} depends on \code{x} and \code{foo}, so \code{get_code(data, names = "y")} will contain all three calls. \emph{Case 2: Some objects are created by a function's side effects.} @@ -66,11 +69,11 @@ teal.code::get_code(data, names = "y") foo() y <- x \}) -teal.code::get_code(data, names = "y") +get_code(data, names = "y") }\if{html}{\out{}} Here, \code{y} depends on \code{x} but \code{x} is modified by \code{foo} as a side effect (not by reassignment) -and so \code{teal.code::get_code(data, names = "y")} will not return the \code{foo()} call.\cr +and so \code{get_code(data, names = "y")} will not return the \code{foo()} call.\cr To overcome this limitation, code dependencies can be specified manually. Lines where side effects occur can be flagged by adding "\verb{# @linksto }" at the end.\cr Note that \code{within} evaluates code passed to \code{expr} as is and comments are ignored. @@ -85,7 +88,7 @@ In order to include comments in code one must use the \code{eval_code} function foo() # @linksto x y <- x ") -get_code(data, datanames = "y") +get_code(data, names = "y") }\if{html}{\out{}} Now the \code{foo()} call will be properly included in the code required to recreate \code{y}. @@ -109,8 +112,8 @@ tdata1 <- within(tdata1, { c <- list(x = 2) }) get_code(tdata1) -teal.code::get_code(tdata1, names = "a") -teal.code::get_code(tdata1, names = "b") +get_code(tdata1, names = "a") +get_code(tdata1, names = "b") tdata2 <- teal_data(x1 = iris, code = "x1 <- iris") get_code(tdata2) From 0c05b8769f44ad3489eb053a1334f96dbf167dd5 Mon Sep 17 00:00:00 2001 From: "27856297+dependabot-preview[bot]@users.noreply.github.com" <27856297+dependabot-preview[bot]@users.noreply.github.com> Date: Thu, 17 Oct 2024 13:29:12 +0000 Subject: [PATCH 08/24] [skip roxygen] [skip vbump] Roxygen Man Pages Auto Update --- DESCRIPTION | 1 - 1 file changed, 1 deletion(-) diff --git a/DESCRIPTION b/DESCRIPTION index d0c550b2e..4c4116a6d 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -78,6 +78,5 @@ Collate: 'teal_data.R' 'testhat-helpers.R' 'topological_sort.R' - 'utils-get_code_dependency.R' 'verify.R' 'zzz.R' From c3942cd8ba4a4218ae538489de24361f4245a1f0 Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 09:09:14 +0200 Subject: [PATCH 09/24] spelling --- NEWS.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/NEWS.md b/NEWS.md index b7fb3b300..e0ed19baf 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,6 +1,6 @@ # teal.data 0.6.0.9012 -### Deprecations +### Deprecation - `get_code(datanames)` `S4` method parameter for `teal_data()` object was soft deprecated. Use `get_code(names)`. From 6753dc8405aa8ca2b07e6ef98459f9f917aebb86 Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 09:10:17 +0200 Subject: [PATCH 10/24] lintr --- R/teal_data-get_code.R | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index 0d55dbb22..220f92cf2 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -105,7 +105,8 @@ #' @aliases get_code,teal_data-method #' #' @export -setMethod("get_code", signature = "teal_data", definition = function(object, deparse = TRUE, datanames = NULL, names = datanames, ...) { +setMethod("get_code", signature = "teal_data", + definition = function(object, deparse = TRUE, datanames = NULL, names = datanames, ...) { checkmate::assert_character(datanames, min.len = 1L, null.ok = TRUE) checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) checkmate::assert_flag(deparse) From 91f5c7ce9cb78c8d9633eefb99b2f4385924b1aa Mon Sep 17 00:00:00 2001 From: github-actions <41898282+github-actions[bot]@users.noreply.github.com> Date: Fri, 18 Oct 2024 07:12:57 +0000 Subject: [PATCH 11/24] [skip style] [skip vbump] Restyle files --- R/teal_data-get_code.R | 30 ++++++++++++++++-------------- 1 file changed, 16 insertions(+), 14 deletions(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index 220f92cf2..af7bb1af7 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -105,19 +105,21 @@ #' @aliases get_code,teal_data-method #' #' @export -setMethod("get_code", signature = "teal_data", - definition = function(object, deparse = TRUE, datanames = NULL, names = datanames, ...) { - checkmate::assert_character(datanames, min.len = 1L, null.ok = TRUE) - checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) - checkmate::assert_flag(deparse) +setMethod("get_code", + signature = "teal_data", + definition = function(object, deparse = TRUE, datanames = NULL, names = datanames, ...) { + checkmate::assert_character(datanames, min.len = 1L, null.ok = TRUE) + checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) + checkmate::assert_flag(deparse) - if (!is.null(datanames)) { - lifecycle::deprecate_warn( - when = "0.6.1", - what = "get_code()", - with = "teal.code::get_code()", - always = TRUE - ) + if (!is.null(datanames)) { + lifecycle::deprecate_warn( + when = "0.6.1", + what = "get_code()", + with = "teal.code::get_code()", + always = TRUE + ) + } + teal.code::get_code(object = object, deparse = deparse, names = names, ...) } - teal.code::get_code(object = object, deparse = deparse, names = names, ...) -}) +) From b9773d17d067d9220825912b118020c2b720267a Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 09:21:27 +0200 Subject: [PATCH 12/24] overwrite the class --- R/teal_data-get_code.R | 1 + 1 file changed, 1 insertion(+) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index 220f92cf2..ff400044d 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -119,5 +119,6 @@ setMethod("get_code", signature = "teal_data", always = TRUE ) } + class(object) <- "qenv" teal.code::get_code(object = object, deparse = deparse, names = names, ...) }) From 530b7ba7578bc5210805ddb20cbb97230d79248c Mon Sep 17 00:00:00 2001 From: github-actions <41898282+github-actions[bot]@users.noreply.github.com> Date: Fri, 18 Oct 2024 07:24:37 +0000 Subject: [PATCH 13/24] [skip style] [skip vbump] Restyle files --- R/teal_data-get_code.R | 1 - 1 file changed, 1 deletion(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index e66c51c6d..7ee197b6d 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -124,4 +124,3 @@ setMethod("get_code", teal.code::get_code(object = object, deparse = deparse, names = names, ...) } ) - From 5318d386680be31a6af65d3c7c9b1b7a981f1bc9 Mon Sep 17 00:00:00 2001 From: Marcin <133694481+m7pr@users.noreply.github.com> Date: Fri, 18 Oct 2024 10:21:27 +0200 Subject: [PATCH 14/24] Update NEWS.md Co-authored-by: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Signed-off-by: Marcin <133694481+m7pr@users.noreply.github.com> --- NEWS.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/NEWS.md b/NEWS.md index e0ed19baf..9bccd4355 100644 --- a/NEWS.md +++ b/NEWS.md @@ -2,7 +2,7 @@ ### Deprecation -- `get_code(datanames)` `S4` method parameter for `teal_data()` object was soft deprecated. Use `get_code(names)`. +- soft deprecate `datanames` argument of `get_code()`. Use `names` instead. ### Enhancements From 7d1c6eebffd989cef1bea7129a1cd2955059bdc5 Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 10:27:13 +0200 Subject: [PATCH 15/24] use callNextMethod instead of setting up the class manually to qenv --- R/teal_data-get_code.R | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index e66c51c6d..6d957e34c 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -120,8 +120,7 @@ setMethod("get_code", always = TRUE ) } - class(object) <- "qenv" - teal.code::get_code(object = object, deparse = deparse, names = names, ...) + callNextMethod(object = object, deparse = deparse, names = names, ...) } ) From d2cec0ef4d82b62b9a751c7cf93cb76ea0002f50 Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 10:29:23 +0200 Subject: [PATCH 16/24] rename chapters in NEWS --- NEWS.md | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/NEWS.md b/NEWS.md index 9bccd4355..82f03c9de 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,10 +1,9 @@ # teal.data 0.6.0.9012 -### Deprecation +### Breaking changes - soft deprecate `datanames` argument of `get_code()`. Use `names` instead. - ### Enhancements - `datanames()` From c12c98333601249e461f3753d91329a28c2bf048 Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 10:32:06 +0200 Subject: [PATCH 17/24] change the order of names and datanames so names is used instead of datanames if arguments are not named --- R/teal_data-get_code.R | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index 7935e963f..c1acbae38 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -107,7 +107,7 @@ #' @export setMethod("get_code", signature = "teal_data", - definition = function(object, deparse = TRUE, datanames = NULL, names = datanames, ...) { + definition = function(object, deparse = TRUE, names = datanames, datanames = NULL, ...) { checkmate::assert_character(datanames, min.len = 1L, null.ok = TRUE) checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) checkmate::assert_flag(deparse) From 185db46a7e65bf1cb2c859181a1a506f6e350eae Mon Sep 17 00:00:00 2001 From: "27856297+dependabot-preview[bot]@users.noreply.github.com" <27856297+dependabot-preview[bot]@users.noreply.github.com> Date: Fri, 18 Oct 2024 08:34:28 +0000 Subject: [PATCH 18/24] [skip roxygen] [skip vbump] Roxygen Man Pages Auto Update --- man/get_code.Rd | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/man/get_code.Rd b/man/get_code.Rd index 9bfdf14f6..286bcc796 100644 --- a/man/get_code.Rd +++ b/man/get_code.Rd @@ -4,7 +4,7 @@ \alias{get_code,teal_data-method} \title{Get code from \code{teal_data} object} \usage{ -\S4method{get_code}{teal_data}(object, deparse = TRUE, datanames = NULL, names = datanames, ...) +\S4method{get_code}{teal_data}(object, deparse = TRUE, names = datanames, datanames = NULL, ...) } \arguments{ \item{object}{(\code{teal_data})} @@ -12,10 +12,10 @@ \item{deparse}{(\code{logical}) flag specifying whether to return code as \code{character} (\code{deparse = TRUE}) or as \code{expression} (\code{deparse = FALSE}).} -\item{datanames}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#experimental}{\figure{lifecycle-experimental.svg}{options: alt='[Experimental]'}}}{\strong{[Experimental]}} (\code{character}) vector of dataset names to return the code for. +\item{names}{(\code{character}) Successor of \code{datanames}. Vector of dataset names to return the code for. For more details see the "Extracting dataset-specific code" section.} -\item{names}{(\code{character}) Successor of \code{datanames}. Vector of dataset names to return the code for. +\item{datanames}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#experimental}{\figure{lifecycle-experimental.svg}{options: alt='[Experimental]'}}}{\strong{[Experimental]}} (\code{character}) vector of dataset names to return the code for. For more details see the "Extracting dataset-specific code" section.} \item{...}{Parameters passed to internal methods. Currently, the only supported parameter is \code{check_names} From 36e0a3305dc3c37aa4356fd79d96bffceff63b7c Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 10:39:07 +0200 Subject: [PATCH 19/24] use lifecycle::deprecate() and lifecycle::is_present() for datanames deprecation --- R/teal_data-get_code.R | 11 ++++++----- man/get_code.Rd | 12 +++++++++--- 2 files changed, 15 insertions(+), 8 deletions(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index c1acbae38..bdac283ca 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -107,18 +107,19 @@ #' @export setMethod("get_code", signature = "teal_data", - definition = function(object, deparse = TRUE, names = datanames, datanames = NULL, ...) { - checkmate::assert_character(datanames, min.len = 1L, null.ok = TRUE) + definition = function(object, deparse = TRUE, names = NULL, datanames = lifecycle::deprecated(), ...) { checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) checkmate::assert_flag(deparse) - if (!is.null(datanames)) { + if (lifecycle::is_present(datanames)) { lifecycle::deprecate_warn( when = "0.6.1", - what = "get_code()", - with = "teal.code::get_code()", + what = "teal.data::get_code(datanames = )", + with = "teal.code::get_code(names = )", always = TRUE ) + checkmate::assert_character(datanames, min.len = 1L, null.ok = TRUE) + names <- datanames } callNextMethod(object = object, deparse = deparse, names = names, ...) } diff --git a/man/get_code.Rd b/man/get_code.Rd index 9bfdf14f6..b7a53b98a 100644 --- a/man/get_code.Rd +++ b/man/get_code.Rd @@ -4,7 +4,13 @@ \alias{get_code,teal_data-method} \title{Get code from \code{teal_data} object} \usage{ -\S4method{get_code}{teal_data}(object, deparse = TRUE, datanames = NULL, names = datanames, ...) +\S4method{get_code}{teal_data}( + object, + deparse = TRUE, + names = NULL, + datanames = lifecycle::deprecated(), + ... +) } \arguments{ \item{object}{(\code{teal_data})} @@ -12,10 +18,10 @@ \item{deparse}{(\code{logical}) flag specifying whether to return code as \code{character} (\code{deparse = TRUE}) or as \code{expression} (\code{deparse = FALSE}).} -\item{datanames}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#experimental}{\figure{lifecycle-experimental.svg}{options: alt='[Experimental]'}}}{\strong{[Experimental]}} (\code{character}) vector of dataset names to return the code for. +\item{names}{(\code{character}) Successor of \code{datanames}. Vector of dataset names to return the code for. For more details see the "Extracting dataset-specific code" section.} -\item{names}{(\code{character}) Successor of \code{datanames}. Vector of dataset names to return the code for. +\item{datanames}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#experimental}{\figure{lifecycle-experimental.svg}{options: alt='[Experimental]'}}}{\strong{[Experimental]}} (\code{character}) vector of dataset names to return the code for. For more details see the "Extracting dataset-specific code" section.} \item{...}{Parameters passed to internal methods. Currently, the only supported parameter is \code{check_names} From 345f398b08f5847ee6392522406459d8836476e1 Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 12:14:40 +0200 Subject: [PATCH 20/24] change the bade from experimental to deprecated --- R/teal_data-get_code.R | 2 +- man/get_code.Rd | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index bdac283ca..b1f459bfc 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -72,7 +72,7 @@ #' #' #' @param object (`teal_data`) -#' @param datanames `r lifecycle::badge("experimental")` (`character`) vector of dataset names to return the code for. +#' @param datanames `r lifecycle::badge("deprecated")` (`character`) vector of dataset names to return the code for. #' For more details see the "Extracting dataset-specific code" section. #' @param names (`character`) Successor of `datanames`. Vector of dataset names to return the code for. #' For more details see the "Extracting dataset-specific code" section. diff --git a/man/get_code.Rd b/man/get_code.Rd index b7a53b98a..f742774c9 100644 --- a/man/get_code.Rd +++ b/man/get_code.Rd @@ -21,7 +21,7 @@ \item{names}{(\code{character}) Successor of \code{datanames}. Vector of dataset names to return the code for. For more details see the "Extracting dataset-specific code" section.} -\item{datanames}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#experimental}{\figure{lifecycle-experimental.svg}{options: alt='[Experimental]'}}}{\strong{[Experimental]}} (\code{character}) vector of dataset names to return the code for. +\item{datanames}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} (\code{character}) vector of dataset names to return the code for. For more details see the "Extracting dataset-specific code" section.} \item{...}{Parameters passed to internal methods. Currently, the only supported parameter is \code{check_names} From 4ce5c54eac9a97bfbde9513d6eab31ae98b582e9 Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 12:16:28 +0200 Subject: [PATCH 21/24] move if before the asserts --- R/teal_data-get_code.R | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index b1f459bfc..2115ca2f8 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -108,9 +108,6 @@ setMethod("get_code", signature = "teal_data", definition = function(object, deparse = TRUE, names = NULL, datanames = lifecycle::deprecated(), ...) { - checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) - checkmate::assert_flag(deparse) - if (lifecycle::is_present(datanames)) { lifecycle::deprecate_warn( when = "0.6.1", @@ -118,9 +115,12 @@ setMethod("get_code", with = "teal.code::get_code(names = )", always = TRUE ) - checkmate::assert_character(datanames, min.len = 1L, null.ok = TRUE) names <- datanames } + + checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) + checkmate::assert_flag(deparse) + callNextMethod(object = object, deparse = deparse, names = names, ...) } ) From c9074a3a95d52ca98d3615680b439ce50a4debc0 Mon Sep 17 00:00:00 2001 From: m7pr Date: Fri, 18 Oct 2024 14:50:41 +0200 Subject: [PATCH 22/24] add methods:: prefix and extract the deprecation message outside the if statement --- R/teal_data-get_code.R | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index 2115ca2f8..0895408ff 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -108,19 +108,19 @@ setMethod("get_code", signature = "teal_data", definition = function(object, deparse = TRUE, names = NULL, datanames = lifecycle::deprecated(), ...) { + lifecycle::deprecate_warn( + when = "0.6.1", + what = "teal.data::get_code()", + with = "teal.code::get_code()", + always = TRUE + ) if (lifecycle::is_present(datanames)) { - lifecycle::deprecate_warn( - when = "0.6.1", - what = "teal.data::get_code(datanames = )", - with = "teal.code::get_code(names = )", - always = TRUE - ) names <- datanames } checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) checkmate::assert_flag(deparse) - callNextMethod(object = object, deparse = deparse, names = names, ...) + methods::callNextMethod(object = object, deparse = deparse, names = names, ...) } ) From 5aa5a7f06dd4877c9a03fc260c4e5101ac2a4a1d Mon Sep 17 00:00:00 2001 From: m7pr Date: Mon, 21 Oct 2024 12:36:31 +0200 Subject: [PATCH 23/24] move back to the version where only datanames is deprecated, not the whole teal.data::get_code --- R/teal_data-get_code.R | 21 +++++++++++++-------- man/get_code.Rd | 4 ++-- 2 files changed, 15 insertions(+), 10 deletions(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index 0895408ff..de4e8c04e 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -1,6 +1,6 @@ #' Get code from `teal_data` object #' -#' `r lifecycle::badge("deprecated")` Use [`teal.code::get_code()`]. Retrieve code from `teal_data` object. +#' Retrieve code from `teal_data` object. #' #' Retrieve code stored in `@code`, which (in principle) can be used to recreate all objects found in `@env`. #' Use `names` to limit the code to one or more of the datasets enumerated in `@datanames`. @@ -73,7 +73,7 @@ #' #' @param object (`teal_data`) #' @param datanames `r lifecycle::badge("deprecated")` (`character`) vector of dataset names to return the code for. -#' For more details see the "Extracting dataset-specific code" section. +#' For more details see the "Extracting dataset-specific code" section. Use `names` instead. #' @param names (`character`) Successor of `datanames`. Vector of dataset names to return the code for. #' For more details see the "Extracting dataset-specific code" section. #' @param deparse (`logical`) flag specifying whether to return code as `character` (`deparse = TRUE`) or as @@ -108,16 +108,21 @@ setMethod("get_code", signature = "teal_data", definition = function(object, deparse = TRUE, names = NULL, datanames = lifecycle::deprecated(), ...) { - lifecycle::deprecate_warn( - when = "0.6.1", - what = "teal.data::get_code()", - with = "teal.code::get_code()", - always = TRUE - ) + if (lifecycle::is_present(datanames)) { + lifecycle::deprecate_warn( + when = "0.6.1", + what = "teal.data::get_code(datanames)", + with = "teal.code::get_code(names)", + always = TRUE + ) names <- datanames } + if (!is.null(names) && lifecycle::is_present(datanames)) { + stop("Please use either 'names' (recommended) or 'datanames' parameter.") + } + checkmate::assert_character(names, min.len = 1L, null.ok = TRUE) checkmate::assert_flag(deparse) diff --git a/man/get_code.Rd b/man/get_code.Rd index f742774c9..8a21424e7 100644 --- a/man/get_code.Rd +++ b/man/get_code.Rd @@ -22,7 +22,7 @@ For more details see the "Extracting dataset-specific code" section.} \item{datanames}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} (\code{character}) vector of dataset names to return the code for. -For more details see the "Extracting dataset-specific code" section.} +For more details see the "Extracting dataset-specific code" section. Use \code{names} instead.} \item{...}{Parameters passed to internal methods. Currently, the only supported parameter is \code{check_names} (\code{logical(1)}) flag, which is \code{TRUE} by default. Function warns about missing objects, if they do not exist in @@ -33,7 +33,7 @@ Either a character string or an expression. If \code{names} is used to request a only code that \emph{creates} that dataset (not code that uses it) is returned. Otherwise, all contents of \verb{@code}. } \description{ -\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} Use \code{\link[teal.code:qenv]{teal.code::get_code()}}. Retrieve code from \code{teal_data} object. +Retrieve code from \code{teal_data} object. } \details{ Retrieve code stored in \verb{@code}, which (in principle) can be used to recreate all objects found in \verb{@env}. From dbf3c29948f22cf5bb7c786595ebe7d23833a518 Mon Sep 17 00:00:00 2001 From: github-actions <41898282+github-actions[bot]@users.noreply.github.com> Date: Mon, 21 Oct 2024 10:38:33 +0000 Subject: [PATCH 24/24] [skip style] [skip vbump] Restyle files --- R/teal_data-get_code.R | 1 - 1 file changed, 1 deletion(-) diff --git a/R/teal_data-get_code.R b/R/teal_data-get_code.R index de4e8c04e..1fd668193 100644 --- a/R/teal_data-get_code.R +++ b/R/teal_data-get_code.R @@ -108,7 +108,6 @@ setMethod("get_code", signature = "teal_data", definition = function(object, deparse = TRUE, names = NULL, datanames = lifecycle::deprecated(), ...) { - if (lifecycle::is_present(datanames)) { lifecycle::deprecate_warn( when = "0.6.1",