From ac416f31c049f575d6e3efc6409a77ccfe407bcc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kirill=20M=C3=BCller?= Date: Sun, 22 Oct 2023 16:24:14 +0200 Subject: [PATCH 1/7] Add ref and host to standalone header --- R/use_standalone.R | 12 +++++++++--- tests/testthat/_snaps/use_standalone.md | 2 +- 2 files changed, 10 insertions(+), 4 deletions(-) diff --git a/R/use_standalone.R b/R/use_standalone.R index 73e845ae5..dc5ce748a 100644 --- a/R/use_standalone.R +++ b/R/use_standalone.R @@ -73,7 +73,7 @@ use_standalone <- function(repo_spec, file = NULL, ref = NULL, host = NULL) { dest_path <- path("R", as_standalone_dest_file(file)) lines <- read_github_file(repo_spec, path = src_path, ref = ref, host = host) - lines <- c(standalone_header(repo_spec, src_path), lines) + lines <- c(standalone_header(repo_spec, src_path, ref, host), lines) write_over(proj_path(dest_path), lines, overwrite = TRUE) dependencies <- standalone_dependencies(lines, path) @@ -155,10 +155,16 @@ as_standalone_dest_file <- function(file) { gsub("standalone-", "import-standalone-", file) } -standalone_header <- function(repo_spec, path) { +standalone_header <- function(repo_spec, path, ref = NULL, host = NULL) { + if (is.null(ref)) { + ref <- "HEAD" + } + if (is.null(host)) { + host <- "https://github.com" + } c( "# Standalone file: do not edit by hand", - glue("# Source: "), + glue("# Source: <{host}/{repo_spec}/blob/{ref}/{path}>"), paste0("# ", strrep("-", 72 - 2)), "#" ) diff --git a/tests/testthat/_snaps/use_standalone.md b/tests/testthat/_snaps/use_standalone.md index 9b1e3532e..c843ef1b8 100644 --- a/tests/testthat/_snaps/use_standalone.md +++ b/tests/testthat/_snaps/use_standalone.md @@ -18,7 +18,7 @@ standalone_header("r-lib/usethis", "R/standalone-test.R") Output [1] "# Standalone file: do not edit by hand" - [2] "# Source: " + [2] "# Source: " [3] "# ----------------------------------------------------------------------" [4] "#" From 1fa63fd63d1984c6400fcd0c24c27c49900c4fd9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kirill=20M=C3=BCller?= Date: Sun, 22 Oct 2023 16:39:24 +0200 Subject: [PATCH 2/7] Add code to generate the standalone --- R/use_standalone.R | 1 + tests/testthat/_snaps/use_standalone.md | 9 +++++---- 2 files changed, 6 insertions(+), 4 deletions(-) diff --git a/R/use_standalone.R b/R/use_standalone.R index dc5ce748a..63ae1489b 100644 --- a/R/use_standalone.R +++ b/R/use_standalone.R @@ -165,6 +165,7 @@ standalone_header <- function(repo_spec, path, ref = NULL, host = NULL) { c( "# Standalone file: do not edit by hand", glue("# Source: <{host}/{repo_spec}/blob/{ref}/{path}>"), + glue('# Generated by: usethis::use_standalone("{repo_spec}", "{basename(path)}", "{ref}", "{host}")'), paste0("# ", strrep("-", 72 - 2)), "#" ) diff --git a/tests/testthat/_snaps/use_standalone.md b/tests/testthat/_snaps/use_standalone.md index c843ef1b8..6433d5be6 100644 --- a/tests/testthat/_snaps/use_standalone.md +++ b/tests/testthat/_snaps/use_standalone.md @@ -17,10 +17,11 @@ Code standalone_header("r-lib/usethis", "R/standalone-test.R") Output - [1] "# Standalone file: do not edit by hand" - [2] "# Source: " - [3] "# ----------------------------------------------------------------------" - [4] "#" + [1] "# Standalone file: do not edit by hand" + [2] "# Source: " + [3] "# Generated by: usethis::use_standalone(\"r-lib/usethis\", \"standalone-test.R\", \"HEAD\", \"https://github.com\")" + [4] "# ----------------------------------------------------------------------" + [5] "#" # can extract imports From 6e37a778575c78485629e877fdaab210c896dd5a Mon Sep 17 00:00:00 2001 From: Jenny Bryan Date: Tue, 23 Jul 2024 22:34:07 -0700 Subject: [PATCH 3/7] Make some more changes --- R/use_standalone.R | 21 ++++--- tests/testthat/_snaps/use_standalone.md | 77 +++++++++++++++++++++---- tests/testthat/test-use_standalone.R | 35 +++++++++-- 3 files changed, 110 insertions(+), 23 deletions(-) diff --git a/R/use_standalone.R b/R/use_standalone.R index aab36a630..af1e1b9e1 100644 --- a/R/use_standalone.R +++ b/R/use_standalone.R @@ -157,16 +157,21 @@ as_standalone_dest_file <- function(file) { } standalone_header <- function(repo_spec, path, ref = NULL, host = NULL) { - if (is.null(ref)) { - ref <- "HEAD" - } - if (is.null(host)) { - host <- "https://github.com" - } + ref_string <- ref %||% "HEAD" + host_string <- host %||% "https://github.com" + source_comment <- + glue("# Source: {host_string}/{repo_spec}/blob/{ref_string}/{path}") + + path_string <- path_ext_remove(sub("^standalone-", "", basename(path))) + ref_string <- if (is.null(ref)) "" else glue(', ref = "{ref}"') + host_string <- if (is.null(host) || host == "https://github.com") "" else glue(', host = "{host}"') + code_hint <- glue('usethis::use_standalone("{repo_spec}", "{path_string}"{ref_string}{host_string})') + generated_comment <- glue('# Generated by: {code_hint}') + c( "# Standalone file: do not edit by hand", - glue("# Source: <{host}/{repo_spec}/blob/{ref}/{path}>"), - glue('# Generated by: usethis::use_standalone("{repo_spec}", "{basename(path)}", "{ref}", "{host}")'), + source_comment, + generated_comment, paste0("# ", strrep("-", 72 - 2)), "#" ) diff --git a/tests/testthat/_snaps/use_standalone.md b/tests/testthat/_snaps/use_standalone.md index 6433d5be6..9abf59eef 100644 --- a/tests/testthat/_snaps/use_standalone.md +++ b/tests/testthat/_snaps/use_standalone.md @@ -1,3 +1,69 @@ +# standalone_header() works with various inputs + + Code + standalone_header("OWNER/REPO", "R/standalone-foo.R") + Output + [1] "# Standalone file: do not edit by hand" + [2] "# Source: https://github.com/OWNER/REPO/blob/HEAD/R/standalone-foo.R" + [3] "# Generated by: usethis::use_standalone(\"OWNER/REPO\", \"foo\")" + [4] "# ----------------------------------------------------------------------" + [5] "#" + +--- + + Code + standalone_header("OWNER/REPO", "R/standalone-foo.R", ref = "blah") + Output + [1] "# Standalone file: do not edit by hand" + [2] "# Source: https://github.com/OWNER/REPO/blob/blah/R/standalone-foo.R" + [3] "# Generated by: usethis::use_standalone(\"OWNER/REPO\", \"foo\", ref = \"blah\")" + [4] "# ----------------------------------------------------------------------" + [5] "#" + +--- + + Code + standalone_header("OWNER/REPO", "R/standalone-foo.R", ref = "blah", host = "https://github.com") + Output + [1] "# Standalone file: do not edit by hand" + [2] "# Source: https://github.com/OWNER/REPO/blob/blah/R/standalone-foo.R" + [3] "# Generated by: usethis::use_standalone(\"OWNER/REPO\", \"foo\", ref = \"blah\")" + [4] "# ----------------------------------------------------------------------" + [5] "#" + +--- + + Code + standalone_header("OWNER/REPO", "R/standalone-foo.R", host = "https://github.acme.com") + Output + [1] "# Standalone file: do not edit by hand" + [2] "# Source: https://github.acme.com/OWNER/REPO/blob/HEAD/R/standalone-foo.R" + [3] "# Generated by: usethis::use_standalone(\"OWNER/REPO\", \"foo\", host = \"https://github.acme.com\")" + [4] "# ----------------------------------------------------------------------" + [5] "#" + +--- + + Code + standalone_header("OWNER/REPO", "R/standalone-foo.R", ref = "blah", host = "https://github.com") + Output + [1] "# Standalone file: do not edit by hand" + [2] "# Source: https://github.com/OWNER/REPO/blob/blah/R/standalone-foo.R" + [3] "# Generated by: usethis::use_standalone(\"OWNER/REPO\", \"foo\", ref = \"blah\")" + [4] "# ----------------------------------------------------------------------" + [5] "#" + +--- + + Code + standalone_header("OWNER/REPO", "R/standalone-foo.R", ref = "blah", host = "https://github.acme.com") + Output + [1] "# Standalone file: do not edit by hand" + [2] "# Source: https://github.acme.com/OWNER/REPO/blob/blah/R/standalone-foo.R" + [3] "# Generated by: usethis::use_standalone(\"OWNER/REPO\", \"foo\", ref = \"blah\", host = \"https://github.acme.com\")" + [4] "# ----------------------------------------------------------------------" + [5] "#" + # can offer choices Code @@ -12,17 +78,6 @@ ! `file` is absent, but must be supplied. i Possible options are cli, downstream-deps, lazyeval, lifecycle, linked-version, obj-type, purrr, rlang, s3-register, sizes, types-check, vctrs, or zeallot. -# header provides useful summary - - Code - standalone_header("r-lib/usethis", "R/standalone-test.R") - Output - [1] "# Standalone file: do not edit by hand" - [2] "# Source: " - [3] "# Generated by: usethis::use_standalone(\"r-lib/usethis\", \"standalone-test.R\", \"HEAD\", \"https://github.com\")" - [4] "# ----------------------------------------------------------------------" - [5] "#" - # can extract imports Code diff --git a/tests/testthat/test-use_standalone.R b/tests/testthat/test-use_standalone.R index 638411c27..41d9337e7 100644 --- a/tests/testthat/test-use_standalone.R +++ b/tests/testthat/test-use_standalone.R @@ -1,3 +1,34 @@ +test_that("standalone_header() works with various inputs", { + expect_snapshot( + standalone_header("OWNER/REPO", "R/standalone-foo.R") + ) + expect_snapshot( + standalone_header("OWNER/REPO", "R/standalone-foo.R", ref = "blah") + ) + expect_snapshot( + standalone_header( + "OWNER/REPO", "R/standalone-foo.R", host = "https://github.com" + ) + ) + expect_snapshot( + standalone_header( + "OWNER/REPO", "R/standalone-foo.R", host = "https://github.acme.com" + ) + ) + expect_snapshot( + standalone_header( + "OWNER/REPO", "R/standalone-foo.R", + ref = "blah", host = "https://github.com" + ) + ) + expect_snapshot( + standalone_header( + "OWNER/REPO", "R/standalone-foo.R", + ref = "blah", host = "https://github.acme.com" + ) + ) +}) + test_that("can import standalone file with dependencies", { skip_if_offline("github.com") create_local_package() @@ -40,10 +71,6 @@ test_that("can offer choices", { }) }) -test_that("header provides useful summary", { - expect_snapshot(standalone_header("r-lib/usethis", "R/standalone-test.R")) -}) - test_that("can extract dependencies", { extract_deps <- function(deps) { out <- standalone_dependencies(c("# ---", deps, "# ---"), "test.R") From afdaded41cffe143c838c045c8c0f43af5f6cfc2 Mon Sep 17 00:00:00 2001 From: Jenny Bryan Date: Tue, 23 Jul 2024 22:34:26 -0700 Subject: [PATCH 4/7] Dogfood it --- R/import-standalone-obj-type.R | 29 ++++++++++++++++++++++------- R/import-standalone-types-check.R | 3 ++- 2 files changed, 24 insertions(+), 8 deletions(-) diff --git a/R/import-standalone-obj-type.R b/R/import-standalone-obj-type.R index 72cfe37dc..47268d620 100644 --- a/R/import-standalone-obj-type.R +++ b/R/import-standalone-obj-type.R @@ -1,17 +1,27 @@ # Standalone file: do not edit by hand -# Source: +# Source: https://github.com/r-lib/rlang/blob/HEAD/R/standalone-obj-type.R +# Generated by: usethis::use_standalone("r-lib/rlang", "obj-type") # ---------------------------------------------------------------------- # # --- # repo: r-lib/rlang # file: standalone-obj-type.R -# last-updated: 2022-10-04 +# last-updated: 2024-02-14 # license: https://unlicense.org # imports: rlang (>= 1.1.0) # --- # # ## Changelog # +# 2024-02-14: +# - `obj_type_friendly()` now works for S7 objects. +# +# 2023-05-01: +# - `obj_type_friendly()` now only displays the first class of S3 objects. +# +# 2023-03-30: +# - `stop_input_type()` now handles `I()` input literally in `arg`. +# # 2022-10-04: # - `obj_type_friendly(value = TRUE)` now shows numeric scalars # literally. @@ -65,7 +75,7 @@ obj_type_friendly <- function(x, value = TRUE) { if (inherits(x, "quosure")) { type <- "quosure" } else { - type <- paste(class(x), collapse = "/") + type <- class(x)[[1L]] } return(sprintf("a <%s> object", type)) } @@ -261,19 +271,19 @@ vec_type_friendly <- function(x, length = FALSE) { #' Return OO type #' @param x Any R object. #' @return One of `"bare"` (for non-OO objects), `"S3"`, `"S4"`, -#' `"R6"`, or `"R7"`. +#' `"R6"`, or `"S7"`. #' @noRd obj_type_oo <- function(x) { if (!is.object(x)) { return("bare") } - class <- inherits(x, c("R6", "R7_object"), which = TRUE) + class <- inherits(x, c("R6", "S7_object"), which = TRUE) if (class[[1]]) { "R6" } else if (class[[2]]) { - "R7" + "S7" } else if (isS4(x)) { "S4" } else { @@ -315,10 +325,15 @@ stop_input_type <- function(x, if (length(what)) { what <- oxford_comma(what) } + if (inherits(arg, "AsIs")) { + format_arg <- identity + } else { + format_arg <- cli$format_arg + } message <- sprintf( "%s must be %s, not %s.", - cli$format_arg(arg), + format_arg(arg), what, obj_type_friendly(x, value = show_value) ) diff --git a/R/import-standalone-types-check.R b/R/import-standalone-types-check.R index 6782d69b1..0943a8d58 100644 --- a/R/import-standalone-types-check.R +++ b/R/import-standalone-types-check.R @@ -1,5 +1,6 @@ # Standalone file: do not edit by hand -# Source: +# Source: https://github.com/r-lib/rlang/blob/HEAD/R/standalone-types-check.R +# Generated by: usethis::use_standalone("r-lib/rlang", "types-check") # ---------------------------------------------------------------------- # # --- From 9fc645787d56c4c8ba5c98dae07db62ee169fe4c Mon Sep 17 00:00:00 2001 From: Jenny Bryan Date: Tue, 23 Jul 2024 22:36:48 -0700 Subject: [PATCH 5/7] Add NEWS bullet --- NEWS.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/NEWS.md b/NEWS.md index ac43cca6a..7b5a0b3a3 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,5 +1,8 @@ # usethis (development version) +* `use_standalone()` inserts an improved header that includes the code needed to + update the standalone file (@krlmlr, #1903). + * `use_release_issue()` and `use_upkeep()` behave better when the user has a fork. The user is asked just once to choose between `origin` and `upstream` as the target repo (#2023). From 4cd2e6c7386624fa42cf9a9de6236e69f5a10915 Mon Sep 17 00:00:00 2001 From: Jenny Bryan Date: Tue, 23 Jul 2024 22:43:39 -0700 Subject: [PATCH 6/7] Use fs --- R/use_standalone.R | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/R/use_standalone.R b/R/use_standalone.R index af1e1b9e1..e23d1cba7 100644 --- a/R/use_standalone.R +++ b/R/use_standalone.R @@ -162,7 +162,7 @@ standalone_header <- function(repo_spec, path, ref = NULL, host = NULL) { source_comment <- glue("# Source: {host_string}/{repo_spec}/blob/{ref_string}/{path}") - path_string <- path_ext_remove(sub("^standalone-", "", basename(path))) + path_string <- path_ext_remove(sub("^standalone-", "", path_file(path))) ref_string <- if (is.null(ref)) "" else glue(', ref = "{ref}"') host_string <- if (is.null(host) || host == "https://github.com") "" else glue(', host = "{host}"') code_hint <- glue('usethis::use_standalone("{repo_spec}", "{path_string}"{ref_string}{host_string})') From 0d32cfa957afdb0afb86c9cedd5f5908a59e5f8d Mon Sep 17 00:00:00 2001 From: Jenny Bryan Date: Tue, 23 Jul 2024 22:46:50 -0700 Subject: [PATCH 7/7] Update a snapshot --- tests/testthat/_snaps/use_standalone.md | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/tests/testthat/_snaps/use_standalone.md b/tests/testthat/_snaps/use_standalone.md index 9abf59eef..1b1a447b6 100644 --- a/tests/testthat/_snaps/use_standalone.md +++ b/tests/testthat/_snaps/use_standalone.md @@ -23,13 +23,13 @@ --- Code - standalone_header("OWNER/REPO", "R/standalone-foo.R", ref = "blah", host = "https://github.com") + standalone_header("OWNER/REPO", "R/standalone-foo.R", host = "https://github.com") Output - [1] "# Standalone file: do not edit by hand" - [2] "# Source: https://github.com/OWNER/REPO/blob/blah/R/standalone-foo.R" - [3] "# Generated by: usethis::use_standalone(\"OWNER/REPO\", \"foo\", ref = \"blah\")" - [4] "# ----------------------------------------------------------------------" - [5] "#" + [1] "# Standalone file: do not edit by hand" + [2] "# Source: https://github.com/OWNER/REPO/blob/HEAD/R/standalone-foo.R" + [3] "# Generated by: usethis::use_standalone(\"OWNER/REPO\", \"foo\")" + [4] "# ----------------------------------------------------------------------" + [5] "#" ---