From 80f536188c2e8664164451827d93fe9bfb43337f Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Fri, 26 Jan 2024 17:14:58 +0100 Subject: [PATCH 01/23] run examples using shinytest2 --- DESCRIPTION | 2 ++ tests/testthat/test-examples.R | 20 ++++++++++++++++++++ 2 files changed, 22 insertions(+) create mode 100644 tests/testthat/test-examples.R diff --git a/DESCRIPTION b/DESCRIPTION index 01eff7e069..ad73048b6e 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -69,6 +69,8 @@ Suggests: knitr, lubridate, nestcolor (>= 0.1.0), + pkgload, + shinytest2, testthat (>= 3.1.5) VignetteBuilder: knitr diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R new file mode 100644 index 0000000000..3a2a2377b7 --- /dev/null +++ b/tests/testthat/test-examples.R @@ -0,0 +1,20 @@ +files <- devtools:::rd_files() + +for (i in files) { + with_mocked_bindings( + test_that( + paste0("example-", i),{ + skip_on_cran() + expect_no_error( + pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE) + ) + } + ), + runApp = \(appDir, ...) { + cat("hello from mocked runApp\n") + app <- shinytest2::AppDriver$new(appDir) + app$stop() + }, + .package = "shiny" + ) +} \ No newline at end of file From 7edbd37a5cd7f8be88240d577afa9eb3ba4932aa Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Fri, 26 Jan 2024 17:49:41 +0100 Subject: [PATCH 02/23] temp silent certain warning --- tests/testthat/test-examples.R | 26 ++++++++++++++++++++------ 1 file changed, 20 insertions(+), 6 deletions(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index 3a2a2377b7..def1710ba3 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -1,14 +1,28 @@ files <- devtools:::rd_files() +# this is temporary +# every app tested with shinytest2 throws a warning because of https://github.com/insightsengineering/teal.code/issues/194 +# silent certain warnings +suppress_warnings <- function(expr, pattern, ...) { + withCallingHandlers(expr, warning = function(w) { + if (grepl(pattern, conditionMessage(w))) { + invokeRestart("muffleWarning") + } + }) +} + for (i in files) { with_mocked_bindings( test_that( - paste0("example-", i),{ - skip_on_cran() - expect_no_error( - pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE) - ) - } + paste0("example-", basename(i)),{ + skip_on_cran() + expect_no_error( + suppress_warnings( # temporary (see above) + pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE), + "may not be available when loading" + ) + ) + } ), runApp = \(appDir, ...) { cat("hello from mocked runApp\n") From 6cfba95fff662ec432d25f94d0b71d3c52a27204 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Mon, 11 Mar 2024 11:34:10 +0100 Subject: [PATCH 03/23] symlink man dir; cleanups --- tests/testthat/man | 1 + tests/testthat/test-examples.R | 34 +++++++++++++++++++--------------- 2 files changed, 20 insertions(+), 15 deletions(-) create mode 120000 tests/testthat/man diff --git a/tests/testthat/man b/tests/testthat/man new file mode 120000 index 0000000000..ee201c1931 --- /dev/null +++ b/tests/testthat/man @@ -0,0 +1 @@ +../../man \ No newline at end of file diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index def1710ba3..a92583caa1 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -1,9 +1,12 @@ -files <- devtools:::rd_files() +# this test requires a `man` directory in the `tests/testthat` directory +# (presumably symlinked to the package root `man` directory to avoid duplication) +# this also requires `devtools::document()` to be run before running the tests -# this is temporary -# every app tested with shinytest2 throws a warning because of https://github.com/insightsengineering/teal.code/issues/194 -# silent certain warnings -suppress_warnings <- function(expr, pattern, ...) { +rd_files <- function() { + list.files(testthat::test_path("man"), pattern = "\\.[Rr]d$", full.names = TRUE) +} + +suppress_warnings <- function(expr, pattern = "*", ...) { withCallingHandlers(expr, warning = function(w) { if (grepl(pattern, conditionMessage(w))) { invokeRestart("muffleWarning") @@ -11,24 +14,25 @@ suppress_warnings <- function(expr, pattern, ...) { }) } -for (i in files) { - with_mocked_bindings( - test_that( - paste0("example-", basename(i)),{ - skip_on_cran() - expect_no_error( - suppress_warnings( # temporary (see above) +for (i in rd_files()) { + testthat::with_mocked_bindings( + testthat::test_that( + paste0("example-", basename(i)), + { + testthat::skip_on_cran() + testthat::expect_no_error( + # surpress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 + suppress_warnings( pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE), "may not be available when loading" ) ) } ), - runApp = \(appDir, ...) { - cat("hello from mocked runApp\n") + runApp = function(appDir, ...) { # nolint object_name_linter. app <- shinytest2::AppDriver$new(appDir) app$stop() }, .package = "shiny" ) -} \ No newline at end of file +} From 4279a72b0017d9939f1e4d8c9cf2b5bea03fa627 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Thu, 21 Mar 2024 12:25:00 +0100 Subject: [PATCH 04/23] quiet=FALSE; better error capture --- tests/testthat/test-examples.R | 24 ++++++++++++++++++------ 1 file changed, 18 insertions(+), 6 deletions(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index a92583caa1..edd433e77d 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -21,17 +21,29 @@ for (i in rd_files()) { { testthat::skip_on_cran() testthat::expect_no_error( - # surpress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 - suppress_warnings( - pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE), - "may not be available when loading" + capture.output( + # surpress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 + suppress_warnings( + # quiet argument must be FALSE - otherwise the shiny apps are not invoked + pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = FALSE), + "may not be available when loading" + ) ) ) } ), runApp = function(appDir, ...) { # nolint object_name_linter. - app <- shinytest2::AppDriver$new(appDir) - app$stop() + app_driver <- shinytest2::AppDriver$new(appDir) + on.exit(app_driver$stop) + # shinytest2 will capture app crash but actually teal continues on error inside the module + # we need to use a different way to check if there are errors + if (!is.null(app_driver$get_html(".shiny-output-error:not(.shiny-output-error-validation)"))) { + stop("module error is observed") + } + # validation errors from shinyvalidate - added by default to assure the examples are "clean" + if (!is.null(app_driver$get_html(".shiny-input-container.has-error"))) { + stop("shinyvalidate error is observed") + } }, .package = "shiny" ) From cb373fbf1ca7f78a65fb8d23cc6b10769a7043e9 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Thu, 21 Mar 2024 12:46:36 +0100 Subject: [PATCH 05/23] fix typo --- tests/testthat/test-examples.R | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index edd433e77d..b201c2c8b4 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -34,7 +34,7 @@ for (i in rd_files()) { ), runApp = function(appDir, ...) { # nolint object_name_linter. app_driver <- shinytest2::AppDriver$new(appDir) - on.exit(app_driver$stop) + on.exit(app_driver$stop()) # shinytest2 will capture app crash but actually teal continues on error inside the module # we need to use a different way to check if there are errors if (!is.null(app_driver$get_html(".shiny-output-error:not(.shiny-output-error-validation)"))) { From fb412b97429464c5e08dc48de6e45cd34c519896 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Fri, 22 Mar 2024 11:46:18 +0100 Subject: [PATCH 06/23] add warning detection --- tests/testthat/test-examples.R | 40 ++++++++++++++++++++++++++-------- 1 file changed, 31 insertions(+), 9 deletions(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index b201c2c8b4..8bbc849674 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -22,7 +22,7 @@ for (i in rd_files()) { testthat::skip_on_cran() testthat::expect_no_error( capture.output( - # surpress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 + # suppress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 suppress_warnings( # quiet argument must be FALSE - otherwise the shiny apps are not invoked pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = FALSE), @@ -32,17 +32,39 @@ for (i in rd_files()) { ) } ), - runApp = function(appDir, ...) { # nolint object_name_linter. - app_driver <- shinytest2::AppDriver$new(appDir) - on.exit(app_driver$stop()) - # shinytest2 will capture app crash but actually teal continues on error inside the module - # we need to use a different way to check if there are errors + runApp = function(x, ...) { # nolint object_name_linter. + app_driver <- shinytest2::AppDriver$new( + x, + shiny_args = list(...), + check_names = FALSE, # explicit check below + options = options() # pass test options; this needs to be done explicitly + ) + on.exit(app_driver$stop(), add = TRUE) + + # Simple testing + ## warning in the app does not invoke a warning in the test + app_logs <- subset(app_driver$get_logs(), location == "shiny")[["message"]] + if (any(grepl("Warning in.*", app_logs))) { + warning( + sprintf( + "Detected following warning(s) (a message might be incomplete):\n%s", + paste0("* ", grep("Warning in.*", app_logs, value = TRUE), collapse = "\n") + ) + ) + } + + ## Throw an error instead of warning (only warnings are raised) + app_driver$expect_unique_names() + + ## shinytest2 captures app crash but teal continues on error inside the module + ## we need to use a different way to check if there are errors if (!is.null(app_driver$get_html(".shiny-output-error:not(.shiny-output-error-validation)"))) { - stop("module error is observed") + stop("Module error is observed.") } - # validation errors from shinyvalidate - added by default to assure the examples are "clean" + + ## validation errors from shinyvalidate - added by default to assure the examples are "clean" if (!is.null(app_driver$get_html(".shiny-input-container.has-error"))) { - stop("shinyvalidate error is observed") + stop("shinyvalidate error is observed.") } }, .package = "shiny" From acf6af6ba55cba37477f2076fb4cbfa206f46d3e 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, 22 Mar 2024 10:51:14 +0000 Subject: [PATCH 07/23] [skip roxygen] [skip vbump] Roxygen Man Pages Auto Update --- man/template_g_km.Rd | 21 +++++++++++---------- 1 file changed, 11 insertions(+), 10 deletions(-) diff --git a/man/template_g_km.Rd b/man/template_g_km.Rd index 514acfc6f0..b7dd855ceb 100644 --- a/man/template_g_km.Rd +++ b/man/template_g_km.Rd @@ -49,9 +49,9 @@ template_g_km( \item{cnsr_var}{(\code{character})\cr name of the censoring variable.} -\item{xticks}{(\code{numeric}, \code{number}, or \code{NULL})\cr numeric vector of ticks or single number with spacing -between ticks on the x axis. If \code{NULL} (default), \code{\link[labeling:extended]{labeling::extended()}} is used to determine -an optimal tick position on the x axis.} +\item{xticks}{(\code{numeric}, \code{number}, or \code{NULL})\cr numeric vector of tick positions or single number with spacing +between ticks on the x-axis. If \code{NULL} (default), \code{\link[labeling:extended]{labeling::extended()}} is used to determine +optimal tick positions on the x-axis.} \item{strata_var}{(\code{character})\cr names of the variables for stratified analysis.} @@ -66,11 +66,12 @@ an optimal tick position on the x axis.} \item{ties}{(\code{string})\cr among \code{exact} (equivalent to \code{DISCRETE} in SAS), \code{efron} and \code{breslow}, see \code{\link[survival:coxph]{survival::coxph()}}. Note: there is no equivalent of SAS \code{EXACT} method in R.} -\item{xlab}{(\code{string})\cr label of x-axis.} +\item{xlab}{(\code{string})\cr x-axis label.} \item{time_unit_var}{(\code{character})\cr name of the variable representing time units.} -\item{yval}{(\code{string})\cr value of y-axis. Options are \code{Survival} (default) and \code{Failure} probability.} +\item{yval}{(\code{string})\cr type of plot, to be plotted on the y-axis. Options are \code{Survival} (default) and \code{Failure} +probability.} \item{pval_method}{(\code{string})\cr the method used for estimation of p.values; \code{wald} (default) or \code{likelihood}.} @@ -79,13 +80,13 @@ median survival time per group.} \item{annot_coxph}{(\code{flag})\cr add the annotation table from a \code{\link[survival:coxph]{survival::coxph()}} model.} -\item{position_coxph}{(\code{numeric})\cr x and y positions for plotting \code{\link[survival:coxph]{survival::coxph()}} model.} +\item{position_coxph}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} Please use the \code{x} and \code{y} elements of +\code{control_annot_coxph} instead.} -\item{width_annots}{(named \code{list} of \code{unit}s)\cr a named list of widths for annotation tables with names \code{surv_med} -(median survival time table) and \code{coxph} (\code{\link[survival:coxph]{survival::coxph()}} model table), where each value is the width -(in units) to implement when printing the annotation table.} +\item{width_annots}{\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} Please use the \code{w} element of \code{control_annot_surv_med} +(for surv_med) and \code{control_annot_coxph} (for coxph)."} -\item{ci_ribbon}{(\code{flag})\cr draw the confidence interval around the Kaplan-Meier curve.} +\item{ci_ribbon}{(\code{flag})\cr whether the confidence interval should be drawn around the Kaplan-Meier curve.} \item{title}{(\code{character})\cr title of the output.} } From 3523d3982ba4d8f48a4321269321b31a93a0d500 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Fri, 22 Mar 2024 14:14:07 +0100 Subject: [PATCH 08/23] Update tests/testthat/test-examples.R Signed-off-by: Pawel Rucki <12943682+pawelru@users.noreply.github.com> --- tests/testthat/test-examples.R | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index 8bbc849674..302eb9555d 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -53,7 +53,7 @@ for (i in rd_files()) { ) } - ## Throw an error instead of warning (only warnings are raised) + ## Throw an error instead of a warning (default `check_names = TRUE` of `$new()` throws a warning) app_driver$expect_unique_names() ## shinytest2 captures app crash but teal continues on error inside the module From e1b1c1cf7c9406cefdc70ed0724529097cf816fe Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Fri, 22 Mar 2024 14:48:40 +0100 Subject: [PATCH 09/23] update wordlist for spelling exclusions --- inst/WORDLIST | 2 ++ 1 file changed, 2 insertions(+) diff --git a/inst/WORDLIST b/inst/WORDLIST index fbf1d1b4e8..33a7079c21 100644 --- a/inst/WORDLIST +++ b/inst/WORDLIST @@ -15,6 +15,7 @@ SMQ TLG UI Univariable +coxph customizable de funder @@ -24,5 +25,6 @@ programmatically repo responder responders +surv unadjusted univariable From 119682eabb344e16d19a3893c9c34e479c0c43ce Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Tue, 26 Mar 2024 15:54:29 +0100 Subject: [PATCH 10/23] better mocking allowing for quiet = TRUE --- R/utils.R | 3 + tests/testthat/test-examples.R | 108 +++++++++++++++++++-------------- 2 files changed, 66 insertions(+), 45 deletions(-) diff --git a/R/utils.R b/R/utils.R index c6ca9432b7..043afdf54c 100644 --- a/R/utils.R +++ b/R/utils.R @@ -943,3 +943,6 @@ set_default_total_label <- function(total_label) { checkmate::assert_character(total_label, len = 1, null.ok = TRUE) options("tmc_default_total_label" = total_label) } + +# for mocking in tests +interactive <- base::interactive diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index 302eb9555d..87a2529eea 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -14,59 +14,77 @@ suppress_warnings <- function(expr, pattern = "*", ...) { }) } -for (i in rd_files()) { +with_mocked_app_bindings <- function(code) { + shiny__shinyApp <- shiny::shinyApp # nolint object_name_linter. + mocked_shinyApp <- function(...) { # nolint object_name_linter. + print(shiny__shinyApp(...)) + } + + mocked_runApp <- function(x, ...) { # nolint object_name_linter. + app_driver <- shinytest2::AppDriver$new( + x, + shiny_args = list(...), + check_names = FALSE, # explicit check below + options = options() # https://github.com/rstudio/shinytest2/issues/377 + ) + on.exit(app_driver$stop(), add = TRUE) + + # Simple testing + ## warning in the app does not invoke a warning in the test + ## https://github.com/rstudio/shinytest2/issues/378 + app_logs <- subset(app_driver$get_logs(), location == "shiny")[["message"]] + if (any(grepl("Warning in.*", app_logs))) { + warning( + sprintf( + "Detected following warning(s) (a message might be incomplete):\n%s", + paste0("* ", grep("Warning in.*", app_logs, value = TRUE), collapse = "\n") + ) + ) + } + + ## Throw an error instead of a warning (default `AppDriver$new(..., check_names = TRUE)` throws a warning) + app_driver$expect_unique_names() + + ## shinytest2 captures app crash but teal continues on error inside the module + ## we need to use a different way to check if there are errors + if (!is.null(app_driver$get_html(".shiny-output-error:not(.shiny-output-error-validation)"))) { + stop("Module error is observed.") + } + + ## validation errors from shinyvalidate - added by default to assure the examples are "clean" + if (!is.null(app_driver$get_html(".shiny-input-container.has-error"))) { + stop("shinyvalidate error is observed.") + } + } + + # mock both local and package bindings to cover both `shinyApp(...)` and `shiny::shinyApp(...)` calls testthat::with_mocked_bindings( + testthat::with_mocked_bindings( + code, + shinyApp = shiny::shinyApp, + runApp = shiny::runApp, + interactive = function() TRUE + ), + shinyApp = mocked_shinyApp, + runApp = mocked_runApp, + .package = "shiny" + ) +} + +for (i in rd_files()) { + with_mocked_app_bindings( testthat::test_that( paste0("example-", basename(i)), { testthat::skip_on_cran() testthat::expect_no_error( - capture.output( - # suppress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 - suppress_warnings( - # quiet argument must be FALSE - otherwise the shiny apps are not invoked - pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = FALSE), - "may not be available when loading" - ) + # suppress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 + suppress_warnings( + pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE), + "may not be available when loading" ) ) } - ), - runApp = function(x, ...) { # nolint object_name_linter. - app_driver <- shinytest2::AppDriver$new( - x, - shiny_args = list(...), - check_names = FALSE, # explicit check below - options = options() # pass test options; this needs to be done explicitly - ) - on.exit(app_driver$stop(), add = TRUE) - - # Simple testing - ## warning in the app does not invoke a warning in the test - app_logs <- subset(app_driver$get_logs(), location == "shiny")[["message"]] - if (any(grepl("Warning in.*", app_logs))) { - warning( - sprintf( - "Detected following warning(s) (a message might be incomplete):\n%s", - paste0("* ", grep("Warning in.*", app_logs, value = TRUE), collapse = "\n") - ) - ) - } - - ## Throw an error instead of a warning (default `check_names = TRUE` of `$new()` throws a warning) - app_driver$expect_unique_names() - - ## shinytest2 captures app crash but teal continues on error inside the module - ## we need to use a different way to check if there are errors - if (!is.null(app_driver$get_html(".shiny-output-error:not(.shiny-output-error-validation)"))) { - stop("Module error is observed.") - } - - ## validation errors from shinyvalidate - added by default to assure the examples are "clean" - if (!is.null(app_driver$get_html(".shiny-input-container.has-error"))) { - stop("shinyvalidate error is observed.") - } - }, - .package = "shiny" + ) ) } From 6099c85e177ce107635539f774e70e20f1f5437a Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Tue, 26 Mar 2024 16:37:11 +0100 Subject: [PATCH 11/23] follow guides for mocking interactive --- R/utils.R | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/R/utils.R b/R/utils.R index 043afdf54c..069b699075 100644 --- a/R/utils.R +++ b/R/utils.R @@ -945,4 +945,4 @@ set_default_total_label <- function(total_label) { } # for mocking in tests -interactive <- base::interactive +interactive <- NULL From abc7f8e0a3e5bf972dfb5f1d3240bdeb0f0248a7 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Wed, 27 Mar 2024 08:59:12 +0100 Subject: [PATCH 12/23] Update tests/testthat/test-examples.R Co-authored-by: Vedha Viyash <49812166+vedhav@users.noreply.github.com> Signed-off-by: Pawel Rucki <12943682+pawelru@users.noreply.github.com> --- tests/testthat/test-examples.R | 1 + 1 file changed, 1 insertion(+) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index 87a2529eea..197e6b255a 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -28,6 +28,7 @@ with_mocked_app_bindings <- function(code) { options = options() # https://github.com/rstudio/shinytest2/issues/377 ) on.exit(app_driver$stop(), add = TRUE) + app_driver$wait_for_idle(timeout = 20000) # Simple testing ## warning in the app does not invoke a warning in the test From da378387a2cf11c630c709b429cb1992811a0477 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Wed, 27 Mar 2024 09:57:45 +0100 Subject: [PATCH 13/23] enhance error checking; enhance err msg; strict exceptions --- tests/testthat/test-examples.R | 48 +++++++++++++++++++++++----------- 1 file changed, 33 insertions(+), 15 deletions(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index 197e6b255a..fad93e2eb7 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -21,11 +21,15 @@ with_mocked_app_bindings <- function(code) { } mocked_runApp <- function(x, ...) { # nolint object_name_linter. - app_driver <- shinytest2::AppDriver$new( - x, - shiny_args = list(...), - check_names = FALSE, # explicit check below - options = options() # https://github.com/rstudio/shinytest2/issues/377 + # suppress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 + suppress_warnings( + app_driver <- shinytest2::AppDriver$new( + x, + shiny_args = list(...), + check_names = FALSE, # explicit check below + options = options() # https://github.com/rstudio/shinytest2/issues/377 + ), + "may not be available when loading" ) on.exit(app_driver$stop(), add = TRUE) app_driver$wait_for_idle(timeout = 20000) @@ -48,13 +52,13 @@ with_mocked_app_bindings <- function(code) { ## shinytest2 captures app crash but teal continues on error inside the module ## we need to use a different way to check if there are errors - if (!is.null(app_driver$get_html(".shiny-output-error:not(.shiny-output-error-validation)"))) { - stop("Module error is observed.") + if (!is.null(err_el <- app_driver$get_html(".shiny-output-error"))) { + stop(sprintf("Module error is observed:\n%s", err_el)) } ## validation errors from shinyvalidate - added by default to assure the examples are "clean" - if (!is.null(app_driver$get_html(".shiny-input-container.has-error"))) { - stop("shinyvalidate error is observed.") + if (!is.null(err_el <- app_driver$get_html(".shiny-input-container.has-error:not(.shiny-output-error-validation)"))) { + stop(sprintf("shinyvalidate error is observed:\n%s", err_el)) } } @@ -72,19 +76,33 @@ with_mocked_app_bindings <- function(code) { ) } +strict_exceptions <- c( + # https://github.com/r-lib/gtable/pull/94 + "tm_g_barchart_simple.Rd", + "tm_g_ci.Rd", + "tm_g_ipp.Rd", + "tm_g_pp_adverse_events.Rd", + "tm_g_pp_vitals.Rd" +) + for (i in rd_files()) { with_mocked_app_bindings( testthat::test_that( paste0("example-", basename(i)), { testthat::skip_on_cran() - testthat::expect_no_error( - # suppress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 - suppress_warnings( - pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE), - "may not be available when loading" + if (basename(i) %in% strict_exceptions) { + withr::with_options( + opts_partial_match_old, + testthat::expect_no_error( + pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE) + ) ) - ) + } else { + testthat::expect_no_error( + pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE) + ) + } } ) ) From 6d817a76e88e2514debb4d0e1ddd30aeca1b3f48 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Wed, 27 Mar 2024 10:03:45 +0100 Subject: [PATCH 14/23] add nolint --- tests/testthat/test-examples.R | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index fad93e2eb7..1c9e738026 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -57,7 +57,7 @@ with_mocked_app_bindings <- function(code) { } ## validation errors from shinyvalidate - added by default to assure the examples are "clean" - if (!is.null(err_el <- app_driver$get_html(".shiny-input-container.has-error:not(.shiny-output-error-validation)"))) { + if (!is.null(err_el <- app_driver$get_html(".shiny-input-container.has-error:not(.shiny-output-error-validation)"))) { # nolint line_length_linter. stop(sprintf("shinyvalidate error is observed:\n%s", err_el)) } } From b8894bd48d46e42daaf073f2c4161d94debed4a6 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Tue, 2 Apr 2024 17:50:42 +0200 Subject: [PATCH 15/23] more DRY call; implement workaround --- tests/testthat/test-examples.R | 52 +++++++++++++++++++--------------- 1 file changed, 29 insertions(+), 23 deletions(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index 1c9e738026..efd6d98040 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -16,8 +16,15 @@ suppress_warnings <- function(expr, pattern = "*", ...) { with_mocked_app_bindings <- function(code) { shiny__shinyApp <- shiny::shinyApp # nolint object_name_linter. - mocked_shinyApp <- function(...) { # nolint object_name_linter. - print(shiny__shinyApp(...)) + + # workaround of https://github.com/rstudio/shinytest2/issues/381 + # change to `print(shiny__shinyApp(...))` once fixed + mocked_shinyApp <- function(ui, server, ...) { # nolint object_name_linter. + functionBody(server) <- bquote({ + library(.(testthat::testing_package()), character.only = TRUE) + .(functionBody(server)) + }) + print(do.call(shiny__shinyApp, append(x = list(ui = ui, server = server), list(...)))) } mocked_runApp <- function(x, ...) { # nolint object_name_linter. @@ -41,8 +48,8 @@ with_mocked_app_bindings <- function(code) { if (any(grepl("Warning in.*", app_logs))) { warning( sprintf( - "Detected following warning(s) (a message might be incomplete):\n%s", - paste0("* ", grep("Warning in.*", app_logs, value = TRUE), collapse = "\n") + "Detected a warning in the application logs:\n%s", + paste0(app_logs, collapse = "\n") ) ) } @@ -62,7 +69,10 @@ with_mocked_app_bindings <- function(code) { } } - # mock both local and package bindings to cover both `shinyApp(...)` and `shiny::shinyApp(...)` calls + # support both `shinyApp(...)` as well as prefixed `shiny::shinyApp(...)` calls + # mock `shinyApp` to `shiny::shinyApp` and `shiny::shinyApp` to custom function + # same for `runApp(...)` and `shiny::runApp` + # additionally mock `interactive()` testthat::with_mocked_bindings( testthat::with_mocked_bindings( code, @@ -86,24 +96,20 @@ strict_exceptions <- c( ) for (i in rd_files()) { - with_mocked_app_bindings( - testthat::test_that( - paste0("example-", basename(i)), - { - testthat::skip_on_cran() - if (basename(i) %in% strict_exceptions) { - withr::with_options( - opts_partial_match_old, - testthat::expect_no_error( - pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE) - ) - ) - } else { - testthat::expect_no_error( - pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE) - ) - } + testthat::test_that( + paste0("example-", basename(i)), + { + testthat::skip_on_cran() + if (basename(i) %in% strict_exceptions) { + op <- options() + withr::local_options(opts_partial_match_old) + withr::defer(options(op)) } - ) + with_mocked_app_bindings( + testthat::expect_no_error( + pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE) + ) + ) + } ) } From bdf2160c39fdc52ced0b2c3f056e33675374454a Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Tue, 9 Apr 2024 12:12:13 +0200 Subject: [PATCH 16/23] suppress warnings on higher level --- tests/testthat/test-examples.R | 33 ++++++++++++++++++--------------- 1 file changed, 18 insertions(+), 15 deletions(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index efd6d98040..d1e6f370a2 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -7,11 +7,14 @@ rd_files <- function() { } suppress_warnings <- function(expr, pattern = "*", ...) { - withCallingHandlers(expr, warning = function(w) { - if (grepl(pattern, conditionMessage(w))) { - invokeRestart("muffleWarning") + withCallingHandlers( + expr, + warning = function(w) { + if (grepl(pattern, conditionMessage(w))) { + invokeRestart("muffleWarning") + } } - }) + ) } with_mocked_app_bindings <- function(code) { @@ -28,15 +31,11 @@ with_mocked_app_bindings <- function(code) { } mocked_runApp <- function(x, ...) { # nolint object_name_linter. - # suppress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 - suppress_warnings( - app_driver <- shinytest2::AppDriver$new( - x, - shiny_args = list(...), - check_names = FALSE, # explicit check below - options = options() # https://github.com/rstudio/shinytest2/issues/377 - ), - "may not be available when loading" + app_driver <- shinytest2::AppDriver$new( + x, + shiny_args = list(...), + check_names = FALSE, # explicit check below + options = options() # https://github.com/rstudio/shinytest2/issues/377 ) on.exit(app_driver$stop(), add = TRUE) app_driver$wait_for_idle(timeout = 20000) @@ -106,8 +105,12 @@ for (i in rd_files()) { withr::defer(options(op)) } with_mocked_app_bindings( - testthat::expect_no_error( - pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE) + # suppress warnings coming from saving qenv https://github.com/insightsengineering/teal.code/issues/194 + suppress_warnings( + testthat::expect_no_error( + pkgload::run_example(i, run_donttest = TRUE, run_dontrun = FALSE, quiet = TRUE) + ), + "may not be available when loading" ) ) } From f5364149cd3a699b862c77d1b73cf9119f2f5ca6 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Wed, 10 Apr 2024 12:17:52 +0200 Subject: [PATCH 17/23] to be FALSE --- tests/testthat/test-examples.R | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index d1e6f370a2..c7c95afdd2 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -31,9 +31,12 @@ with_mocked_app_bindings <- function(code) { } mocked_runApp <- function(x, ...) { # nolint object_name_linter. + args <- list(...) + args[["launch.browser"]] <- FALSE # needed for RStudio + app_driver <- shinytest2::AppDriver$new( x, - shiny_args = list(...), + shiny_args = args, check_names = FALSE, # explicit check below options = options() # https://github.com/rstudio/shinytest2/issues/377 ) From 2bb43aa7f7898ce0fb0a40166d1906a0653de76f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andr=C3=A9=20Ver=C3=ADssimo?= <211358+averissimo@users.noreply.github.com> Date: Wed, 10 Apr 2024 12:57:53 +0200 Subject: [PATCH 18/23] docs: adds packages to verdepcheck --- DESCRIPTION | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/DESCRIPTION b/DESCRIPTION index 2dc08c6144..0bca0f25cb 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -89,8 +89,8 @@ Config/Needs/verdepcheck: insightsengineering/teal, insightsengineering/teal.reporter, insightsengineering/teal.widgets, insightsengineering/tern.gee, insightsengineering/tern.mmrm, tidyverse/tidyr, shosaco/vistime, tidyverse/forcats, yihui/knitr, - tidyverse/lubridate, insightsengineering/nestcolor, r-lib/styler, - r-lib/testthat, r-lib/withr + tidyverse/lubridate, insightsengineering/nestcolor, r-lib/pkgload, + rstudio/shinytest2, r-lib/styler, r-lib/testthat, r-lib/withr Config/Needs/website: insightsengineering/nesttemplate Config/testthat/edition: 3 Encoding: UTF-8 From c084ad3ca08097f4397eea53d91a42df706820ab Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Wed, 10 Apr 2024 14:43:09 +0200 Subject: [PATCH 19/23] load_all instead of library --- tests/testthat/test-examples.R | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index c7c95afdd2..4a6e6ac528 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -21,10 +21,10 @@ with_mocked_app_bindings <- function(code) { shiny__shinyApp <- shiny::shinyApp # nolint object_name_linter. # workaround of https://github.com/rstudio/shinytest2/issues/381 - # change to `print(shiny__shinyApp(...))` once fixed + # change to `print(shiny__shinyApp(...))` and remove allow warning once fixed mocked_shinyApp <- function(ui, server, ...) { # nolint object_name_linter. functionBody(server) <- bquote({ - library(.(testthat::testing_package()), character.only = TRUE) + pkgload::load_all(.(normalizePath(file.path(testthat::test_path(), "../.."))), export_all = FALSE, attach_testthat = FALSE, warn_conflicts = FALSE) .(functionBody(server)) }) print(do.call(shiny__shinyApp, append(x = list(ui = ui, server = server), list(...)))) @@ -47,7 +47,8 @@ with_mocked_app_bindings <- function(code) { ## warning in the app does not invoke a warning in the test ## https://github.com/rstudio/shinytest2/issues/378 app_logs <- subset(app_driver$get_logs(), location == "shiny")[["message"]] - if (any(grepl("Warning in.*", app_logs))) { + # allow `Warning in file(con, "r")` warning coming from pkgload::load_all() + if (any(grepl("Warning in.*", app_logs) & !grepl("Warning in file\\(con, \"r\"\\)", app_logs))) { warning( sprintf( "Detected a warning in the application logs:\n%s", From 18b94e066a94978e58008d2bc651479b2f92fbf5 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Wed, 10 Apr 2024 14:47:07 +0200 Subject: [PATCH 20/23] lint --- tests/testthat/test-examples.R | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index 4a6e6ac528..dab0c65976 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -24,7 +24,12 @@ with_mocked_app_bindings <- function(code) { # change to `print(shiny__shinyApp(...))` and remove allow warning once fixed mocked_shinyApp <- function(ui, server, ...) { # nolint object_name_linter. functionBody(server) <- bquote({ - pkgload::load_all(.(normalizePath(file.path(testthat::test_path(), "../.."))), export_all = FALSE, attach_testthat = FALSE, warn_conflicts = FALSE) + pkgload::load_all( + .(normalizePath(file.path(testthat::test_path(), "../.."))), + export_all = FALSE, + attach_testthat = FALSE, + warn_conflicts = FALSE + ) .(functionBody(server)) }) print(do.call(shiny__shinyApp, append(x = list(ui = ui, server = server), list(...)))) From 7693dbbc024d1e77ee61292802b6740b639e33b9 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Wed, 10 Apr 2024 15:34:37 +0200 Subject: [PATCH 21/23] Update tests/testthat/test-examples.R MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: André Veríssimo <211358+averissimo@users.noreply.github.com> Signed-off-by: Pawel Rucki <12943682+pawelru@users.noreply.github.com> --- tests/testthat/test-examples.R | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index dab0c65976..005bb3de8f 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -25,7 +25,7 @@ with_mocked_app_bindings <- function(code) { mocked_shinyApp <- function(ui, server, ...) { # nolint object_name_linter. functionBody(server) <- bquote({ pkgload::load_all( - .(normalizePath(file.path(testthat::test_path(), "../.."))), + .(normalizePath(file.path(testthat::test_path(), "..", ".."))), export_all = FALSE, attach_testthat = FALSE, warn_conflicts = FALSE From 6858c728ee31aedf645e77590c4e00ffaf1e47a7 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Wed, 10 Apr 2024 16:03:17 +0200 Subject: [PATCH 22/23] use ../../man and rm symlink --- tests/testthat/man | 1 - tests/testthat/test-examples.R | 6 +++++- 2 files changed, 5 insertions(+), 2 deletions(-) delete mode 120000 tests/testthat/man diff --git a/tests/testthat/man b/tests/testthat/man deleted file mode 120000 index ee201c1931..0000000000 --- a/tests/testthat/man +++ /dev/null @@ -1 +0,0 @@ -../../man \ No newline at end of file diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index 005bb3de8f..a713636262 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -3,7 +3,11 @@ # this also requires `devtools::document()` to be run before running the tests rd_files <- function() { - list.files(testthat::test_path("man"), pattern = "\\.[Rr]d$", full.names = TRUE) + list.files( + normalizePath(file.path(testthat::test_path(), "..", "..", "man")), + pattern = "\\.[Rr]d$", + full.names = TRUE + ) } suppress_warnings <- function(expr, pattern = "*", ...) { From f6d93a259a11f9791aa01cb1aac2b87e62ffcaa8 Mon Sep 17 00:00:00 2001 From: Pawel Rucki <12943682+pawelru@users.noreply.github.com> Date: Thu, 11 Apr 2024 11:39:02 +0200 Subject: [PATCH 23/23] enhance path to man --- tests/testthat/test-examples.R | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/tests/testthat/test-examples.R b/tests/testthat/test-examples.R index a713636262..601cb269a6 100644 --- a/tests/testthat/test-examples.R +++ b/tests/testthat/test-examples.R @@ -3,8 +3,18 @@ # this also requires `devtools::document()` to be run before running the tests rd_files <- function() { + man_path <- if (testthat::is_checking()) { + testthat::test_path("..", "..", "00_pkg_src", testthat::testing_package(), "man") + } else { + testthat::test_path("..", "..", "man") + } + + if (!dir.exists(man_path)) { + stop("Cannot find path to `man` directory.") + } + list.files( - normalizePath(file.path(testthat::test_path(), "..", "..", "man")), + man_path, pattern = "\\.[Rr]d$", full.names = TRUE )