Skip to content

Commit

Permalink
Merge pull request #144 from carpentries/differentiate-image-links-143
Browse files Browse the repository at this point in the history
`$validate_links()` method: differentiate between images and links in report summary
  • Loading branch information
zkamvar authored Nov 17, 2023
2 parents e7e5c45 + c6e3967 commit 0d84122
Show file tree
Hide file tree
Showing 11 changed files with 283 additions and 21 deletions.
5 changes: 5 additions & 0 deletions NEWS.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,11 @@

* Snapshot update for testthat 3.2.0

## BUGFIX

* `validate_links()` now differentiates between links and images in reporting
(reported: @joelnitta, #143; fixed: @zkamvar, #144)

# pegboard 0.7.1 (2023-10-03)

* child chunk options that would fail out of context no longer cause a failure.
Expand Down
8 changes: 5 additions & 3 deletions R/utils-cli.R
Original file line number Diff line number Diff line change
Expand Up @@ -98,8 +98,10 @@ pb_message <- function (..., domain = NULL, appendLF = TRUE) {
#'
#' @param path path to the file to report
#' @param pos position of the error
#' @param type the type of warning that should be thrown (defaults to warning)
#' @param sep a character to use to separate the human message and the line number
#' @param type (used in the context of CI only) the type of warning that should
#' be thrown (defaults to warning)
#' @param sep a character to use to separate the human message and the line
#' number
#' @rdname cli_helpers
line_report <- function(msg = "", path, pos, sep = "\t", type = "warning") {
ci <- Sys.getenv("CI") != ""
Expand Down Expand Up @@ -130,7 +132,7 @@ line_report <- function(msg = "", path, pos, sep = "\t", type = "warning") {
#' cli = requireNamespace("cli", quietly = TRUE),
#' f = "col_cyan"
#' )
#' cat(glue::glue("[{x}]->[{x2}]"))
#' writeLines(glue::glue("[{x}]->[{x2}]"))
append_labels <- function(l, i = TRUE, e = "", cli = FALSE, f = "style_inverse") {
f <- if (cli) utils::getFromNamespace(f, "cli") else function(e) e
l[i] <- paste(l[i], f(e))
Expand Down
110 changes: 107 additions & 3 deletions R/utils-validation.R
Original file line number Diff line number Diff line change
@@ -1,3 +1,68 @@
#' Throw a validation report as a single message
#'
#' Collapse a variable number of validation reports into a single message that
#' can be formatted for the CLI or GitHub.
#'
#' @param VAL `[data.frame]` a validation report derived from one of the
#' `validate` functions.
#' @return NULL, invisibly. This is used for it's side-effect of formatting and
#' issuing messages via [issue_warning()].
#' @details One of the key features of {pegboard} is the ability to parse and
#' validate markdown elements. These functions provide a standard way of
#' creating the reports that are for the user based on whether or not they are
#' on the CLI or on GitHub. The prerequisites of these functions are the input
#' data frame (generated from the actual validation function) and an internal
#' set of known templating vectors that contain templates for each test to show
#' the actual error along with general information that can help correct the
#' error (see below).
#'
#'
#' ## Input Data Frame
#'
#' The validations are initially reported in a data frame that has the
#' following properties:
#' - one row per element
#' - columns that indicate the parsed attributes of the element, source
#' file, source position, and the actual element XML node object.
#' - boolean columns that indicate the tests for each element, used with
#' [collect_labels()] to add a "labels" column to the data.
#'
#' ## Templating vectors
#'
#' These vectors come in two forms `[thing]_tests` and `[thing]_info` (e.g.
#' for [validate_links()], we have `link_tests` and `link_info`). These are
#' named vectors that match the boolean columns of the data frame produced
#' by the validation function. The `[thing]_tests` vector contains templates
#' that describes the error and shows the text that caused the error. The
#' `[thing]_info` contains general information about how to address that
#' particular error. For example, one common link error is that a link is not
#' descriptive (e.g. the link text says "click here"). The column in the `VAL`
#' data frame that contains the result of this test is called "descriptive", so
#' if we look at the values from the link info and tests vectors:
#'
#' ```{r}
#' link_info["descriptive"]
#' link_tests["descriptive"]
#' ```
#'
#' If the `throw_*_warnings()` functions detect any errors, they will use the
#' info and tests vectors to construct a composite message.
#'
#' ## Process
#'
#' The `throw_*_warnings()` functions all do the same basic procedure (and
#' indeed could be consolidated into a single function in the future)
#'
#' 1. pass data to [collect_labels()], which will parse the `[thing]_tests`
#' templating vector and label each failing element in `VAL` with the
#' appropriate failure message
#' 2. gather the source information for each failure
#' 3. pass failures with the `[thing]_info` elements that matched the unique
#' failures to [issue_warning()]
#' @seealso
#' [validate_links()], [validate_divs()], and [validate_headings()] for
#' input sources for these functions.
#' @rdname throw_warnings
throw_heading_warnings <- function(VAL) {
if (length(VAL) == 0 || nrow(VAL) == 0) {
return(invisible(NULL))
Expand All @@ -20,6 +85,7 @@ throw_heading_warnings <- function(VAL) {
reports = reports)
}

#' @rdname throw_warnings
throw_div_warnings <- function(VAL) {
if (length(VAL) == 0 || nrow(VAL) == 0) {
return(invisible(NULL))
Expand All @@ -41,6 +107,7 @@ throw_div_warnings <- function(VAL) {
reports = reports)
}

#' @rdname throw_warnings
throw_link_warnings <- function(VAL) {
if (length(VAL) == 0 || nrow(VAL) == 0) {
return(invisible(NULL))
Expand All @@ -55,18 +122,55 @@ throw_link_warnings <- function(VAL) {

reports <- line_report(msg = err$labels, err$filepath, err$sourcepos, sep = " ")
failed <- !apply(err[names(link_tests)], MARGIN = 2, all)
issue_warning(what = "links",
types <- paste0(unique(sub("img", "image", err$type)), "s")
issue_warning(what = paste(types, collapse = " and "),
cli = has_cli(),
n = nrow(err),
N = nrow(VAL),
infos = link_info[failed],
reports = reports)
}

#' Collect and append validation messages
#'
#' Given a data frame containing the results of validation tests, this will
#' append a column of labels that describes each failure.
#'
#' @param VAL a data frame containing the results of tests
#' @param cli indicator to use the cli package to format warnings
#' @param msg (collect_labels) a named vector of messages to provide for each test
#' @noRd
#' @param msg a named vector of template messages to provide for each test
#' formatted for the \pkg{glue} package.
#'
#' @seealso [throw_link_warnings()] for details on how this is implemented.
#' @examples
#' # As an example, consider a data frame where you have observations in rows
#' # and the results of individual tests in columns:
#' set.seed(2023-11-16)
#' dat <- data.frame(
#' name = letters[1:10],
#' rank = sample(1:3, 10, replace = TRUE),
#' A = sample(c(TRUE, FALSE), 10, replace = TRUE, prob = c(7, 3)),
#' B = sample(c(TRUE, FALSE), 10, replace = TRUE, prob = c(7, 3)),
#' C = sample(c(TRUE, FALSE), 10, replace = TRUE, prob = c(7, 3))
#' )
#' dat
#' # you can see what the results of the tests were, but it would be a good
#' # idea to have a lookup table describing what these results mean
#' dat_tests <- c(
#' A = "[missing widget]: {name}",
#' B = "[incorrect rank]: {rank}",
#' C = "[something else]"
#' )
#' # collect_labels will create the output you need:
#' pb <- asNamespace("pegboard")
#' res <- pb$collect_labels(dat, msg = dat_tests)
#' res
#' writeLines(res$labels)
#' if (requireNamespace("cli", quietly = TRUE)) {
#' # you can also specify cli to TRUE to format with CLI
#' res <- pb$collect_labels(dat, cli = TRUE, msg = dat_tests)
#' writeLines(res$labels)
#' }
collect_labels <- function(VAL, cli = FALSE, msg = heading_tests) {
labels <- character(nrow(VAL))
for (test in names(msg)) {
Expand Down
2 changes: 2 additions & 0 deletions _pkgdown.yml
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,8 @@ reference:
- validate_headings
- validate_links
- validate_divs
- throw_heading_warnings
- collect_labels
- issue_warning
- show_heading_tree
- title: "Helper functions"
Expand Down
8 changes: 5 additions & 3 deletions man/cli_helpers.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

53 changes: 53 additions & 0 deletions man/collect_labels.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

5 changes: 4 additions & 1 deletion man/make_link_table.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

91 changes: 91 additions & 0 deletions man/throw_warnings.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading

0 comments on commit 0d84122

Please sign in to comment.