From ba9a60e96c3f0b6abc669c9495868299b2affccc Mon Sep 17 00:00:00 2001 From: Hadley Wickham Date: Tue, 7 May 2024 13:47:45 -0500 Subject: [PATCH] Unify sitrep and check * Introduce new `check_urls()` function that reports on url problems with errors and call it in `check_pkgdown()` * `pkgdown_sitrep()` now calls the same functions as `check_pkgdown()` but wrapped in a helper to convert errors to messages * `check_built()` is a separate concern so it's moved to a separate file Fixes #2463. Fixes #2380. --- NEWS.md | 2 + R/build.R | 2 + R/check-built.R | 27 +++++++++ R/check.R | 84 +++++++++++++++++++--------- R/sitrep.R | 47 ---------------- man/check_pkgdown.Rd | 19 +++++-- man/pkgdown_sitrep.Rd | 18 ------ pkgdown/_pkgdown.yml | 2 - tests/testthat/_snaps/check-built.md | 11 ++++ tests/testthat/_snaps/check.md | 61 +++++++++++++------- tests/testthat/_snaps/sitrep.md | 29 ---------- tests/testthat/test-check-built.R | 20 +++++++ tests/testthat/test-check.R | 49 +++++++--------- tests/testthat/test-sitrep.R | 11 ---- 14 files changed, 196 insertions(+), 186 deletions(-) create mode 100644 R/check-built.R delete mode 100644 R/sitrep.R delete mode 100644 man/pkgdown_sitrep.Rd create mode 100644 tests/testthat/_snaps/check-built.md delete mode 100644 tests/testthat/_snaps/sitrep.md create mode 100644 tests/testthat/test-check-built.R delete mode 100644 tests/testthat/test-sitrep.R diff --git a/NEWS.md b/NEWS.md index 601d80cff..1ff95d09d 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,5 +1,7 @@ # pkgdown (development version) +* `check_pkgdown()` and `pkgdown_sitrep()` have been unified so that they both report on the same problems. They now only differ in the style of their output: `pkgdown_sitrep()` reports whether each category is ok or not ok, while `check_pkgdown()` errors on the first issue (#2463). +* `build_site()` automatically runs `pkgdown_sitrep()` at the start of the process (#2380). * `build_home()` no longer checks if the README is missing any images. This check is now performed in `build_site()`, after `build_articles()` so you can refer to images created by vignettes with warnings (#2194). * `build_home()` now includes the contents of `inst/AUTHORS` on the authors page (#2506). * If you put a dropdown menu (e.g. articles) on the right hand side of the navbar, it will now be right aligned. This makes longer titles more likely to stay on the page (#2421). diff --git a/R/build.R b/R/build.R index 84a061c5b..2c5f5e9cf 100644 --- a/R/build.R +++ b/R/build.R @@ -443,6 +443,8 @@ build_site_local <- function(pkg = ".", cli::cli_inform("Reading from: {src_path(path_abs(pkg$src_path))}") cli::cli_inform("Writing to: {dst_path(path_abs(pkg$dst_path))}") + pkgdown_sitrep(pkg) + init_site(pkg) build_home(pkg, override = override, preview = FALSE) diff --git a/R/check-built.R b/R/check-built.R new file mode 100644 index 000000000..e0ae66440 --- /dev/null +++ b/R/check-built.R @@ -0,0 +1,27 @@ + +check_built_site <- function(pkg = ".") { + pkg <- as_pkgdown(pkg) + + cli::cli_rule("Checking for problems") + index_path <- path_index(pkg) + if (!is.null(index_path)) { + check_missing_images(pkg, index_path, "index.html") + } +} + +check_missing_images <- function(pkg, src_path, dst_path) { + html <- xml2::read_html(path(pkg$dst_path, dst_path), encoding = "UTF-8") + src <- xml2::xml_attr(xml2::xml_find_all(html, ".//img"), "src") + + rel_src <- src[xml2::url_parse(src)$scheme == ""] + rel_path <- fs::path_norm(path(fs::path_dir(dst_path), rel_src)) + exists <- fs::file_exists(path(pkg$dst_path, rel_path)) + + if (any(!exists)) { + paths <- rel_src[!exists] + cli::cli_warn(c( + "Missing images in {.file {path_rel(src_path, pkg$src_path)}}: {.file {paths}}", + i = "pkgdown can only use images in {.file man/figures} and {.file vignettes}" + )) + } +} diff --git a/R/check.R b/R/check.R index 4a5b57bfa..a7d7e7c96 100644 --- a/R/check.R +++ b/R/check.R @@ -1,20 +1,27 @@ #' Check `_pkgdown.yml` #' #' @description -#' Check that your `_pkgdown.yml` is valid without building the whole -#' site. Currently this: +#' This pair of functions checks that your `_pkgdown.yml` is valid without +#' building the whole site. `check_pkgdown()` errors at the first problem; +#' `pkgdown_sitrep()` reports the status of all checks. +#' +#' Currently they check that: #' -#' * Checks the reference and article indexes to ensure that pkgdown can -#' read them, and that every documentation topic and vignette/article is -#' included in the index. +#' * There's a `url` in the pkgdown configuration, which is also recorded +#' in the `URL` field of the `DESCRIPTION`. #' -#' * Validates any opengraph metadata that you might have supplied +#' * All opengraph metadata is valid. #' +#' * All reference topics are included in the index. +#' +#' * All articles/vignettes are included in the index. +# #' @export #' @inheritParams as_pkgdown check_pkgdown <- function(pkg = ".") { pkg <- as_pkgdown(pkg) + check_urls(pkg) data_open_graph(pkg) data_articles_index(pkg) data_reference_index(pkg) @@ -22,29 +29,56 @@ check_pkgdown <- function(pkg = ".") { cli::cli_inform(c("v" = "No problems found.")) } -check_built_site <- function(pkg = ".") { - pkg <- as_pkgdown(pkg) +#' @export +#' @rdname check_pkgdown +pkgdown_sitrep <- function(pkg = ".") { + cli::cli_rule("Sitrep") - cli::cli_rule("Checking for problems") - index_path <- path_index(pkg) - if (!is.null(index_path)) { - check_missing_images(pkg, index_path, "index.html") - } + error_to_sitrep("Package structure", pkg <- as_pkgdown(pkg)) + error_to_sitrep("URLs", check_urls(pkg)) + error_to_sitrep("Open graph metadata", data_open_graph(pkg)) + error_to_sitrep("Articles metadata", data_articles_index(pkg)) + error_to_sitrep("Reference metadata", data_reference_index(pkg)) } -check_missing_images <- function(pkg, src_path, dst_path) { - html <- xml2::read_html(path(pkg$dst_path, dst_path), encoding = "UTF-8") - src <- xml2::xml_attr(xml2::xml_find_all(html, ".//img"), "src") +error_to_sitrep <- function(title, code) { + tryCatch( + { + code + cli::cli_inform(c("v" = "{title} ok.")) + }, + rlang_error = function(e) { + bullets <- c(cnd_header(e), cnd_body(e)) + cli::cli_inform(c(x = "{title} not ok.", set_names(bullets, " "))) + } + ) + invisible() +} - rel_src <- src[xml2::url_parse(src)$scheme == ""] - rel_path <- fs::path_norm(path(fs::path_dir(dst_path), rel_src)) - exists <- fs::file_exists(path(pkg$dst_path, rel_path)) +check_urls <- function(pkg = ".", call = caller_env()) { + pkg <- as_pkgdown(pkg) + url <- pkg$meta[["url"]] - if (any(!exists)) { - paths <- rel_src[!exists] - cli::cli_warn(c( - "Missing images in {.file {path_rel(src_path, pkg$src_path)}}: {.file {paths}}", - i = "pkgdown can only use images in {.file man/figures} and {.file vignettes}" - )) + if (is.null(url)) { + cli::cli_abort( + c( + x = "{config_path(pkg)} lacks {.field url}.", + i = "See details in {.vignette pkgdown::metadata}." + ), + call = call + ) + } else { + desc_urls <- pkg$desc$get_urls() + desc_urls <- sub("/$", "", desc_urls) + + if (!pkg$meta[["url"]] %in% desc_urls) { + cli::cli_abort( + c( + x = "{.file DESCRIPTION} {.field URL} lacks package url ({url}).", + i = "See details in {.vignette pkgdown::metadata}." + ), + call = call + ) + } } } diff --git a/R/sitrep.R b/R/sitrep.R deleted file mode 100644 index 74505da9f..000000000 --- a/R/sitrep.R +++ /dev/null @@ -1,47 +0,0 @@ -#' Report package pkgdown situation -#' -#' @description -#' -#' `pkgdown_sitrep()` reports -#' -#' * If there is an `url` field in the pkgdown configuration; -#' -#' * If that pkgdown website URL is stored in the DESCRIPTION file. -#' -#' @inheritParams as_pkgdown -#' -#' @export -#' -pkgdown_sitrep <- function(pkg = ".") { - pkg <- as_pkgdown(pkg) - warns <- c() - - if (is.null(pkg$meta[["url"]])) { - warns <- c( - warns, - x = "{.field url} is absent. See {.vignette pkgdown::metadata}." - ) - } - - desc_urls <- pkg$desc$get_urls() - desc_urls <- sub("/$", "", desc_urls) - if (length(desc_urls) == 0 || !pkg$meta[["url"]] %in% desc_urls) { - warns <- c(warns, x = "{.file DESCRIPTION} {.field URL} is empty.") - } - - if (length(warns) == 0) { - cli::cli_inform(c( - "v" = "pkgdown situation report: {.emph {cli::col_green('all clear')}}", - "!" = "{.emph Double-check the following URLs:}", - " " = "{config_path(pkg)} contains URL {.url {pkg$meta['url']}}", - " " = "{.file DESCRIPTION} contains URL{?s} {.url {desc_urls}}" - )) - } else { - cli::cli_warn(c( - "pkgdown situation report: {.emph {cli::col_red('configuration error')}}", - warns - )) - } - - invisible() -} diff --git a/man/check_pkgdown.Rd b/man/check_pkgdown.Rd index d049c9439..3854e8ca6 100644 --- a/man/check_pkgdown.Rd +++ b/man/check_pkgdown.Rd @@ -2,20 +2,27 @@ % Please edit documentation in R/check.R \name{check_pkgdown} \alias{check_pkgdown} +\alias{pkgdown_sitrep} \title{Check \verb{_pkgdown.yml}} \usage{ check_pkgdown(pkg = ".") + +pkgdown_sitrep(pkg = ".") } \arguments{ \item{pkg}{Path to package.} } \description{ -Check that your \verb{_pkgdown.yml} is valid without building the whole -site. Currently this: +This pair of functions checks that your \verb{_pkgdown.yml} is valid without +building the whole site. \code{check_pkgdown()} errors at the first problem; +\code{pkgdown_sitrep()} reports the status of all checks. + +Currently they check that: \itemize{ -\item Checks the reference and article indexes to ensure that pkgdown can -read them, and that every documentation topic and vignette/article is -included in the index. -\item Validates any opengraph metadata that you might have supplied +\item There's a \code{url} in the pkgdown configuration, which is also recorded +in the \code{URL} field of the \code{DESCRIPTION}. +\item All opengraph metadata is valid. +\item All reference topics are included in the index. +\item All articles/vignettes are included in the index. } } diff --git a/man/pkgdown_sitrep.Rd b/man/pkgdown_sitrep.Rd deleted file mode 100644 index 90ada0334..000000000 --- a/man/pkgdown_sitrep.Rd +++ /dev/null @@ -1,18 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/sitrep.R -\name{pkgdown_sitrep} -\alias{pkgdown_sitrep} -\title{Report package pkgdown situation} -\usage{ -pkgdown_sitrep(pkg = ".") -} -\arguments{ -\item{pkg}{Path to package.} -} -\description{ -\code{pkgdown_sitrep()} reports -\itemize{ -\item If there is an \code{url} field in the pkgdown configuration; -\item If that pkgdown website URL is stored in the DESCRIPTION file. -} -} diff --git a/pkgdown/_pkgdown.yml b/pkgdown/_pkgdown.yml index 1cecac508..15c045b2f 100644 --- a/pkgdown/_pkgdown.yml +++ b/pkgdown/_pkgdown.yml @@ -1,4 +1,3 @@ -url: https://pkgdown.r-lib.org home: title: Build websites for R packages @@ -34,7 +33,6 @@ articles: contents: - customise - linking - - search - metadata - title: Advanced techniques diff --git a/tests/testthat/_snaps/check-built.md b/tests/testthat/_snaps/check-built.md new file mode 100644 index 000000000..0f750c920 --- /dev/null +++ b/tests/testthat/_snaps/check-built.md @@ -0,0 +1,11 @@ +# warn about missing images in readme + + Code + check_built_site(pkg) + Message + -- Checking for problems ------------------------------------------------------- + Condition + Warning: + Missing images in 'README.md': 'articles/kitten.jpg' + i pkgdown can only use images in 'man/figures' and 'vignettes' + diff --git a/tests/testthat/_snaps/check.md b/tests/testthat/_snaps/check.md index a5479cafc..28ed10291 100644 --- a/tests/testthat/_snaps/check.md +++ b/tests/testthat/_snaps/check.md @@ -1,37 +1,60 @@ -# fails if reference index incomplete +# sitrep reports all problems Code - check_pkgdown(pkg) - Condition - Error in `check_pkgdown()`: - ! 1 topic missing from index: "?". - i Either use `@keywords internal` to drop from index, or - i Edit _pkgdown.yml to fix the problem. - -# fails if article index incomplete + pkgdown_sitrep(pkg) + Message + -- Sitrep ---------------------------------------------------------------------- + v Package structure ok. + x URLs not ok. + 'DESCRIPTION' URL lacks package url (http://test.org). + See details in `vignette(pkgdown::metadata)`. + v Open graph metadata ok. + v Articles metadata ok. + x Reference metadata not ok. + 1 topic missing from index: "?". + Either use `@keywords internal` to drop from index, or + Edit _pkgdown.yml to fix the problem. + +# checks fails on first problem Code check_pkgdown(pkg) Condition Error in `check_pkgdown()`: - ! 2 vignettes missing from index: "articles/nested" and "width". - i Edit _pkgdown.yml to fix the problem. + x 'DESCRIPTION' URL lacks package url (http://test.org). + i See details in `vignette(pkgdown::metadata)`. -# informs if everything is ok +# both inform if everything is ok + Code + pkgdown_sitrep(pkg) + Message + -- Sitrep ---------------------------------------------------------------------- + v Package structure ok. + v URLs ok. + v Open graph metadata ok. + v Articles metadata ok. + v Reference metadata ok. Code check_pkgdown(pkg) Message v No problems found. -# warn about missing images in readme +# check_urls reports problems Code - check_built_site(pkg) - Message - -- Checking for problems ------------------------------------------------------- + check_urls(pkg) + Condition + Error: + x _pkgdown.yml lacks url. + i See details in `vignette(pkgdown::metadata)`. + +--- + + Code + check_urls(pkg) Condition - Warning: - Missing images in 'README.md': 'articles/kitten.jpg' - i pkgdown can only use images in 'man/figures' and 'vignettes' + Error: + x 'DESCRIPTION' URL lacks package url (https://testpackage.r-lib.org). + i See details in `vignette(pkgdown::metadata)`. diff --git a/tests/testthat/_snaps/sitrep.md b/tests/testthat/_snaps/sitrep.md deleted file mode 100644 index 60db03167..000000000 --- a/tests/testthat/_snaps/sitrep.md +++ /dev/null @@ -1,29 +0,0 @@ -# pkgdown_sitrep works - - Code - pkgdown_sitrep(pkg) - Condition - Warning: - pkgdown situation report: configuration error - x url is absent. See `vignette(pkgdown::metadata)`. - x 'DESCRIPTION' URL is empty. - ---- - - Code - pkgdown_sitrep(pkg) - Condition - Warning: - pkgdown situation report: configuration error - x 'DESCRIPTION' URL is empty. - ---- - - Code - pkgdown_sitrep(pkg) - Message - v pkgdown situation report: all clear - ! Double-check the following URLs: - _pkgdown.yml contains URL - 'DESCRIPTION' contains URL - diff --git a/tests/testthat/test-check-built.R b/tests/testthat/test-check-built.R new file mode 100644 index 000000000..29a6170be --- /dev/null +++ b/tests/testthat/test-check-built.R @@ -0,0 +1,20 @@ +test_that("warn about missing images in readme", { + pkg <- local_pkgdown_site(test_path("assets/bad-images")) + suppressMessages(build_home(pkg)) + + expect_snapshot(check_built_site(pkg)) +}) + +test_that("readme can use images from vignettes", { + pkg <- local_pkgdown_site(test_path("assets/bad-images")) + file_copy( + test_path("assets/articles-images/man/figures/kitten.jpg"), + path(pkg$src_path, "vignettes/kitten.jpg") + ) + withr::defer(unlink(path(pkg$src_path, "vignettes/kitten.jpg"))) + + suppressMessages(build_home(pkg)) + suppressMessages(build_articles(pkg)) + + suppressMessages(expect_no_warning(check_built_site(pkg))) +}) diff --git a/tests/testthat/test-check.R b/tests/testthat/test-check.R index 4a85d4bca..4d451f1fa 100644 --- a/tests/testthat/test-check.R +++ b/tests/testthat/test-check.R @@ -1,46 +1,37 @@ -test_that("fails if reference index incomplete", { +test_that("sitrep reports all problems", { pkg <- local_pkgdown_site(test_path("assets/reference"), meta = " reference: - title: Title contents: [a, b, c, e] ") - expect_snapshot(check_pkgdown(pkg), error = TRUE) + expect_snapshot(pkgdown_sitrep(pkg)) }) - -test_that("fails if article index incomplete", { - pkg <- local_pkgdown_site(test_path("assets/articles"), meta = " - articles: +test_that("checks fails on first problem", { + pkg <- local_pkgdown_site(test_path("assets/reference"), meta = " + reference: - title: Title - contents: [starts_with('html'), random, standard, toc-false, widget, needs-escape] + contents: [a, b, c, e] ") expect_snapshot(check_pkgdown(pkg), error = TRUE) }) -test_that("informs if everything is ok", { - pkg <- local_pkgdown_site(test_path("assets/reference")) - expect_snapshot(check_pkgdown(pkg)) -}) - -# built site --------------------------------------------------------------- - -test_that("warn about missing images in readme", { - pkg <- local_pkgdown_site(test_path("assets/bad-images")) - suppressMessages(build_home(pkg)) - - expect_snapshot(check_built_site(pkg)) +test_that("both inform if everything is ok", { + pkg <- test_path("assets/open-graph") + expect_snapshot({ + pkgdown_sitrep(pkg) + check_pkgdown(pkg) + }) }) -test_that("readme can use images from vignettes", { - pkg <- local_pkgdown_site(test_path("assets/bad-images")) - file_copy( - test_path("assets/articles-images/man/figures/kitten.jpg"), - path(pkg$src_path, "vignettes/kitten.jpg") - ) - withr::defer(unlink(path(pkg$src_path, "vignettes/kitten.jpg"))) +# check urls ------------------------------------------------------------------ - suppressMessages(build_home(pkg)) - suppressMessages(build_articles(pkg)) +test_that("check_urls reports problems", { + # URL not in the pkgdown config + pkg <- test_path("assets/figure") + expect_snapshot(check_urls(pkg), error = TRUE) - suppressMessages(expect_no_warning(check_built_site(pkg))) + # URL only in the pkgdown config + pkg <- test_path("assets/cname") + expect_snapshot(check_urls(pkg), error = TRUE) }) diff --git a/tests/testthat/test-sitrep.R b/tests/testthat/test-sitrep.R deleted file mode 100644 index b710b89a8..000000000 --- a/tests/testthat/test-sitrep.R +++ /dev/null @@ -1,11 +0,0 @@ -test_that("pkgdown_sitrep works", { - # URL not in the pkgdown config - pkg <- test_path("assets/figure") - expect_snapshot(pkgdown_sitrep(pkg)) - # URL only in the pkgdown config - pkg <- test_path("assets/cname") - expect_snapshot(pkgdown_sitrep(pkg)) - # URL everywhere - pkg <- test_path("assets/open-graph") - expect_snapshot(pkgdown_sitrep(pkg)) -})