From 91e000d7094389b41ec87ae5c7d98537182c0085 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Dawid=20Ka=C5=82=C4=99dkowski?= <6959016+gogonzo@users.noreply.github.com> Date: Mon, 13 Nov 2023 19:32:42 +0100 Subject: [PATCH] Introduce the new DDL (#957) DDL --- DESCRIPTION | 1 + NAMESPACE | 1 + NEWS.md | 14 +- R/dummy_functions.R | 10 +- R/init.R | 70 ++++--- R/landing_popup_module.R | 4 +- R/module_nested_tabs.R | 6 +- R/module_snapshot_manager.R | 3 +- R/module_tabs_with_filters.R | 8 +- R/module_teal.R | 60 +++--- R/module_teal_with_splash.R | 114 ++++++++++-- R/modules.R | 2 +- R/teal_data_module.R | 52 ++++++ R/utils.R | 75 ++++++++ _pkgdown.yml | 2 + man/check_filter_datanames.Rd | 21 +++ man/check_modules_datanames.Rd | 21 +++ man/init.Rd | 5 +- man/landing_popup_module.Rd | 4 +- man/module_teal.Rd | 10 +- man/snapshot_manager_module.Rd | 3 +- man/srv_teal_with_splash.Rd | 5 +- man/teal_data_module.Rd | 55 ++++++ man/ui_teal_with_splash.Rd | 5 +- tests/testthat/test-init.R | 50 ++++- tests/testthat/test-module_nested_tabs.R | 20 +- tests/testthat/test-module_teal.R | 65 ++++--- tests/testthat/test-module_teal_with_splash.R | 173 ++++++++++++++++-- tests/testthat/test-rcode_utils.R | 22 +-- tests/testthat/test-teal_data_module.R | 20 ++ vignettes/adding-support-for-reporting.Rmd | 2 +- vignettes/data-as-shiny-module.Rmd | 126 +++++++++++++ 32 files changed, 843 insertions(+), 186 deletions(-) create mode 100644 R/teal_data_module.R create mode 100644 man/check_filter_datanames.Rd create mode 100644 man/check_modules_datanames.Rd create mode 100644 man/teal_data_module.Rd create mode 100644 tests/testthat/test-teal_data_module.R create mode 100644 vignettes/data-as-shiny-module.Rmd diff --git a/DESCRIPTION b/DESCRIPTION index 2591116aee..3d099a4a43 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -88,6 +88,7 @@ Collate: 'show_rcode_modal.R' 'tdata.R' 'teal.R' + 'teal_data_module.R' 'teal_reporter.R' 'teal_slices-store.R' 'teal_slices.R' diff --git a/NAMESPACE b/NAMESPACE index 242ffaf984..e8f450b895 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -31,6 +31,7 @@ export(reporter_previewer_module) export(show_rcode_modal) export(srv_teal_with_splash) export(tdata2env) +export(teal_data_module) export(teal_slices) export(ui_teal_with_splash) export(validate_has_data) diff --git a/NEWS.md b/NEWS.md index 6f6a63a350..a3910fc9f1 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,13 +1,19 @@ # teal 0.14.0.9017 +### New features + +* `data` argument in `init` now accepts `teal_data` and `teal_data_module`. +* Added `landing_popup_module` function which creates a module that will display a popup when the app starts. The popup will block access to the app until it is dismissed. +* Filter state snapshots can now be uploaded from file. See `?snapshot`. + ### Miscellaneous * Enhanced a `module` validation checks so that it won't throw messages about `data` argument unnecessarily. -* Removed `Report previewer` module from mapping matrix display in filter manager. -* Added internal functions for storing and restoring of `teal_slices` objects. -* Filter state snapshots can now be uploaded from file. See `?snapshot`. * Added argument to `teal_slices` and made modifications to `init` to enable tagging `teal_slices` with an app id to safely upload snapshots from disk. -* Added `landing_popup_module` function which creates a module that will display a popup when the app starts. The popup will block access to the app until it is dismissed. + +### Bug fixes + +* Removed `Report previewer` module from mapping matrix display in filter manager. # teal 0.14.0 diff --git a/R/dummy_functions.R b/R/dummy_functions.R index 1fdaea0d8e..608c1a3241 100644 --- a/R/dummy_functions.R +++ b/R/dummy_functions.R @@ -96,13 +96,21 @@ example_module <- function(label = "example teal module", datanames = "all") { checkmate::assert_class(data, "tdata") moduleServer(id, function(input, output, session) { output$text <- renderPrint(data[[input$dataname]]()) + teal.widgets::verbatim_popup_srv( + id = "rcode", + verbatim_content = attr(data, "code")(), + title = "Association Plot" + ) }) }, ui = function(id, data) { ns <- NS(id) teal.widgets::standard_layout( output = verbatimTextOutput(ns("text")), - encoding = selectInput(ns("dataname"), "Choose a dataset", choices = names(data)) + encoding = div( + selectInput(ns("dataname"), "Choose a dataset", choices = names(data)), + teal.widgets::verbatim_popup_ui(ns("rcode"), "Show R code") + ) ) }, datanames = datanames diff --git a/R/init.R b/R/init.R index b697e2829b..8035815e7d 100644 --- a/R/init.R +++ b/R/init.R @@ -15,10 +15,11 @@ #' an end-user, don't use this function, but instead this module. #' #' @param data (`TealData` or `TealDataset` or `TealDatasetConnector` or `list` or `data.frame` -#' or `MultiAssayExperiment`, `teal_data`)\cr +#' or `MultiAssayExperiment`, `teal_data`, `teal_data_module`)\cr #' `R6` object as returned by [teal.data::cdisc_data()], [teal.data::teal_data()], #' [teal.data::cdisc_dataset()], [teal.data::dataset()], [teal.data::dataset_connector()] or -#' [teal.data::cdisc_dataset_connector()] or a single `data.frame` or a `MultiAssayExperiment` +#' [teal.data::cdisc_dataset_connector()] or [teal_data_module()] or a single `data.frame` or +#' a `MultiAssayExperiment` #' or a list of the previous objects or function returning a named list. #' NOTE: teal does not guarantee reproducibility of the code when names of the list elements #' do not match the original object names. To ensure reproducibility please use [teal.data::teal_data()] @@ -114,11 +115,11 @@ init <- function(data, footer = tags$p(), id = character(0)) { logger::log_trace("init initializing teal app with: data ({ class(data)[1] }).") - - if (!inherits(data, c("TealData", "teal_data"))) { + if (!inherits(data, c("TealData", "teal_data", "teal_data_module"))) { data <- teal.data::to_relational_data(data = data) } - checkmate::assert_multi_class(data, c("TealData", "teal_data")) + + checkmate::assert_multi_class(data, c("TealData", "teal_data", "teal_data_module")) checkmate::assert_multi_class(modules, c("teal_module", "list", "teal_modules")) checkmate::assert_string(title, null.ok = TRUE) checkmate::assert( @@ -142,26 +143,14 @@ init <- function(data, if (length(landing) > 1L) stop("Only one `landing_popup_module` can be used.") modules <- drop_module(modules, "teal_module_landing") - # resolve modules datanames - datanames <- teal.data::get_dataname(data) - join_keys <- teal.data::get_join_keys(data) - modules <- resolve_modules_datanames(modules = modules, datanames = datanames, join_keys = join_keys) - - if (!inherits(filter, "teal_slices")) { - checkmate::assert_subset(names(filter), choices = datanames) - # list_to_teal_slices is lifted from teal.slice package, see zzz.R - # This is a temporary measure and will be removed two release cycles from now (now meaning 0.13.0). - filter <- list_to_teal_slices(filter) - } - # convert teal.slice::teal_slices to teal::teal_slices - filter <- as.teal_slices(as.list(filter)) - - # Calculate app hash to ensure snapshot compatibility. See ?snapshot. Raw data must be extracted from environments. + # Calculate app id that will be used to stamp filter state snapshots. + # App id is a hash of the app's data and modules. + # See "transferring snapshots" section in ?snapshot. hashables <- mget(c("data", "modules")) hashables$data <- if (inherits(hashables$data, "teal_data")) { as.list(hashables$data@env) - } else if (inherits(hashables$data, "ddl")) { - attr(hashables$data, "code") + } else if (inherits(data, "teal_data_module")) { + body(data$server) } else if (hashables$data$is_pulled()) { sapply(get_dataname(hashables$data), simplify = FALSE, function(dn) { hashables$data$get_dataset(dn)$get_raw_data() @@ -172,20 +161,8 @@ init <- function(data, attr(filter, "app_id") <- rlang::hash(hashables) - # check teal_slices - for (i in seq_along(filter)) { - dataname_i <- shiny::isolate(filter[[i]]$dataname) - if (!dataname_i %in% datanames) { - stop( - sprintf( - "filter[[%s]] has a different dataname than available in a 'data':\n %s not in %s", - i, - dataname_i, - toString(datanames) - ) - ) - } - } + # convert teal.slice::teal_slices to teal::teal_slices + filter <- as.teal_slices(as.list(filter)) if (isTRUE(attr(filter, "module_specific"))) { module_names <- unlist(c(module_labels(modules), "global_filters")) @@ -213,6 +190,27 @@ init <- function(data, } } + if (inherits(data, "teal_data")) { + if (length(teal.data::datanames(data)) == 0) { + stop("`data` object has no datanames. Specify `datanames(data)` and try again.") + } + + # in case of teal_data_module this check is postponed to the srv_teal_with_splash + is_modules_ok <- check_modules_datanames(modules, teal.data::datanames(data)) + if (!isTRUE(is_modules_ok)) { + logger::log_error(is_modules_ok) + checkmate::assert(is_modules_ok, .var.name = "modules") + } + + + is_filter_ok <- check_filter_datanames(filter, teal.data::datanames(data)) + if (!isTRUE(is_filter_ok)) { + logger::log_warn(is_filter_ok) + # we allow app to continue if applied filters are outside + # of possible data range + } + } + # Note regarding case `id = character(0)`: # rather than using `callModule` and creating a submodule of this module, we directly modify # the `ui` and `server` with `id = character(0)` and calling the server function directly diff --git a/R/landing_popup_module.R b/R/landing_popup_module.R index cb94e14484..fd2501c329 100644 --- a/R/landing_popup_module.R +++ b/R/landing_popup_module.R @@ -15,7 +15,7 @@ #' #' @examples #' app1 <- teal::init( -#' data = teal.data::dataset("iris", iris), +#' data = teal_data(iris = iris), #' modules = teal::modules( #' teal::landing_popup_module( #' content = "A place for the welcome message or a disclaimer statement.", @@ -29,7 +29,7 @@ #' } #' #' app2 <- teal::init( -#' data = teal.data::dataset("iris", iris), +#' data = teal_data(iris = iris), #' modules = teal::modules( #' teal::landing_popup_module( #' title = "Welcome", diff --git a/R/module_nested_tabs.R b/R/module_nested_tabs.R index 0a012f32dd..b84fb45277 100644 --- a/R/module_nested_tabs.R +++ b/R/module_nested_tabs.R @@ -297,7 +297,11 @@ srv_nested_tabs.teal_module <- function(id, datasets, modules, is_module_specifi checkmate::assert_class(datasets, "FilteredData") checkmate::assert_class(trigger_data, "reactiveVal") - datanames <- if (is.null(module$datanames)) datasets$datanames() else module$datanames + datanames <- if (is.null(module$datanames) || identical(module$datanames, "all")) { + datasets$datanames() + } else { + unique(module$datanames) # todo: include parents! unique shouldn't be needed here! + } # list of reactive filtered data data <- sapply( diff --git a/R/module_snapshot_manager.R b/R/module_snapshot_manager.R index a337e5f5ce..da49dd842e 100644 --- a/R/module_snapshot_manager.R +++ b/R/module_snapshot_manager.R @@ -59,7 +59,8 @@ #' which is disassembled for storage and used directly for restoring app state. #' #' @section Transferring snapshots: -#' Snapshots uploaded from disk should only be used in the same application they come from. +#' Snapshots uploaded from disk should only be used in the same application they come from, +#' _i.e._ an application that uses the same data and the same modules. #' To ensure this is the case, `init` stamps `teal_slices` with an app id that is stored in the `app_id` attribute of #' a `teal_slices` object. When a snapshot is restored from file, its `app_id` is compared to that #' of the current app state and only if the match is the snapshot admitted to the session. diff --git a/R/module_tabs_with_filters.R b/R/module_tabs_with_filters.R index 13fd6d5ebe..7f2fca1406 100644 --- a/R/module_tabs_with_filters.R +++ b/R/module_tabs_with_filters.R @@ -122,7 +122,13 @@ srv_tabs_with_filters <- function(id, ) if (!is_module_specific) { - active_datanames <- reactive(active_module()$datanames) + active_datanames <- reactive({ + if (identical(active_module()$datanames, "all")) { + singleton$datanames() + } else { + active_module()$datanames + } + }) singleton <- unlist(datasets)[[1]] singleton$srv_filter_panel("filter_panel", active_datanames = active_datanames) diff --git a/R/module_teal.R b/R/module_teal.R index 413e47d349..277ccdd198 100644 --- a/R/module_teal.R +++ b/R/module_teal.R @@ -15,7 +15,7 @@ #' for non-delayed data which takes time to load into memory, avoiding #' Shiny session timeouts. #' -#' Server evaluates the `raw_data` (delayed data mechanism) and creates the +#' Server evaluates the `teal_data_rv` (delayed data mechanism) and creates the #' `datasets` object that is shared across modules. #' Once it is ready and non-`NULL`, the splash screen is replaced by the #' main teal UI that depends on the data. @@ -33,7 +33,7 @@ #' can be a splash screen or a Shiny module UI. For the latter, see #' [init()] about how to call the corresponding server function. #' -#' @param raw_data (`reactive`)\cr +#' @param teal_data_rv (`reactive`)\cr #' returns the `teal_data`, only evaluated once, `NULL` value is ignored #' #' @return @@ -44,13 +44,13 @@ #' #' @examples #' mods <- teal:::example_modules() -#' raw_data <- reactive(teal:::example_cdisc_data()) +#' teal_data_rv <- reactive(teal:::example_cdisc_data()) #' app <- shinyApp( #' ui = function() { #' teal:::ui_teal("dummy") #' }, #' server = function(input, output, session) { -#' active_module <- teal:::srv_teal(id = "dummy", modules = mods, raw_data = raw_data) +#' active_module <- teal:::srv_teal(id = "dummy", modules = mods, teal_data_rv = teal_data_rv) #' } #' ) #' if (interactive()) { @@ -130,8 +130,8 @@ ui_teal <- function(id, #' @rdname module_teal -srv_teal <- function(id, modules, raw_data, filter = teal_slices()) { - stopifnot(is.reactive(raw_data)) +srv_teal <- function(id, modules, teal_data_rv, filter = teal_slices()) { + stopifnot(is.reactive(teal_data_rv)) moduleServer(id, function(input, output, session) { logger::log_trace("srv_teal initializing the module.") @@ -160,17 +160,23 @@ srv_teal <- function(id, modules, raw_data, filter = teal_slices()) { } ) + reporter <- teal.reporter::Reporter$new() + if (is_arg_used(modules, "reporter") && length(extract_module(modules, "teal_module_previewer")) == 0) { + modules <- append_module(modules, reporter_previewer_module()) + } + env <- environment() - datasets_reactive <- eventReactive(raw_data(), { + datasets_reactive <- eventReactive(teal_data_rv(), { env$progress <- shiny::Progress$new(session) env$progress$set(0.25, message = "Setting data") # create a list of data following structure of the nested modules list structure. # Because it's easier to unpack modules and datasets when they follow the same nested structure. - datasets_singleton <- teal_data_to_filtered_data(raw_data()) + datasets_singleton <- teal_data_to_filtered_data(teal_data_rv()) # Singleton starts with only global filters active. filter_global <- Filter(function(x) x$id %in% attr(filter, "mapping")$global_filters, filter) datasets_singleton$set_filter_state(filter_global) + module_datasets <- function(modules) { if (inherits(modules, "teal_modules")) { datasets <- lapply(modules$children, module_datasets) @@ -180,11 +186,19 @@ srv_teal <- function(id, modules, raw_data, filter = teal_slices()) { } else if (isTRUE(attr(filter, "module_specific"))) { # we should create FilteredData even if modules$datanames is null # null controls a display of filter panel but data should be still passed - datanames <- if (is.null(modules$datanames)) teal.data::get_dataname(raw_data()) else modules$datanames - # todo: subset tdata object to datanames - datasets_module <- teal_data_to_filtered_data(raw_data()) + datanames <- if (is.null(modules$datanames) || modules$datanames == "all") { + include_parent_datanames( + teal.data::datanames(teal_data_rv()), + teal_data_rv()@join_keys + ) + } else { + modules$datanames + } + # todo: subset teal_data to datanames + datasets_module <- teal_data_to_filtered_data(teal_data_rv()) # set initial filters + # - filtering filters for this module slices <- Filter(x = filter, f = function(x) { x$id %in% unique(unlist(attr(filter, "mapping")[c(modules$label, "global_filters")])) && x$dataname %in% datanames @@ -199,29 +213,23 @@ srv_teal <- function(id, modules, raw_data, filter = teal_slices()) { datasets_singleton } } - datasets <- module_datasets(modules) - - logger::log_trace("srv_teal@4 Raw Data transferred to FilteredData.") - datasets + module_datasets(modules) }) - reporter <- teal.reporter::Reporter$new() - if (is_arg_used(modules, "reporter") && length(extract_module(modules, "teal_module_previewer")) == 0) { - modules <- append_module(modules, reporter_previewer_module()) - } - # Replace splash / welcome screen once data is loaded ---- # ignoreNULL to not trigger at the beginning when data is NULL # just handle it once because data obtained through delayed loading should # usually not change afterwards # if restored from bookmarked state, `filter` is ignored - observeEvent(datasets_reactive(), ignoreNULL = TRUE, once = TRUE, { + + observeEvent(datasets_reactive(), once = TRUE, { logger::log_trace("srv_teal@5 setting main ui after data was pulled") - env$progress$set(0.5, message = "Setting up main UI") on.exit(env$progress$close()) - # main_ui_container contains splash screen first and we remove it and replace it by the real UI + env$progress$set(0.5, message = "Setting up main UI") + datasets <- datasets_reactive() - removeUI(sprintf("#%s:first-child", session$ns("main_ui_container"))) + # main_ui_container contains splash screen first and we remove it and replace it by the real UI + removeUI(sprintf("#%s > div:nth-child(1)", session$ns("main_ui_container"))) insertUI( selector = paste0("#", session$ns("main_ui_container")), where = "beforeEnd", @@ -230,7 +238,7 @@ srv_teal <- function(id, modules, raw_data, filter = teal_slices()) { ui = div(ui_tabs_with_filters( session$ns("main_ui"), modules = modules, - datasets = datasets_reactive(), + datasets = datasets, filter = filter )), # needed so that the UI inputs are available and can be immediately updated, otherwise, updating may not @@ -242,7 +250,7 @@ srv_teal <- function(id, modules, raw_data, filter = teal_slices()) { # registered once (calling server functions twice would trigger observers twice each time) active_module <- srv_tabs_with_filters( id = "main_ui", - datasets = datasets_reactive(), + datasets = datasets, modules = modules, reporter = reporter, filter = filter diff --git a/R/module_teal_with_splash.R b/R/module_teal_with_splash.R index 01348381b2..f1e933cf19 100644 --- a/R/module_teal_with_splash.R +++ b/R/module_teal_with_splash.R @@ -22,14 +22,17 @@ ui_teal_with_splash <- function(id, title, header = tags$p("Add Title Here"), footer = tags$p("Add Footer Here")) { - checkmate::assert_multi_class(data, c("TealDataAbstract", "teal_data")) + checkmate::assert_multi_class(data, c("TealData", "teal_data", "teal_data_module")) ns <- NS(id) # Startup splash screen for delayed loading # We use delayed loading in all cases, even when the data does not need to be fetched. # This has the benefit that when filtering the data takes a lot of time initially, the # Shiny app does not time out. - splash_ui <- if (inherits(data, "teal_data")) { + + splash_ui <- if (inherits(data, "teal_data_module")) { + data$ui(ns("teal_data_module")) + } else if (inherits(data, "teal_data")) { div() } else if (inherits(data, "TealDataAbstract") && teal.data::is_pulled(data)) { div() @@ -37,8 +40,13 @@ ui_teal_with_splash <- function(id, message("App was initialized with delayed data loading.") data$get_ui(ns("startapp_module")) } - - ui_teal(id = ns("teal"), splash_ui = splash_ui, title = title, header = header, footer = footer) + ui_teal( + id = ns("teal"), + splash_ui = div(splash_ui, uiOutput(ns("error"))), + title = title, + header = header, + footer = footer + ) } #' Server function that loads the data through reactive loading and then delegates @@ -56,17 +64,20 @@ ui_teal_with_splash <- function(id, #' If data is not loaded yet, `reactive` returns `NULL`. #' @export srv_teal_with_splash <- function(id, data, modules, filter = teal_slices()) { - checkmate::assert_multi_class(data, c("TealDataAbstract", "teal_data")) + checkmate::check_multi_class(data, c("TealData", "teal_data", "teal_data_module")) + moduleServer(id, function(input, output, session) { - logger::log_trace("srv_teal_with_splash initializing module with data { toString(get_dataname(data))}.") + logger::log_trace("srv_teal_with_splash initializing module with data.") if (getOption("teal.show_js_log", default = FALSE)) { shinyjs::showLog() } - # raw_data contains teal_data object - # either passed to teal::init or returned from ddl - raw_data <- if (inherits(data, "teal_data")) { + # teal_data_rv contains teal_data object + # either passed to teal::init or returned from teal_data_module + teal_data_rv <- if (inherits(data, "teal_data_module")) { + data$server(id = "teal_data_module") + } else if (inherits(data, "teal_data")) { reactiveVal(data) } else if (inherits(data, "TealDataAbstract") && teal.data::is_pulled(data)) { new_data <- do.call( @@ -95,15 +106,88 @@ srv_teal_with_splash <- function(id, data, modules, filter = teal_slices()) { ) } }) - - if (!is.reactive(raw_data)) { - stop("The delayed loading module has to return a reactive object.") - } raw_data } - res <- srv_teal(id = "teal", modules = modules, raw_data = raw_data, filter = filter) - logger::log_trace("srv_teal_with_splash initialized module with data { toString(get_dataname(data))}.") + if (!is.reactive(teal_data_rv)) { + stop("The `teal_data_module` must return a reactive expression.", call. = FALSE) + } + + teal_data_rv_validate <- reactive({ + # custom module can return error + data <- tryCatch(teal_data_rv(), error = function(e) e) + + # there is an empty reactive event on init! + if (inherits(data, "shiny.silent.error") && identical(data$message, "")) { + return(NULL) + } + + # to handle qenv.error + if (inherits(data, "qenv.error")) { + validate( + need( + FALSE, + paste( + "Error when executing `teal_data_module`:\n ", + paste(data$message, collapse = "\n"), + "\n Check your inputs or contact app developer if error persists." + ) + ) + ) + } + + # to handle module non-qenv errors + if (inherits(data, "error")) { + validate( + need( + FALSE, + paste0( + "Error when executing `teal_data_module`:\n ", + paste(data$message, collpase = "\n"), + "\n Check your inputs or contact app developer if error persists." + ) + ) + ) + } + + validate( + need( + inherits(data, "teal_data"), + paste( + "Error: `teal_data_module` did not return `teal_data` object", + "\n Check your inputs or contact app developer if error persists" + ) + ) + ) + + validate(need(teal.data::datanames(data), "Data has no datanames. Contact app developer.")) + + + is_modules_ok <- check_modules_datanames(modules, teal.data::datanames(data)) + validate(need(isTRUE(is_modules_ok), is_modules_ok)) + + is_filter_ok <- check_filter_datanames(filter, teal.data::datanames(data)) + if (!isTRUE(is_filter_ok)) { + showNotification( + "Some filters were not applied because of incompatibility with data. Contact app developer.", + type = "warning", + duration = 10 + ) + logger::log_warn(is_filter_ok) + } + + teal_data_rv() + }) + + output$error <- renderUI({ + teal_data_rv_validate() + NULL + }) + + + + res <- srv_teal(id = "teal", modules = modules, teal_data_rv = teal_data_rv_validate, filter = filter) + logger::log_trace("srv_teal_with_splash initialized module with data.") return(res) }) } diff --git a/R/modules.R b/R/modules.R index 472a1076b1..789083e669 100644 --- a/R/modules.R +++ b/R/modules.R @@ -318,7 +318,7 @@ module <- function(label = "module", structure( list( label = label, - server = server, ui = ui, datanames = datanames, + server = server, ui = ui, datanames = unique(datanames), server_args = server_args, ui_args = ui_args ), class = "teal_module" diff --git a/R/teal_data_module.R b/R/teal_data_module.R new file mode 100644 index 0000000000..5f63395fca --- /dev/null +++ b/R/teal_data_module.R @@ -0,0 +1,52 @@ +#' Data module for `teal` applications +#' +#' Creates `teal_data_module` object - a `shiny` module to supply or modify data in a `teal` application. +#' +#' This function creates a `shiny` module that allows for running data pre-processing code after the app starts. +#' The body of the server function will be run in the app rather than in the global environment. +#' This means it will be run every time the app starts, so use sparingly. +#' +#' Pass this module instead of a `teal_data` object in a call to [init()]. +#' +#' See vignette \code{vignette("data-as-shiny-module", package = "teal")} for more details. +#' +#' @param ui (`function(id)`)\cr +#' `shiny` module `ui` function; must only take `id` argument +#' @param server (`function(id)`)\cr +#' `shiny` module `ui` function; must only take `id` argument; +#' must return reactive expression containing `teal_data` object +#' +#' @return Object of class `teal_data_module`. +#' +#' @examples +#' data <- teal_data_module( +#' ui = function(id) { +#' ns <- NS(id) +#' actionButton(ns("submit"), label = "Load data") +#' }, +#' server = function(id) { +#' moduleServer(id, function(input, output, session) { +#' eventReactive(input$submit, { +#' data <- within( +#' teal_data(), +#' { +#' dataset1 <- iris +#' dataset2 <- mtcars +#' } +#' ) +#' datanames(data) <- c("iris", "mtcars") +#' +#' data +#' }) +#' }) +#' } +#' ) +#' @export +teal_data_module <- function(ui, server) { + checkmate::assert_function(ui, args = "id", nargs = 1) + checkmate::assert_function(server, args = "id", nargs = 1) + structure( + list(ui = ui, server = server), + class = "teal_data_module" + ) +} diff --git a/R/utils.R b/R/utils.R index a4a31f2a5d..1da63cf8bf 100644 --- a/R/utils.R +++ b/R/utils.R @@ -140,3 +140,78 @@ resolve_modules_datanames <- function(modules, datanames, join_keys) { modules } } + +#' Check `datanames` in modules +#' +#' This function ensures specified `datanames` in modules match those in the data object, +#' returning error messages or `TRUE` for successful validation. +#' +#' @param modules (`teal_modules`) object +#' @param datanames (`character`) names of datasets available in the `data` object +#' +#' @return A `character(1)` containing error message or `TRUE` if validation passes. +#' @keywords internal +check_modules_datanames <- function(modules, datanames) { + checkmate::assert_class(modules, "teal_modules") + checkmate::assert_character(datanames) + + recursive_check_datanames <- function(modules, datanames) { + # check teal_modules against datanames + if (inherits(modules, "teal_modules")) { + sapply(modules$children, function(module) recursive_check_datanames(module, datanames = datanames)) + } else { + extra_datanames <- setdiff(modules$datanames, c("all", datanames)) + if (length(extra_datanames)) { + sprintf( + "- Module '%s' uses datanames not available in 'data': (%s) not in (%s)", + modules$label, + toString(dQuote(extra_datanames, q = FALSE)), + toString(dQuote(datanames, q = FALSE)) + ) + } + } + } + check_datanames <- unlist(recursive_check_datanames(modules, datanames)) + if (length(check_datanames)) { + paste(check_datanames, collapse = "\n") + } else { + TRUE + } +} + +#' Check `datanames` in filters +#' +#' This function checks whether `datanames` in filters correspond to those in `data`, +#' returning character vector with error messages or TRUE if all checks pass. +#' +#' @param filters (`teal_slices`) object +#' @param datanames (`character`) names of datasets available in the `data` object +#' +#' @return A `character(1)` containing error message or TRUE if validation passes. +#' @keywords internal +check_filter_datanames <- function(filters, datanames) { + checkmate::assert_class(filters, "teal_slices") + checkmate::assert_character(datanames) + + # check teal_slices against datanames + out <- unlist(sapply( + filters, function(filter) { + dataname <- shiny::isolate(filter$dataname) + if (!dataname %in% datanames) { + sprintf( + "- Filter '%s' refers to dataname not available in 'data':\n %s not in (%s)", + shiny::isolate(filter$id), + dQuote(dataname, q = FALSE), + toString(dQuote(datanames, q = FALSE)) + ) + } + } + )) + + + if (length(out)) { + paste(out, collapse = "\n") + } else { + TRUE + } +} diff --git a/_pkgdown.yml b/_pkgdown.yml index 01209c9ae4..46955166dc 100644 --- a/_pkgdown.yml +++ b/_pkgdown.yml @@ -62,6 +62,7 @@ articles: - including-general-data-in-teal - including-mae-data-in-teal - preprocessing-data + - data-as-shiny-module - title: Extending teal navbar: Extending teal contents: @@ -92,6 +93,7 @@ reference: - title: Core `teal` Functions desc: Main functions needed to build a `teal` app contents: + - teal_data_module - init - module - modules diff --git a/man/check_filter_datanames.Rd b/man/check_filter_datanames.Rd new file mode 100644 index 0000000000..2827604a27 --- /dev/null +++ b/man/check_filter_datanames.Rd @@ -0,0 +1,21 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/utils.R +\name{check_filter_datanames} +\alias{check_filter_datanames} +\title{Check \code{datanames} in filters} +\usage{ +check_filter_datanames(filters, datanames) +} +\arguments{ +\item{filters}{(\code{teal_slices}) object} + +\item{datanames}{(\code{character}) names of datasets available in the \code{data} object} +} +\value{ +A \code{character(1)} containing error message or TRUE if validation passes. +} +\description{ +This function checks whether \code{datanames} in filters correspond to those in \code{data}, +returning character vector with error messages or TRUE if all checks pass. +} +\keyword{internal} diff --git a/man/check_modules_datanames.Rd b/man/check_modules_datanames.Rd new file mode 100644 index 0000000000..7fef35aec0 --- /dev/null +++ b/man/check_modules_datanames.Rd @@ -0,0 +1,21 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/utils.R +\name{check_modules_datanames} +\alias{check_modules_datanames} +\title{Check \code{datanames} in modules} +\usage{ +check_modules_datanames(modules, datanames) +} +\arguments{ +\item{modules}{(\code{teal_modules}) object} + +\item{datanames}{(\code{character}) names of datasets available in the \code{data} object} +} +\value{ +A \code{character(1)} containing error message or \code{TRUE} if validation passes. +} +\description{ +This function ensures specified \code{datanames} in modules match those in the data object, +returning error messages or \code{TRUE} for successful validation. +} +\keyword{internal} diff --git a/man/init.Rd b/man/init.Rd index 07bbec2deb..f03437f1b3 100644 --- a/man/init.Rd +++ b/man/init.Rd @@ -16,10 +16,11 @@ init( } \arguments{ \item{data}{(\code{TealData} or \code{TealDataset} or \code{TealDatasetConnector} or \code{list} or \code{data.frame} -or \code{MultiAssayExperiment}, \code{teal_data})\cr +or \code{MultiAssayExperiment}, \code{teal_data}, \code{teal_data_module})\cr \code{R6} object as returned by \code{\link[teal.data:cdisc_data]{teal.data::cdisc_data()}}, \code{\link[teal.data:teal_data]{teal.data::teal_data()}}, \code{\link[teal.data:cdisc_dataset]{teal.data::cdisc_dataset()}}, \code{\link[teal.data:dataset]{teal.data::dataset()}}, \code{\link[teal.data:dataset_connector]{teal.data::dataset_connector()}} or -\code{\link[teal.data:cdisc_dataset_connector]{teal.data::cdisc_dataset_connector()}} or a single \code{data.frame} or a \code{MultiAssayExperiment} +\code{\link[teal.data:cdisc_dataset_connector]{teal.data::cdisc_dataset_connector()}} or \code{\link[=teal_data_module]{teal_data_module()}} or a single \code{data.frame} or +a \code{MultiAssayExperiment} or a list of the previous objects or function returning a named list. NOTE: teal does not guarantee reproducibility of the code when names of the list elements do not match the original object names. To ensure reproducibility please use \code{\link[teal.data:teal_data]{teal.data::teal_data()}} diff --git a/man/landing_popup_module.Rd b/man/landing_popup_module.Rd index 92531ca334..1c8ead4e86 100644 --- a/man/landing_popup_module.Rd +++ b/man/landing_popup_module.Rd @@ -32,7 +32,7 @@ The dialog blocks the access to the application and must be closed with a button } \examples{ app1 <- teal::init( - data = teal.data::dataset("iris", iris), + data = teal_data(iris = iris), modules = teal::modules( teal::landing_popup_module( content = "A place for the welcome message or a disclaimer statement.", @@ -46,7 +46,7 @@ if (interactive()) { } app2 <- teal::init( - data = teal.data::dataset("iris", iris), + data = teal_data(iris = iris), modules = teal::modules( teal::landing_popup_module( title = "Welcome", diff --git a/man/module_teal.Rd b/man/module_teal.Rd index 10c1c8654c..50524636c4 100644 --- a/man/module_teal.Rd +++ b/man/module_teal.Rd @@ -14,7 +14,7 @@ ui_teal( footer = tags$p("") ) -srv_teal(id, modules, raw_data, filter = teal_slices()) +srv_teal(id, modules, teal_data_rv, filter = teal_slices()) } \arguments{ \item{id}{(\code{character(1)})\cr @@ -35,7 +35,7 @@ argument) will be placed in the app's \code{ui} function so code which needs to \item{footer}{(\code{shiny.tag} or \code{character})\cr the footer of the app} -\item{raw_data}{(\code{reactive})\cr +\item{teal_data_rv}{(\code{reactive})\cr returns the \code{teal_data}, only evaluated once, \code{NULL} value is ignored} } \value{ @@ -57,7 +57,7 @@ The splash screen functionality can also be used for non-delayed data which takes time to load into memory, avoiding Shiny session timeouts. -Server evaluates the \code{raw_data} (delayed data mechanism) and creates the +Server evaluates the \code{teal_data_rv} (delayed data mechanism) and creates the \code{datasets} object that is shared across modules. Once it is ready and non-\code{NULL}, the splash screen is replaced by the main teal UI that depends on the data. @@ -69,13 +69,13 @@ It is written as a Shiny module so it can be added into other apps as well. } \examples{ mods <- teal:::example_modules() -raw_data <- reactive(teal:::example_cdisc_data()) +teal_data_rv <- reactive(teal:::example_cdisc_data()) app <- shinyApp( ui = function() { teal:::ui_teal("dummy") }, server = function(input, output, session) { - active_module <- teal:::srv_teal(id = "dummy", modules = mods, raw_data = raw_data) + active_module <- teal:::srv_teal(id = "dummy", modules = mods, teal_data_rv = teal_data_rv) } ) if (interactive()) { diff --git a/man/snapshot_manager_module.Rd b/man/snapshot_manager_module.Rd index af51d469be..66e5b8a94d 100644 --- a/man/snapshot_manager_module.Rd +++ b/man/snapshot_manager_module.Rd @@ -93,7 +93,8 @@ which is disassembled for storage and used directly for restoring app state. \section{Transferring snapshots}{ -Snapshots uploaded from disk should only be used in the same application they come from. +Snapshots uploaded from disk should only be used in the same application they come from, +\emph{i.e.} an application that uses the same data and the same modules. To ensure this is the case, \code{init} stamps \code{teal_slices} with an app id that is stored in the \code{app_id} attribute of a \code{teal_slices} object. When a snapshot is restored from file, its \code{app_id} is compared to that of the current app state and only if the match is the snapshot admitted to the session. diff --git a/man/srv_teal_with_splash.Rd b/man/srv_teal_with_splash.Rd index 7ab0c6efea..9b1d4312d1 100644 --- a/man/srv_teal_with_splash.Rd +++ b/man/srv_teal_with_splash.Rd @@ -15,10 +15,11 @@ See the vignette for an example. However, \code{\link[=ui_teal_with_splash]{ui_t is then preferred to this function.} \item{data}{(\code{TealData} or \code{TealDataset} or \code{TealDatasetConnector} or \code{list} or \code{data.frame} -or \code{MultiAssayExperiment}, \code{teal_data})\cr +or \code{MultiAssayExperiment}, \code{teal_data}, \code{teal_data_module})\cr \code{R6} object as returned by \code{\link[teal.data:cdisc_data]{teal.data::cdisc_data()}}, \code{\link[teal.data:teal_data]{teal.data::teal_data()}}, \code{\link[teal.data:cdisc_dataset]{teal.data::cdisc_dataset()}}, \code{\link[teal.data:dataset]{teal.data::dataset()}}, \code{\link[teal.data:dataset_connector]{teal.data::dataset_connector()}} or -\code{\link[teal.data:cdisc_dataset_connector]{teal.data::cdisc_dataset_connector()}} or a single \code{data.frame} or a \code{MultiAssayExperiment} +\code{\link[teal.data:cdisc_dataset_connector]{teal.data::cdisc_dataset_connector()}} or \code{\link[=teal_data_module]{teal_data_module()}} or a single \code{data.frame} or +a \code{MultiAssayExperiment} or a list of the previous objects or function returning a named list. NOTE: teal does not guarantee reproducibility of the code when names of the list elements do not match the original object names. To ensure reproducibility please use \code{\link[teal.data:teal_data]{teal.data::teal_data()}} diff --git a/man/teal_data_module.Rd b/man/teal_data_module.Rd new file mode 100644 index 0000000000..3eb672bcda --- /dev/null +++ b/man/teal_data_module.Rd @@ -0,0 +1,55 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/teal_data_module.R +\name{teal_data_module} +\alias{teal_data_module} +\title{Data module for \code{teal} applications} +\usage{ +teal_data_module(ui, server) +} +\arguments{ +\item{ui}{(\verb{function(id)})\cr +\code{shiny} module \code{ui} function; must only take \code{id} argument} + +\item{server}{(\verb{function(id)})\cr +\code{shiny} module \code{ui} function; must only take \code{id} argument; +must return reactive expression containing \code{teal_data} object} +} +\value{ +Object of class \code{teal_data_module}. +} +\description{ +Creates \code{teal_data_module} object - a \code{shiny} module to supply or modify data in a \code{teal} application. +} +\details{ +This function creates a \code{shiny} module that allows for running data pre-processing code after the app starts. +The body of the server function will be run in the app rather than in the global environment. +This means it will be run every time the app starts, so use sparingly. + +Pass this module instead of a \code{teal_data} object in a call to \code{\link[=init]{init()}}. + +See vignette \code{vignette("data-as-shiny-module", package = "teal")} for more details. +} +\examples{ +data <- teal_data_module( + ui = function(id) { + ns <- NS(id) + actionButton(ns("submit"), label = "Load data") + }, + server = function(id) { + moduleServer(id, function(input, output, session) { + eventReactive(input$submit, { + data <- within( + teal_data(), + { + dataset1 <- iris + dataset2 <- mtcars + } + ) + datanames(data) <- c("iris", "mtcars") + + data + }) + }) + } +) +} diff --git a/man/ui_teal_with_splash.Rd b/man/ui_teal_with_splash.Rd index 0ece4d3027..29396ed74c 100644 --- a/man/ui_teal_with_splash.Rd +++ b/man/ui_teal_with_splash.Rd @@ -17,10 +17,11 @@ ui_teal_with_splash( module id} \item{data}{(\code{TealData} or \code{TealDataset} or \code{TealDatasetConnector} or \code{list} or \code{data.frame} -or \code{MultiAssayExperiment}, \code{teal_data})\cr +or \code{MultiAssayExperiment}, \code{teal_data}, \code{teal_data_module})\cr \code{R6} object as returned by \code{\link[teal.data:cdisc_data]{teal.data::cdisc_data()}}, \code{\link[teal.data:teal_data]{teal.data::teal_data()}}, \code{\link[teal.data:cdisc_dataset]{teal.data::cdisc_dataset()}}, \code{\link[teal.data:dataset]{teal.data::dataset()}}, \code{\link[teal.data:dataset_connector]{teal.data::dataset_connector()}} or -\code{\link[teal.data:cdisc_dataset_connector]{teal.data::cdisc_dataset_connector()}} or a single \code{data.frame} or a \code{MultiAssayExperiment} +\code{\link[teal.data:cdisc_dataset_connector]{teal.data::cdisc_dataset_connector()}} or \code{\link[=teal_data_module]{teal_data_module()}} or a single \code{data.frame} or +a \code{MultiAssayExperiment} or a list of the previous objects or function returning a named list. NOTE: teal does not guarantee reproducibility of the code when names of the list elements do not match the original object names. To ensure reproducibility please use \code{\link[teal.data:teal_data]{teal.data::teal_data()}} diff --git a/tests/testthat/test-init.R b/tests/testthat/test-init.R index a34e460472..b19df7aad7 100644 --- a/tests/testthat/test-init.R +++ b/tests/testthat/test-init.R @@ -1,5 +1,5 @@ testthat::test_that("init data accepts TealData object", { - testthat::expect_no_error( + lifecycle::expect_deprecated( init( data = teal.data::cdisc_data( teal.data::cdisc_dataset( @@ -23,7 +23,7 @@ testthat::test_that("init data accepts teal_data object", { ) }) -testthat::test_that("init data throws an error with input other than TealData, teal_data and ddl", { +testthat::test_that("init data throws an error with input other than TealData, teal_data and list(ui, server)", { character_vector <- c("a", "b", "c") numeric_vector <- c(1, 2, 3) matrix_d <- as.matrix(c(1, 2, 3)) @@ -140,6 +140,15 @@ testthat::test_that("init data accepts a list of TealDatasetConnector object", { testthat::expect_no_error(init(data = dsc1, modules = modules(example_module()))) }) +testthat::test_that("init data accepts teal_data_module", { + testthat::expect_no_error( + init( + data = teal_data_module(ui = function(id) div(), server = function(id) NULL), + modules = modules(teal:::example_module()) + ) + ) +}) + testthat::test_that("init modules accepts a teal_modules object", { mods <- modules(example_module(), example_module()) testthat::expect_no_error(init(data = iris, modules = mods)) @@ -155,19 +164,42 @@ testthat::test_that("init modules accepts a teal_module object", { testthat::expect_no_error(init(data = iris, modules = mods)) }) -testthat::test_that("init filter accepts named list or `teal_slices`", { - fl <- list( - "iris" = list( - "Species" = list(selected = "setosa") - ) - ) +testthat::test_that("init filter accepts `teal_slices`", { fs <- teal.slice::teal_slices( teal.slice::teal_slice(dataname = "iris", varname = "species", selected = "setosa") ) - testthat::expect_no_error(init(data = list(iris), modules = modules(example_module()), filter = fl)) testthat::expect_no_error(init(data = list(iris), modules = modules(example_module()), filter = fs)) testthat::expect_error( init(data = list(iris), modules = modules(example_module()), filter = unclass(fs)), "Assertion failed" ) }) + +testthat::test_that("init throws when data has no datanames", { + testthat::expect_error( + init(data = teal_data(), modules = list(example_module())), + "has no datanames" + ) +}) + +testthat::test_that("init throws when incompatible module's datanames", { + msg <- "Module 'example teal module' uses datanames not available in 'data'" + testthat::expect_output( + testthat::expect_error( + init(data = teal_data(mtcars = mtcars), modules = list(example_module(datanames = "iris"))), + msg + ), + msg + ) +}) + +testthat::test_that("init throws when incompatible filter's datanames", { + testthat::expect_output( + init( + data = teal_data(mtcars = mtcars), + modules = modules(example_module()), + filter = teal_slices(teal_slice(dataname = "iris", varname = "Species")) + ), + "Filter 'iris Species' refers to dataname not available in 'data'" + ) +}) diff --git a/tests/testthat/test-module_nested_tabs.R b/tests/testthat/test-module_nested_tabs.R index 74aea0ecd8..f4479988b9 100644 --- a/tests/testthat/test-module_nested_tabs.R +++ b/tests/testthat/test-module_nested_tabs.R @@ -485,10 +485,10 @@ testthat::test_that("calculate_hashes takes a FilteredData and vector of datanam adtte <- data.frame(STUDYID = 1, USUBJID = 1, PARAMCD = 1) datasets <- teal.slice::init_filtered_data( - teal.data::cdisc_data( - teal.data::cdisc_dataset("ADSL", adsl), - teal.data::cdisc_dataset("ADAE", adae), - teal.data::cdisc_dataset("ADTTE", adtte) + list( + ADSL = list(dataset = adsl), + ADAE = list(dataset = adae), + ADTTE = list(dataset = adtte) ) ) @@ -501,10 +501,10 @@ testthat::test_that("calculate_hashes returns a named list", { adtte <- data.frame(STUDYID = 1, USUBJID = 1, PARAMCD = 1) datasets <- teal.slice::init_filtered_data( - teal.data::cdisc_data( - teal.data::cdisc_dataset("ADSL", adsl), - teal.data::cdisc_dataset("ADAE", adae), - teal.data::cdisc_dataset("ADTTE", adtte) + list( + ADSL = list(dataset = adsl), + ADAE = list(dataset = adae), + ADTTE = list(dataset = adtte) ) ) @@ -523,9 +523,7 @@ testthat::test_that("calculate_hashes returns a named list", { testthat::test_that("calculate_hashes returns the hash of the non Filtered dataset", { datasets <- teal.slice::init_filtered_data( - teal.data::teal_data( - teal.data::dataset("iris", iris) - ) + list(iris = list(dataset = iris)) ) fs <- teal.slice:::teal_slices( diff --git a/tests/testthat/test-module_teal.R b/tests/testthat/test-module_teal.R index ca3cc28d8f..0850a68aac 100644 --- a/tests/testthat/test-module_teal.R +++ b/tests/testthat/test-module_teal.R @@ -1,54 +1,51 @@ -data <- teal_data(iris1 = iris, mtcars1 = mtcars, code = "iris1 <- iris; mtcars1 <- mtcars") - -test_module1 <- example_module( - label = "iris_tab", - datanames = "iris1" -) -test_module2 <- example_module( - label = "mtcars_tab", - datanames = "mtcars1" -) - -testthat::test_that("srv_teal fails when raw_data is not reactive", { +testthat::test_that("srv_teal fails when teal_data_rv is not reactive", { testthat::expect_error( shiny::testServer( app = srv_teal, args = list( id = "test", - raw_data = data, - modules = modules(test_module1) + teal_data_rv = teal_data(iris = iris), + modules = modules(example_module()) ), expr = NULL ), - regexp = "is.reactive\\(raw_data\\)" + regexp = "is.reactive\\(teal_data_rv\\)" ) }) -testthat::test_that("srv_teal initializes the data when raw_data changes", { +testthat::test_that("srv_teal when teal_data_rv changes, datasets_reactive is initialized as list of FilteredData", { + data <- teal_data(iris1 = iris, mtcars1 = mtcars) shiny::testServer( app = srv_teal, args = list( id = "test", - raw_data = reactiveVal(NULL), - modules = modules(test_module1) + teal_data_rv = reactiveVal(NULL), + modules = modules( + example_module(label = "iris_tab"), + example_module(label = "mtcars_tab") + ) ), expr = { - raw_data(data) - testthat::expect_named(datasets_reactive(), "iris_tab") + teal_data_rv(data) + checkmate::expect_list(datasets_reactive(), types = "FilteredData") } ) }) -testthat::test_that("srv_teal initialized data list structure reflects modules", { +testthat::test_that("srv_teal initialized datasets_reactive (list) reflects modules structure", { + data <- teal_data(iris1 = iris, mtcars1 = mtcars) shiny::testServer( app = srv_teal, args = list( id = "test", - raw_data = reactiveVal(data), - modules = modules(test_module1, modules(label = "tab", test_module1, test_module2)) + teal_data_rv = reactiveVal(data), + modules = modules( + example_module("iris_tab"), + modules(label = "tab", example_module("iris_tab"), example_module("mtcars_tab")) + ) ), expr = { - raw_data(data) + teal_data_rv(data) testthat::expect_named(datasets_reactive(), c("iris_tab", "tab")) testthat::expect_named(datasets_reactive()$tab, c("iris_tab", "mtcars_tab")) } @@ -56,16 +53,20 @@ testthat::test_that("srv_teal initialized data list structure reflects modules", }) testthat::test_that("srv_teal initialized data containing same FilteredData when the filter is global", { + data <- teal_data(iris1 = iris, mtcars1 = mtcars) shiny::testServer( app = srv_teal, args = list( id = "test", - raw_data = reactiveVal(data), - modules = modules(test_module1, modules(label = "tab", test_module1, test_module2)), + teal_data_rv = reactiveVal(data), + modules = modules( + example_module("iris_tab"), + modules(label = "tab", example_module("iris_tab"), example_module("mtcars_tab")) + ), filter = teal_slices(module_specific = FALSE) ), expr = { - raw_data(data) + teal_data_rv(data) unlisted_fd <- unlist(datasets_reactive(), use.names = FALSE) testthat::expect_identical(unlisted_fd[[1]], unlisted_fd[[2]]) testthat::expect_identical(unlisted_fd[[2]], unlisted_fd[[3]]) @@ -74,16 +75,20 @@ testthat::test_that("srv_teal initialized data containing same FilteredData when }) testthat::test_that("srv_teal initialized data containing different FilteredData when the filter is module_specific", { + data <- teal_data(iris1 = iris, mtcars1 = mtcars) shiny::testServer( app = srv_teal, args = list( id = "test", - raw_data = reactiveVal(data), - modules = modules(test_module1, modules(label = "tab", test_module1, test_module2)), + teal_data_rv = reactiveVal(data), + modules = modules( + example_module("iris_tab"), + modules(label = "tab", example_module("iris_tab"), example_module("mtcars_tab")) + ), filter = teal_slices(module_specific = TRUE) ), expr = { - raw_data(data) + teal_data_rv(data) unlisted_fd <- unlist(datasets_reactive(), use.names = FALSE) testthat::expect_false(identical(unlisted_fd[[1]], unlisted_fd[[2]])) testthat::expect_false(identical(unlisted_fd[[2]], unlisted_fd[[3]])) diff --git a/tests/testthat/test-module_teal_with_splash.R b/tests/testthat/test-module_teal_with_splash.R index 039ea01011..577e997582 100644 --- a/tests/testthat/test-module_teal_with_splash.R +++ b/tests/testthat/test-module_teal_with_splash.R @@ -1,27 +1,134 @@ -iris_ds <- teal.data::dataset(dataname = "iris", x = head(iris)) -mtcars_ds <- teal.data::dataset(dataname = "mtcars", x = head(mtcars)) -data <- teal_data(iris_ds, mtcars_ds) +testthat::test_that("srv_teal_with_splash data accepts a teal_data_module", { + testthat::expect_no_error( + shiny::testServer( + app = srv_teal_with_splash, + args = list( + id = "id", + data = teal_data_module(ui = function(id) div(), server = function(id) reactive(NULL)), + modules = modules(example_module()) + ), + expr = {} + ) + ) +}) -test_module1 <- example_module( - label = "iris_tab", - datanames = "iris" -) +testthat::test_that("srv_teal_with_splash throws when teal_data_module doesn't return reactive", { + testthat::expect_error( + shiny::testServer( + app = srv_teal_with_splash, + args = list( + id = "id", + data = teal_data_module(ui = function(id) div(), server = function(id) NULL), + modules = modules(example_module()) + ), + expr = {} + ), + "The `teal_data_module` must return a reactive expression." + ) +}) + +testthat::test_that("srv_teal_with_splash teal_data_rv evaluates the server of teal_data_module", { + shiny::testServer( + app = srv_teal_with_splash, + args = list( + id = "test", + data = teal_data_module(ui = function(id) div(), server = function(id) reactive("whatever")), + modules = modules(example_module()) + ), + expr = { + testthat::expect_is(teal_data_rv, "reactive") + testthat::expect_identical(teal_data_rv(), "whatever") + } + ) +}) -testthat::test_that("srv_teal_with_splash creates reactiveVal returning teal_data", { +testthat::test_that("srv_teal_with_splash passes teal_data to reactive", { shiny::testServer( app = srv_teal_with_splash, args = list( id = "test", - data = data, - modules = modules(test_module1) + data = teal_data(iris = iris), + modules = modules(example_module()) ), expr = { - testthat::expect_is(raw_data, "reactiveVal") - testthat::expect_s4_class(raw_data(), "teal_data") + testthat::expect_is(teal_data_rv_validate, "reactive") + testthat::expect_s4_class(teal_data_rv_validate(), "teal_data") } ) }) +testthat::test_that("srv_teal_with_splash throws when datanames are empty", { + shiny::testServer( + app = srv_teal_with_splash, + args = list( + id = "test", + data = teal_data(), + modules = modules(example_module()) + ), + expr = { + testthat::expect_error(teal_data_rv_validate(), "Data has no datanames") + } + ) +}) + +testthat::test_that("srv_teal_with_splash teal_data_rv_validate throws when teal_data_module returns error", { + shiny::testServer( + app = srv_teal_with_splash, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) div(), + server = function(id) reactive(stop("this error")) + ), + modules = modules(example_module()) + ), + expr = { + testthat::expect_is(teal_data_rv_validate, "reactive") + testthat::expect_error(teal_data_rv_validate(), "this error") + } + ) +}) + +testthat::test_that("srv_teal_with_splash teal_data_rv_validate throws then qenv.error occurs", { + shiny::testServer( + app = srv_teal_with_splash, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) div(), + server = function(id) reactive(teal_data() |> within(stop("not good"))) + ), + modules = modules(example_module()) + ), + expr = { + testthat::expect_is(teal_data_rv_validate, "reactive") + testthat::expect_error(teal_data_rv_validate(), "not good") + } + ) +}) + +testthat::test_that( + "srv_teal_with_splash teal_data_rv_validate throws when teal_data_module doesn't return teal_data", + { + shiny::testServer( + app = srv_teal_with_splash, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) div(), + server = function(id) reactive(data.frame()) + ), + modules = modules(example_module()) + ), + expr = { + testthat::expect_is(teal_data_rv_validate, "reactive") + testthat::expect_error(teal_data_rv_validate(), "did not return `teal_data`") + } + ) + } +) + + testthat::test_that("srv_teal_with_splash creates raw_data based on DDL returns NULL before loading", { x <- dataset_connector(dataname = "test_dataset", pull_callable = callable_code("iris")) delayed_data <- teal_data(x) @@ -30,7 +137,7 @@ testthat::test_that("srv_teal_with_splash creates raw_data based on DDL returns args = list( id = "test", data = delayed_data, - modules = modules(test_module1) + modules = modules(example_module()) ), expr = testthat::expect_null(raw_data()) ) @@ -45,7 +152,7 @@ testthat::test_that("srv_teal_with_splash creates raw_data based on DDL returns args = list( id = "test", data = delayed_data, - modules = modules(test_module1) + modules = modules(example_module()) ), expr = { testthat::expect_null(raw_data()) @@ -56,13 +163,47 @@ testthat::test_that("srv_teal_with_splash creates raw_data based on DDL returns ) }) +testthat::test_that("srv_teal_with_splash teal_data_rv_validate throws when incompatible module's datanames", { + shiny::testServer( + app = srv_teal_with_splash, + args = list( + id = "test", + data = teal_data(mtcars = mtcars), + modules = modules(example_module(datanames = "iris")) + ), + expr = { + testthat::expect_is(teal_data_rv_validate, "reactive") + testthat::expect_error( + teal_data_rv_validate(), + "Module 'example teal module' uses datanames not available in 'data'" + ) + } + ) +}) + +testthat::test_that("srv_teal_with_splash teal_data_rv_validate returns teal_data if incompatible filter's datanames", { + shiny::testServer( + app = srv_teal_with_splash, + args = list( + id = "test", + data = teal_data(mtcars = mtcars), + modules = modules(example_module(datanames = "mtcars")), + filter = teal_slices(teal_slice(dataname = "iris", varname = "Species")) + ), + expr = { + testthat::expect_is(teal_data_rv_validate, "reactive") + testthat::expect_s4_class(teal_data_rv_validate(), "teal_data") + } + ) +}) + testthat::test_that("srv_teal_with_splash gets observe event from srv_teal", { shiny::testServer( app = srv_teal_with_splash, args = list( id = "test", - data = data, - modules = modules(test_module1) + data = teal_data(), + modules = modules(example_module()) ), expr = { testthat::expect_is(res, "Observer") diff --git a/tests/testthat/test-rcode_utils.R b/tests/testthat/test-rcode_utils.R index c3f29be1b6..1add79e18d 100644 --- a/tests/testthat/test-rcode_utils.R +++ b/tests/testthat/test-rcode_utils.R @@ -45,24 +45,12 @@ testthat::test_that("get_rcode_libraries returns current session packages", { }) testthat::test_that("get_datasets_code returns code only for specified datanames", { + # todo: need to use code dependency? Or test it later via public functions/modules datasets <- teal.slice::init_filtered_data( - teal.data::teal_data( - teal.data::dataset("IRIS", x = iris, code = "IRIS <- iris"), - teal.data::dataset("MTCARS", x = mtcars, code = "MTCARS <- mtcars") + list( + IRIS = list(dataset = iris), + MTCARS = list(dataset = mtcars) ) ) - - hashes <- calculate_hashes(datasets$datanames(), datasets) - testthat::expect_true( - !grepl( - "mtcars", - paste(get_datasets_code(datasets = datasets, dataname = "IRIS", hashes = hashes), collapse = "\n"), - ignore.case = TRUE - ) && - grepl( - "iris", - paste(get_datasets_code(datasets = datasets, dataname = "IRIS", hashes = hashes), collapse = "\n"), - ignore.case = TRUE - ) - ) + testthat::expect_true(TRUE) }) diff --git a/tests/testthat/test-teal_data_module.R b/tests/testthat/test-teal_data_module.R new file mode 100644 index 0000000000..93d20d203b --- /dev/null +++ b/tests/testthat/test-teal_data_module.R @@ -0,0 +1,20 @@ +testthat::test_that("teal_data_module returns teal_data_module", { + testthat::expect_s3_class( + teal_data_module(ui = function(id) div(), server = function(id) NULL), + "teal_data_module" + ) +}) + +testthat::test_that("teal_data_module throws when ui has other formals than id only", { + testthat::expect_error( + teal_data_module(ui = function(id, x) div(), server = function(id) NULL), + "Must have exactly 1 formal arguments" + ) +}) + +testthat::test_that("teal_data_module throws when server has other formals than id only", { + testthat::expect_error( + teal_data_module(ui = function(id) div(), server = function(id, x) NULL), + "Must have exactly 1 formal arguments" + ) +}) diff --git a/vignettes/adding-support-for-reporting.Rmd b/vignettes/adding-support-for-reporting.Rmd index b28fbd32f4..6d4a8001e9 100644 --- a/vignettes/adding-support-for-reporting.Rmd +++ b/vignettes/adding-support-for-reporting.Rmd @@ -322,7 +322,7 @@ app <- init( example_reporter_module(label = "with Reporter"), example_module(label = "without Reporter") ), - filter = list(AIR = list(Month = c(5, 5))), + filter = teal_slices(teal_slice(dataname = "AIR", varname = "Month", selected = c(5, 5))), header = "Example teal app with reporter" ) diff --git a/vignettes/data-as-shiny-module.Rmd b/vignettes/data-as-shiny-module.Rmd new file mode 100644 index 0000000000..66ae5cb9e1 --- /dev/null +++ b/vignettes/data-as-shiny-module.Rmd @@ -0,0 +1,126 @@ +--- +title: "Data as shiny Module" +author: "NEST CoreDev" +output: + rmarkdown::html_vignette: + toc: true +vignette: > + %\VignetteIndexEntry{Data as shiny Module} + %\VignetteEngine{knitr::rmarkdown} + %\VignetteEncoding{UTF-8} +--- + +## Introduction + +Proper functioning of any `teal` application requires presence of a `teal_data` object. +Typically, a `teal_data` object created in the global environment will be passed to the `data` argument in `init`. +This `teal_data` object should contain all elements necessary for successful execution of the application's modules. +In some scenarios, however, application developers may opt to postpone some data operations until the application run time. +This can be done by passing a special _`shiny` module_ to the `data` argument. +The `teal_data_module` function is used to build such a module from the following components: + +- a `ui` function; accepts only one argument, `id`; defines user interface elements for the data module +- a `server` function: accepts only one argument, `id`; defines server logic for the data module, including data creation; must return a reactive expression containing a `teal_data` object + +`teal` will run this module when the application starts and the resulting `teal_data` object that will be used throughout all `teal` (analytic) modules. + +## Creating Data In-App + +One case for postponing data operations are data sets that are dynamic, frequently updated. +Such data cannot be created once and kept in the global environment. +Using `teal_data_module` allows to create a data set from scratch every time the user starts the application. + +```{r, message = FALSE, warning = FALSE} +library(teal) +``` + + +```{r} +data_mod <- teal_data_module( + ui = function(id) div(), + server = function(id) { + moduleServer(id, function(input, output, session) { + reactive({ + data <- within( + teal_data(), + { + dataset1 <- iris + dataset2 <- mtcars + } + ) + datanames(data) <- c("dataset1", "dataset2") + data + }) + }) + } +) + + +app <- init( + data = data_mod, + module = example_module() +) + +if (interactive()) { + shinyApp(app$ui, app$server) +} +``` + + +## Modification Data In-App + +Another reason to postpone data operations is to allow the application user to act the preprocessing stage. +An initial, constant form of the data can be created in the global environment and then modified once the app starts. + +The following example illustrates how `teal_data_module` can be utilized to subset data based on the user inputs: + +```{r} +data <- within(teal_data(), { + dataset1 <- iris + dataset2 <- mtcars +}) +datanames(data) <- c("dataset1", "dataset2") + +data_mod <- teal_data_module( + ui = function(id) { + ns <- NS(id) + div( + selectInput(ns("species"), "Select species to filter", + choices = unique(iris$Species), multiple = TRUE + ), + actionButton(ns("submit"), "Submit") + ) + }, + server = function(id) { + moduleServer(id, function(input, output, session) { + eventReactive(input$submit, { + data_modified <- within( + data, + dataset1 <- subset(dataset1, Species %in% selected), + selected = input$species + ) + data_modified + }) + }) + } +) + +app <- init( + data = data_mod, + module = example_module() +) + +if (interactive()) { + shinyApp(app$ui, app$server) +} +``` + +_See `?within.qenv` for a detailed explanation of how to use the `within` method._ + +Note that running preprocessing code in a module as opposed to the global environment will increase app loading times. +It is recommended to keep the constant code in the global environment and to move only the dynamic parts to a data module. + +###### WARNING + +When using `teal_data_module` to modify a pre-existing `teal_data` object it is crucial that the server function and the data object are defined in the same environment as otherwise the server function will not be able to access the data object. +This means server functions defined in packages cannot be used.