Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

All datanames to transform@669 insert UI@main #1302

Merged
Merged
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
4ddb845
remove unused
gogonzo Aug 6, 2024
66172c8
don't reexport %>%
gogonzo Aug 6, 2024
b53841d
Merge remote-tracking branch 'origin/669_insertUI@main' into 905_data…
gogonzo Aug 7, 2024
f24241e
Merge branch '669_insertUI@main' into 905_datanames@669_insertUI@main
gogonzo Aug 7, 2024
74afb31
WIP
gogonzo Aug 8, 2024
61b184c
wip
gogonzo Aug 8, 2024
ea6d67d
Merge branch '669_insertUI@main' into all_datanames_to_transform@669_…
gogonzo Aug 8, 2024
b20ad2f
reload docs
gogonzo Aug 8, 2024
6904ce9
minor
gogonzo Aug 8, 2024
c556530
Merge remote-tracking branch 'origin/669_insertUI@main' into all_data…
gogonzo Aug 8, 2024
58c6d66
handle summary when no datanames
gogonzo Aug 8, 2024
a118338
Merge remote-tracking branch 'origin/669_insertUI@main' into all_data…
gogonzo Aug 9, 2024
845ef81
after review
gogonzo Aug 9, 2024
5778048
Merge remote-tracking branch 'origin/669_insertUI@main' into all_data…
gogonzo Aug 9, 2024
224de11
reload docs
gogonzo Aug 9, 2024
3d6579e
Merge remote-tracking branch 'origin/669_insertUI@main' into all_data…
gogonzo Aug 9, 2024
aaa866e
_raw to ._raw_
gogonzo Aug 9, 2024
3d16308
Merge branch '669_insertUI@main' into all_datanames_to_transform@669_…
gogonzo Aug 9, 2024
caa4519
suggestion @m7pr
gogonzo Aug 9, 2024
74be916
sample.int to seq
gogonzo Aug 9, 2024
f1ea412
closing bracket
gogonzo Aug 9, 2024
e597cdc
cast id sequence from double to int
gogonzo Aug 9, 2024
3c097b7
quote datanames in docs
m7pr Aug 9, 2024
8f7d726
Merge branch '1299_insufficient_datanames_error@669_insertUI@main' in…
gogonzo Aug 9, 2024
893cb9d
Merge remote-tracking branch 'origin/669_insertUI@main' into all_data…
gogonzo Aug 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion DESCRIPTION
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,6 @@ Imports:
jsonlite,
lifecycle (>= 0.2.0),
logger (>= 0.2.0),
magrittr (>= 1.5),
methods,
promises (>= 1.3.0),
renv (>= 1.0.7),
Expand Down Expand Up @@ -114,6 +113,7 @@ Collate:
'teal_data_module.R'
'teal_data_module-eval_code.R'
'teal_data_module-within.R'
'teal_data_utils.R'
'teal_lockfile.R'
'teal_reporter.R'
'teal_slices-store.R'
Expand Down
2 changes: 0 additions & 2 deletions NAMESPACE
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,6 @@ S3method(ui_teal_module,shiny.tag)
S3method(ui_teal_module,teal_module)
S3method(ui_teal_module,teal_modules)
S3method(within,teal_data_module)
export("%>%")
export(TealReportCard)
export(as.teal_slices)
export(as_tdata)
Expand Down Expand Up @@ -49,7 +48,6 @@ export(validate_one_row_per_id)
import(shiny)
import(teal.data)
import(teal.slice)
importFrom(magrittr,"%>%")
importFrom(methods,new)
importFrom(methods,setMethod)
importFrom(shiny,reactiveVal)
Expand Down
2 changes: 2 additions & 0 deletions NEWS.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,10 @@

### Breaking changes
* The `landing_popup_module()` needs to be passed as the `landing_popup` argument of `init` instead of being passed as a module of the `modules` argument of `init`
* `teal` no longer re-export `%>%` from `magrittr`. Please load `library(magrittr)` instead or use `|>` from `base`.

### Enhancement

* Provided progress bar for modules loading and data filtering during teal app startup.

### New features
Expand Down
44 changes: 0 additions & 44 deletions R/get_rcode_utils.R
Original file line number Diff line number Diff line change
Expand Up @@ -26,47 +26,3 @@ get_rcode_str_install <- function() {
"# Add any code to install/load your NEST environment here\n"
}
}

