From f6ee3647f808abcb587c9ad2a705f59eadc22c77 Mon Sep 17 00:00:00 2001 From: Aleksander Chlebowski <114988527+chlebowa@users.noreply.github.com> Date: Thu, 19 Oct 2023 11:32:50 +0200 Subject: [PATCH 1/2] 938 transfer slices store (#939) Closes #938 Added functions `slices_store` and `slices_restore` as internals. --- DESCRIPTION | 2 + NEWS.md | 1 + R/module_snapshot_manager.R | 4 +- R/teal_slices-store.R | 86 ++++++++++++++ man/slices_restore.Rd | 27 +++++ man/slices_store.Rd | 44 +++++++ man/snapshot_manager_module.Rd | 2 +- tests/testthat/test-teal_slices-store.R | 151 ++++++++++++++++++++++++ 8 files changed, 314 insertions(+), 3 deletions(-) create mode 100644 R/teal_slices-store.R create mode 100644 man/slices_restore.Rd create mode 100644 man/slices_store.Rd create mode 100644 tests/testthat/test-teal_slices-store.R diff --git a/DESCRIPTION b/DESCRIPTION index 95328de84b..32ca9afac8 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -36,6 +36,7 @@ Depends: teal.transform (>= 0.4.0) Imports: checkmate, + jsonlite, lifecycle, logger (>= 0.2.0), magrittr, @@ -87,6 +88,7 @@ Collate: 'tdata.R' 'teal.R' 'teal_reporter.R' + 'teal_slices-store.R' 'teal_slices.R' 'utils.R' 'validate_inputs.R' diff --git a/NEWS.md b/NEWS.md index 55545d90bf..0fc641f62f 100644 --- a/NEWS.md +++ b/NEWS.md @@ -4,6 +4,7 @@ * 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. # teal 0.14.0 diff --git a/R/module_snapshot_manager.R b/R/module_snapshot_manager.R index 377ac92909..d2e05db923 100644 --- a/R/module_snapshot_manager.R +++ b/R/module_snapshot_manager.R @@ -47,7 +47,7 @@ #' The snapshot is then set as the current content of `slices_global`. #' #' To save a snapshot, the snapshot is retrieved and reassembled just like for restoring, -#' and then saved to file with [`teal.slice::slices_store`]. +#' and then saved to file with [`slices_store`]. #' #' @param id (`character(1)`) `shiny` module id #' @param slices_global (`reactiveVal`) that contains a `teal_slices` object @@ -208,7 +208,7 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat content = function(file) { snapshot <- snapshot_history()[[s]] snapshot_state <- as.teal_slices(snapshot) - teal.slice::slices_store(tss = snapshot_state, file = file) + slices_store(tss = snapshot_state, file = file) } ) handlers[[id_saveme]] <- id_saveme diff --git a/R/teal_slices-store.R b/R/teal_slices-store.R new file mode 100644 index 0000000000..8c33733103 --- /dev/null +++ b/R/teal_slices-store.R @@ -0,0 +1,86 @@ +#' Store teal_slices object to a file +#' +#' This function takes a `teal_slices` object and saves it to a file in `JSON` format. +#' The `teal_slices` object contains information about filter states and can be used to +#' create, modify, and delete filter states. The saved file can be later loaded using +#' the `slices_restore` function. +#' +#' @param tss (`teal_slices`) object to be stored. +#' @param file (`character(1)`) The file path where `teal_slices` object will be saved. +#' The file extension should be `".json"`. +#' +#' @details `Date` class is stored in `"ISO8601"` format (`YYYY-MM-DD`). `POSIX*t` classes are converted to a +#' character by using `format.POSIX*t(usetz = TRUE, tz = "UTC")` (`YYYY-MM-DD {N}{N}:{N}{N}:{N}{N} UTC`, where +#' `{N} = [0-9]` is a number and `UTC` is `Coordinated Universal Time` timezone short-code). +#' This format is assumed during `slices_restore`. All `POSIX*t` objects in `selected` or `choices` fields of +#' `teal_slice` objects are always printed in `UTC` timezone as well. +#' +#' @return `NULL`, invisibly. +#' +#' @keywords internal +#' +#' @examples +#' # Create a teal_slices object +#' tss <- teal_slices( +#' teal_slice(dataname = "data", varname = "var"), +#' teal_slice(dataname = "data", expr = "x > 0", id = "positive_x", title = "Positive x") +#' ) +#' +#' if (interactive()) { +#' # Store the teal_slices object to a file +#' slices_store(tss, "path/to/file.json") +#' } +#' +slices_store <- function(tss, file) { + checkmate::assert_class(tss, "teal_slices") + checkmate::assert_path_for_output(file, overwrite = TRUE, extension = "json") + + cat(format(tss, trim_lines = FALSE), "\n", file = file) +} + +#' Restore teal_slices object from a file +#' +#' This function takes a file path to a `JSON` file containing a `teal_slices` object +#' and restores it to its original form. The restored `teal_slices` object can be used +#' to access filter states and their corresponding attributes. +#' +#' @param file Path to file where `teal_slices` is stored. Must have a `.json` extension and read access. +#' +#' @return A `teal_slices` object restored from the file. +#' +#' @keywords internal +#' +#' @examples +#' if (interactive()) { +#' # Restore a teal_slices object from a file +#' tss_restored <- slices_restore("path/to/file.json") +#' } +#' +slices_restore <- function(file) { + checkmate::assert_file_exists(file, access = "r", extension = "json") + + tss_json <- jsonlite::fromJSON(file, simplifyDataFrame = FALSE) + tss_json$slices <- + lapply(tss_json$slices, function(slice) { + for (field in c("selected", "choices")) { + if (!is.null(slice[[field]])) { + date_partial_regex <- "^[0-9]{4}-[0-9]{2}-[0-9]{2}" + time_stamp_regex <- paste0(date_partial_regex, "\\s[0-9]{2}:[0-9]{2}:[0-9]{2}\\sUTC$") + + slice[[field]] <- + if (all(grepl(paste0(date_partial_regex, "$"), slice[[field]]))) { + as.Date(slice[[field]]) + } else if (all(grepl(time_stamp_regex, slice[[field]]))) { + as.POSIXct(slice[[field]], tz = "UTC") + } else { + slice[[field]] + } + } + } + slice + }) + + tss_elements <- lapply(tss_json$slices, as.teal_slice) + + do.call(teal_slices, c(tss_elements, tss_json$attributes)) +} diff --git a/man/slices_restore.Rd b/man/slices_restore.Rd new file mode 100644 index 0000000000..c2b2c39192 --- /dev/null +++ b/man/slices_restore.Rd @@ -0,0 +1,27 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/teal_slices-store.R +\name{slices_restore} +\alias{slices_restore} +\title{Restore teal_slices object from a file} +\usage{ +slices_restore(file) +} +\arguments{ +\item{file}{Path to file where \code{teal_slices} is stored. Must have a \code{.json} extension and read access.} +} +\value{ +A \code{teal_slices} object restored from the file. +} +\description{ +This function takes a file path to a \code{JSON} file containing a \code{teal_slices} object +and restores it to its original form. The restored \code{teal_slices} object can be used +to access filter states and their corresponding attributes. +} +\examples{ +if (interactive()) { + # Restore a teal_slices object from a file + tss_restored <- slices_restore("path/to/file.json") +} + +} +\keyword{internal} diff --git a/man/slices_store.Rd b/man/slices_store.Rd new file mode 100644 index 0000000000..9ee4a96c5c --- /dev/null +++ b/man/slices_store.Rd @@ -0,0 +1,44 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/teal_slices-store.R +\name{slices_store} +\alias{slices_store} +\title{Store teal_slices object to a file} +\usage{ +slices_store(tss, file) +} +\arguments{ +\item{tss}{(\code{teal_slices}) object to be stored.} + +\item{file}{(\code{character(1)}) The file path where \code{teal_slices} object will be saved. +The file extension should be \code{".json"}.} +} +\value{ +\code{NULL}, invisibly. +} +\description{ +This function takes a \code{teal_slices} object and saves it to a file in \code{JSON} format. +The \code{teal_slices} object contains information about filter states and can be used to +create, modify, and delete filter states. The saved file can be later loaded using +the \code{slices_restore} function. +} +\details{ +\code{Date} class is stored in \code{"ISO8601"} format (\code{YYYY-MM-DD}). \code{POSIX*t} classes are converted to a +character by using \code{format.POSIX*t(usetz = TRUE, tz = "UTC")} (\verb{YYYY-MM-DD \{N\}\{N\}:\{N\}\{N\}:\{N\}\{N\} UTC}, where +\verb{\{N\} = [0-9]} is a number and \code{UTC} is \verb{Coordinated Universal Time} timezone short-code). +This format is assumed during \code{slices_restore}. All \code{POSIX*t} objects in \code{selected} or \code{choices} fields of +\code{teal_slice} objects are always printed in \code{UTC} timezone as well. +} +\examples{ +# Create a teal_slices object +tss <- teal_slices( + teal_slice(dataname = "data", varname = "var"), + teal_slice(dataname = "data", expr = "x > 0", id = "positive_x", title = "Positive x") +) + +if (interactive()) { + # Store the teal_slices object to a file + slices_store(tss, "path/to/file.json") +} + +} +\keyword{internal} diff --git a/man/snapshot_manager_module.Rd b/man/snapshot_manager_module.Rd index 3dbab564fc..ec883c2c07 100644 --- a/man/snapshot_manager_module.Rd +++ b/man/snapshot_manager_module.Rd @@ -79,7 +79,7 @@ and set anew according to the \code{mapping} attribute of the snapshot. The snapshot is then set as the current content of \code{slices_global}. To save a snapshot, the snapshot is retrieved and reassembled just like for restoring, -and then saved to file with \code{\link[teal.slice:slices_store]{teal.slice::slices_store}}. +and then saved to file with \code{\link{slices_store}}. } \author{ diff --git a/tests/testthat/test-teal_slices-store.R b/tests/testthat/test-teal_slices-store.R new file mode 100644 index 0000000000..be2d481b87 --- /dev/null +++ b/tests/testthat/test-teal_slices-store.R @@ -0,0 +1,151 @@ +testthat::test_that("teal_slice store/restore supports saving `POSIXct` timestamps in selected", { + slices_path <- withr::local_file("slices.json") + + time_stamps <- Sys.time() + c(-10 * 60 * 60 * 24, -30, 0) + + # ISO8601 does not keep milliseconds + time_stamps <- as.POSIXct( + ceiling(as.double(time_stamps)), + tz = "UTC", + origin = "1970-01-01" + ) + + tss <- teal_slices( + teal_slice( + dataname = "ADSL", + varname = "EOSDTM", + selected = time_stamps, + fixed = TRUE + ) + ) + + # Store the teal_slices object to a file + slices_store(tss, slices_path) + tss_restored <- slices_restore(slices_path) + + tss_restored_list <- shiny::isolate(shiny::reactiveValuesToList(tss_restored[[1]])) + testthat::expect_s3_class(tss_restored_list$selected, "POSIXct") + + teal.slice:::expect_identical_slice(tss[[1]], tss_restored[[1]]) +}) + +testthat::test_that("teal_slice store/restore supports saving `Date` dates in selected", { + slices_path <- withr::local_file("slices.json") + + time_stamps <- Sys.Date() + c(-10 * 600, -30, 0) + + tss <- teal_slices( + teal_slice( + dataname = "ADSL", + varname = "EOSDT", + selected = time_stamps, + fixed = TRUE + ) + ) + + # Store the teal_slices object to a file + slices_store(tss, slices_path) + tss_restored <- slices_restore(slices_path) + + tss_restored_list <- shiny::isolate(shiny::reactiveValuesToList(tss_restored[[1]])) + testthat::expect_s3_class(tss_restored_list$selected, "Date") + + teal.slice:::expect_identical_slice(tss[[1]], tss_restored[[1]]) +}) + +testthat::test_that("teal_slice store/restore supports saving `POSIXct` timestamps in choices", { + slices_path <- withr::local_file("slices.json") + + time_stamps <- Sys.time() + c(-10 * 60 * 60 * 24, -30, 0) + + # ISO8601 does not keep milliseconds + time_stamps <- as.POSIXct( + ceiling(as.double(time_stamps)), + tz = "UTC", + origin = "1970-01-01" + ) + + tss <- teal_slices( + teal_slice( + dataname = "ADSL", + varname = "EOSDTM", + selected = sample(time_stamps, 2), + choices = time_stamps, + fixed = TRUE + ) + ) + + # Store the teal_slices object to a file + slices_store(tss, slices_path) + tss_restored <- slices_restore(slices_path) + + tss_restored_list <- shiny::isolate(shiny::reactiveValuesToList(tss_restored[[1]])) + testthat::expect_s3_class(tss_restored_list$choices, "POSIXct") + + teal.slice:::expect_identical_slice(tss[[1]], tss_restored[[1]]) +}) + +testthat::test_that("teal_slice store/restore supports saving `Date` timestamps in choices", { + slices_path <- withr::local_file("slices.json") + + time_stamps <- Sys.Date() + c(-10 * 600, -30, 0) + + tss <- teal_slices( + teal_slice( + dataname = "ADSL", + varname = "EOSDT", + selected = sample(time_stamps, 2), + choices = time_stamps, + fixed = TRUE + ) + ) + + # Store the teal_slices object to a file + slices_store(tss, slices_path) + tss_restored <- slices_restore(slices_path) + + tss_restored_list <- shiny::isolate(shiny::reactiveValuesToList(tss_restored[[1]])) + testthat::expect_s3_class(tss_restored_list$choices, "Date") + + teal.slice:::expect_identical_slice(tss[[1]], tss_restored[[1]]) +}) + + +testthat::test_that("teal_slice store/restore restores mixed `Date`-characters as characters in selected", { + slices_path <- withr::local_file("slices.json") + tss <- teal_slices( + teal_slice( + dataname = "ADSL", + varname = "EOSDTM", + selected = c( + "beta 2023-09-11", + "release candidate 2023-09-21", + "release 2023-09-21" + ), + fixed = TRUE + ) + ) + + slices_store(tss, slices_path) + tss_restored <- slices_restore(slices_path) + teal.slice:::expect_identical_slice(tss[[1]], tss_restored[[1]]) +}) + +testthat::test_that("teal_slice store/restore restores characters as characters in selected and choices", { + slices_path <- withr::local_file("slices.json") + tss <- teal_slices( + teal_slice( + dataname = "ADSL", + varname = "EOSDTM", + choices = c("a", "b", "c"), + selected = c("a", "b") + ) + ) + + slices_store(tss, slices_path) + tss_restored <- slices_restore(slices_path) + + testthat::expect_type(shiny::isolate(tss_restored[[1]]$selected), "character") + testthat::expect_type(shiny::isolate(tss_restored[[1]]$choices), "character") + teal.slice:::expect_identical_slice(tss[[1]], tss_restored[[1]]) +}) From 031fc308af6c9c0956419560460ac31a35aa4629 Mon Sep 17 00:00:00 2001 From: chlebowa Date: Thu, 19 Oct 2023 09:34:08 +0000 Subject: [PATCH 2/2] [skip actions] Bump version to 0.14.0.9013 --- .pre-commit-config.yaml | 2 +- DESCRIPTION | 4 ++-- NEWS.md | 2 +- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 7c429caeb2..8d2f4ddaeb 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -6,7 +6,7 @@ default_language_version: python: python3 repos: - repo: https://github.com/lorenzwalthert/precommit - rev: v0.3.2.9021 + rev: v0.3.2.9023 hooks: - id: style-files name: Style code with `styler` diff --git a/DESCRIPTION b/DESCRIPTION index 32ca9afac8..9c37b46189 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -1,8 +1,8 @@ Type: Package Package: teal Title: Exploratory Web Apps for Analyzing Clinical Trials Data -Version: 0.14.0.9012 -Date: 2023-10-13 +Version: 0.14.0.9013 +Date: 2023-10-19 Authors@R: c( person("Dawid", "Kaledkowski", , "dawid.kaledkowski@roche.com", role = c("aut", "cre")), person("Pawel", "Rucki", , "pawel.rucki@roche.com", role = "aut"), diff --git a/NEWS.md b/NEWS.md index 0fc641f62f..986272c1a5 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,4 +1,4 @@ -# teal 0.14.0.9012 +# teal 0.14.0.9013 ### Miscellaneous