#' Get datasets code
#'
#' Retrieve complete code to create, verify, and filter a dataset.
#'
#' @param datanames (`character`) names of datasets to extract code from
#' @param datasets (`FilteredData`) object
#' @param hashes named (`list`) of hashes per dataset
#'
#' @return Character string concatenated from the following elements:
#' - data pre-processing code (from `data` argument in `init`)
#' - hash check of loaded objects
#' - filter code (if any)
#'
#' @keywords internal
get_datasets_code <- function(datanames, datasets, hashes) {
# preprocessing code
str_prepro <- attr(datasets, "preprocessing_code")
if (length(str_prepro) == 0) {
str_prepro <- "message('Preprocessing is empty')"
} else {
str_prepro <- paste(str_prepro, collapse = "\n")
}

# hash checks
str_hash <- vapply(datanames, function(dataname) {
sprintf(
"stopifnot(%s == %s)",
deparse1(bquote(rlang::hash(.(as.name(dataname))))),
deparse1(hashes[[dataname]])
)
}, character(1))
str_hash <- paste(str_hash, collapse = "\n")

# filter expressions
str_filter <- teal.slice::get_filter_expr(datasets, datanames)
if (str_filter == "") {
str_filter <- character(0)
}

# concatenate all code
str_code <- paste(c(str_prepro, str_hash, str_filter), collapse = "\n\n")
sprintf("%s\n", str_code)
}
6 changes: 3 additions & 3 deletions R/init.R
Original file line number Diff line number Diff line change
Expand Up @@ -215,20 +215,20 @@ init <- function(data,

## `data` - `modules`
if (inherits(data, "teal_data")) {
if (length(teal_data_datanames(data)) == 0) {
if (length(.teal_data_datanames(data)) == 0) {
stop("The environment of `data` is empty.")
}

if (!length(teal.data::datanames(data))) {
warning("`data` object has no datanames. Default datanames are set using `teal_data`'s environment.")
}

is_modules_ok <- check_modules_datanames(modules, teal_data_datanames(data))
is_modules_ok <- check_modules_datanames(modules, .teal_data_datanames(data))
if (!isTRUE(is_modules_ok)) {
logger::log_warn(is_modules_ok)
}

is_filter_ok <- check_filter_datanames(filter, teal_data_datanames(data))
is_filter_ok <- check_filter_datanames(filter, .teal_data_datanames(data))
if (!isTRUE(is_filter_ok)) {
warning(is_filter_ok)
# we allow app to continue if applied filters are outside
Expand Down
4 changes: 3 additions & 1 deletion R/module_data_summary.R
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ ui_data_summary <- function(id) {
ns <- NS(id)
content_id <- ns("filters_overview_contents")
tags$div(
id = id, # not used, can be used to customize CSS behavior
id = id,
class = "well",
tags$div(
class = "row",
Expand Down Expand Up @@ -93,6 +93,8 @@ srv_data_summary <- function(id, teal_data) {
if (!inherits(attr(summary_table_out, "condition"), "shiny.silent.error")) {
stop("Error occurred during data processing. See details in the main panel.")
}
} else if (is.null(summary_table_out)) {
"no datasets to show"
} else {
body_html <- apply(
summary_table_out,
Expand Down
49 changes: 15 additions & 34 deletions R/module_filter_data.R
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,8 @@
#' this special implementation all modules' data are recalculated only for those modules which are
#' currently displayed.
#'
#' @return A `eventReactive` which triggers only if all conditions are met:
#' @return A `eventReactive` containing `teal_data` containing filtered objects and filter code.
#' `eventReactive` triggers only if all conditions are met:
#' - tab is selected (`is_active`)
#' - when filters are changed (`get_filter_expr` is different than previous)
#'
Expand All @@ -32,10 +33,11 @@ srv_filter_data <- function(id, datasets, active_datanames, data_rv, is_active)
# render will be triggered only when FilteredData object changes (not when filters change)
# technically it means that teal_data_module needs to be refreshed
logger::log_debug("srv_filter_panel rendering filter panel.")
filtered_data <- datasets()
filtered_data$srv_active("filters", active_datanames = active_datanames)
# todo: make sure to bump the `teal.slice` version. Please use the branch `669_insertUI@main` in `teal.slice`.
filtered_data$ui_active(session$ns("filters"), active_datanames = active_datanames)
if (length(active_datanames())) {
datasets()$srv_active("filters", active_datanames = active_datanames)
# todo: make sure to bump the `teal.slice` version. Please use the branch `669_insertUI@main` in `teal.slice`.
datasets()$ui_active(session$ns("filters"), active_datanames = active_datanames)
}
})
})

Expand All @@ -49,33 +51,12 @@ srv_filter_data <- function(id, datasets, active_datanames, data_rv, is_active)

#' @rdname module_filter_data
.make_filtered_teal_data <- function(modules, data, datasets = NULL, datanames) {
new_datasets <- c(
# Filtered data
sapply(datanames, function(x) datasets$get_data(x, filtered = TRUE), simplify = FALSE),
# Raw (unfiltered data)
stats::setNames(
lapply(datanames, function(x) datasets$get_data(x, filtered = FALSE)),
sprintf("%s_raw", datanames)
)
)

data_code <- teal.data::get_code(data, datanames = datanames)
raw_data_code <- sprintf("%1$s_raw <- %1$s", datanames)
filter_code <- get_filter_expr(datasets = datasets, datanames = datanames)

all_code <- paste(unlist(c(data_code, raw_data_code, "", filter_code)), collapse = "\n")
tdata <- do.call(
teal.data::teal_data,
c(
list(code = trimws(all_code, which = "right")),
list(join_keys = teal.data::join_keys(data)),
new_datasets
)
)
tdata@verified <- data@verified
# we want to keep same datanames that app dev initially set with respect to new teal_data's @env
teal.data::datanames(tdata) <- intersect(teal.data::datanames(data), teal_data_ls(tdata))
tdata
data <- eval_code(data, sprintf("%1$s_raw <- %1$s", datanames))
filtered_code <- teal.slice::get_filter_expr(datasets = datasets, datanames = datanames)
filtered_teal_data <- .append_evaluated_code(data, filtered_code)
filtered_datasets <- sapply(datanames, function(x) datasets$get_data(x, filtered = TRUE), simplify = FALSE)
filtered_teal_data <- .append_modified_data(filtered_teal_data, filtered_datasets)
filtered_teal_data
}

#' @rdname module_filter_data
Expand All @@ -84,8 +65,8 @@ srv_filter_data <- function(id, datasets, active_datanames, data_rv, is_active)
filter_changed <- reactive({
req(inherits(datasets(), "FilteredData"))
new_signature <- c(
get_code(data_rv()),
get_filter_expr(datasets = datasets(), datanames = active_datanames())
teal.data::get_code(data_rv()),
teal.slice::get_filter_expr(datasets = datasets(), datanames = active_datanames())
)
if (!identical(previous_signature(), new_signature)) {
previous_signature(new_signature)
Expand Down
4 changes: 2 additions & 2 deletions R/module_init_data.R
Original file line number Diff line number Diff line change
Expand Up @@ -107,7 +107,7 @@ srv_init_data <- function(id, data, modules, filter = teal_slices()) {
)
}

is_filter_ok <- check_filter_datanames(filter, teal_data_datanames(data_validated()))
is_filter_ok <- check_filter_datanames(filter, .teal_data_datanames(data_validated()))
if (!isTRUE(is_filter_ok)) {
showNotification(
"Some filters were not applied because of incompatibility with data. Contact app developer.",
Expand Down Expand Up @@ -172,7 +172,7 @@ srv_init_data <- function(id, data, modules, filter = teal_slices()) {
#' @return A character vector with the code lines.
#' @keywords internal
#'
.get_hashes_code <- function(data, datanames = teal_data_datanames(data)) {
.get_hashes_code <- function(data, datanames = .teal_data_datanames(data)) {
# todo: this should be based on data_rv object not on datasets
vapply(
datanames,
Expand Down
9 changes: 4 additions & 5 deletions R/module_nested_tabs.R
Original file line number Diff line number Diff line change
Expand Up @@ -228,9 +228,8 @@ srv_teal_module.teal_module <- function(id,

module_teal_data <- reactive({
all_teal_data <- transformed_teal_data()
# todo: create a new teal_data object with code subset, datasets and datanames (not just limit datanames)
teal.data::datanames(all_teal_data) <- .resolve_module_datanames(data = all_teal_data, modules = modules)
all_teal_data
module_datanames <- .resolve_module_datanames(data = all_teal_data, modules = modules)
.subset_teal_data(all_teal_data, module_datanames)
})

module_teal_data_validated <- srv_validate_reactive_teal_data(
Expand Down Expand Up @@ -302,11 +301,11 @@ srv_teal_module.teal_module <- function(id,
.resolve_module_datanames <- function(data, modules) {
stopifnot("data_rv must be teal_data object." = inherits(data, "teal_data"))
if (is.null(modules$datanames) || identical(modules$datanames, "all")) {
teal_data_datanames(data)
.teal_data_datanames(data)
} else {
intersect(
include_parent_datanames(modules$datanames, teal.data::join_keys(data)),
teal_data_ls(data)
.teal_data_ls(data)
)
}
}
2 changes: 1 addition & 1 deletion R/module_teal_data.R
Original file line number Diff line number Diff line change
Expand Up @@ -162,7 +162,7 @@ srv_validate_reactive_teal_data <- function(id, # nolint: object_length

output$shiny_warnings <- renderUI({
if (inherits(data_out_rv(), "teal_data")) {
is_modules_ok <- check_modules_datanames(modules = modules, datanames = teal_data_ls(data_validated()))
is_modules_ok <- check_modules_datanames(modules = modules, datanames = .teal_data_ls(data_validated()))
if (!isTRUE(is_modules_ok)) {
tags$div(is_modules_ok, class = "teal-output-warning")
}
Expand Down
4 changes: 2 additions & 2 deletions R/tdata.R
Original file line number Diff line number Diff line change
Expand Up @@ -188,12 +188,12 @@ as_tdata <- function(x) {
}
if (is.reactive(x)) {
checkmate::assert_class(isolate(x()), "teal_data")
datanames <- isolate(teal_data_datanames(x()))
datanames <- isolate(.teal_data_datanames(x()))
datasets <- sapply(datanames, function(dataname) reactive(x()[[dataname]]), simplify = FALSE)
code <- reactive(teal.code::get_code(x()))
join_keys <- isolate(teal.data::join_keys(x()))
} else if (inherits(x, "teal_data")) {
datanames <- teal_data_datanames(x)
datanames <- .teal_data_datanames(x)
datasets <- sapply(datanames, function(dataname) reactive(x[[dataname]]), simplify = FALSE)
code <- reactive(teal.code::get_code(x))
join_keys <- isolate(teal.data::join_keys(x))
Expand Down
2 changes: 0 additions & 2 deletions R/teal.R
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,4 @@
#' @import shiny teal.data teal.slice
#' @importFrom stats setNames
#' @export
#' @importFrom magrittr %>%
magrittr::`%>%`
NULL
69 changes: 69 additions & 0 deletions R/teal_data_utils.R
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
#' `teal_data` utils
#'
#' In `teal` we need to recreate the `teal_data` object due to two operations:integer.max
#' - we need to append filter-data code and objects which have been evaluated in `FilteredData` and
#' we want to avoid double-evaluation.
#' - we need to subset `teal_data` to datanames used by the module, to shorten obtainable R-code
#'
#' Due to above recreation of `teal_data` object can't be done simply by using public
#' `teal.code` and `teal.data` methods.
#'
#' @param data (`teal_data`)
#' @param code (`character`) code to append to `data@code`
#' @param objects (`list`) objects to append to `data@env`
#' @param datanames (`character`) names of the datasets
#' @return modified `teal_data`
#' @name teal_data_utilities
NULL

#' @rdname teal_data_utilities
.append_evaluated_code <- function(data, code) {
checkmate::assert_class(data, "teal_data")
data@code <- c(data@code, code)
data@id <- c(data@id, sample.int(.Machine$integer.max, size = length(code)))
gogonzo marked this conversation as resolved.
Show resolved Hide resolved
data@messages <- c(data@messages, rep("", length(code)))
data@warnings <- c(data@warnings, rep("", length(code)))
methods::validObject(data)
data
}

#' @rdname teal_data_utilities
.append_modified_data <- function(data, objects) {
checkmate::assert_class(data, "teal_data")
checkmate::assert_class(objects, "list")
new_env <- list2env(objects, parent = .GlobalEnv)
rlang::env_coalesce(new_env, data@env)
m7pr marked this conversation as resolved.
Show resolved Hide resolved
data@env <- new_env
data
}

#' @rdname teal_data_utilities
.subset_teal_data <- function(data, datanames) {
checkmate::assert_class(data, "teal_data")
checkmate::assert_class(datanames, "character")
new_data <- data
datasets_to_extract <- intersect(c(datanames, sprintf("%s_raw", datanames)), ls(data@env))
new_data@code <- get_code_dependency(data@code, datasets_to_extract)
m7pr marked this conversation as resolved.
Show resolved Hide resolved
new_data@id <- sample.int(.Machine$integer.max, size = length(new_data@code))
new_data@warnings <- rep("", length(new_data@code))
new_data@messages <- rep("", length(new_data@code))
m7pr marked this conversation as resolved.
Show resolved Hide resolved
new_data@env <- list2env(mget(x = datasets_to_extract, envir = data@env))
m7pr marked this conversation as resolved.
Show resolved Hide resolved
teal.data::datanames(new_data) <- datanames
methods::validObject(new_data)
new_data
}

#' @rdname teal_data_utilities
.teal_data_datanames <- function(data) {
checkmate::assert_class(data, "teal_data")
if (length(teal.data::datanames(data))) {
teal.data::datanames(data)
} else {
.teal_data_ls(data)
}
}
m7pr marked this conversation as resolved.
Show resolved Hide resolved

#' @rdname teal_data_utilities
.teal_data_ls <- function(data) {
grep("_raw$", ls(teal.code::get_env(data), all.names = TRUE), value = TRUE, invert = TRUE)
}
Loading
Loading