From 0cc7b57ef12a0c6a015878d3d983568d0b898a97 Mon Sep 17 00:00:00 2001 From: Ricardo Rodrigo Basa Date: Thu, 5 Sep 2024 06:07:45 +0200 Subject: [PATCH] Add box.linters styling functions. add check for treesitter dependencies (#607) --- DESCRIPTION | 6 +-- NEWS.md | 3 ++ R/dependencies.R | 3 ++ R/tools.R | 52 +++++++++++++++++++-- man/format_r.Rd | 24 +++++++++- man/lint_r.Rd | 4 ++ tests/e2e/test-lint-r.R | 2 + vignettes/explanation/rhino-style-guide.Rmd | 39 ++++++++++++++++ 8 files changed, 125 insertions(+), 8 deletions(-) diff --git a/DESCRIPTION b/DESCRIPTION index efafee12..d02db3f4 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -1,6 +1,6 @@ Package: rhino Title: A Framework for Enterprise Shiny Applications -Version: 1.9.0.9000 +Version: 1.9.0.9001 Authors@R: c( person("Kamil", "Żyła", role = c("aut", "cre"), email = "opensource+kamil@appsilon.com"), @@ -19,12 +19,12 @@ BugReports: https://github.com/Appsilon/rhino/issues License: LGPL-3 Encoding: UTF-8 Roxygen: list(markdown = TRUE) -RoxygenNote: 7.3.1 +RoxygenNote: 7.3.2 Depends: R (>= 2.10) Imports: box (>= 1.1.3), - box.linters (>= 0.9.1), + box.linters (>= 0.10.4), cli, config, fs, diff --git a/NEWS.md b/NEWS.md index b108da0e..74acf05e 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,5 +1,8 @@ # rhino (development version) +* Integrated {box.linters} styling functions to style `box::use()` calls according to the Rhino style guide. +* Added compatibility check for `treesitter` and `treesitter.r` dependencies + # [rhino 1.9.0](https://github.com/Appsilon/rhino/releases/tag/v1.9.0) See _[How-to: Rhino 1.9 Migration Guide](https://appsilon.github.io/rhino/articles/how-to/migrate-1-9.html)_ diff --git a/R/dependencies.R b/R/dependencies.R index 151d1c06..c802037f 100644 --- a/R/dependencies.R +++ b/R/dependencies.R @@ -12,6 +12,9 @@ read_dependencies <- function() { } write_dependencies <- function(deps) { + if (as.numeric(R.Version()$major) >= 4 && as.numeric(R.Version()$minor) >= 3.0) { + deps <- c(deps, "treesitter", "treesitter.r") + } deps <- sort(unique(c("rhino", deps))) # Rhino is always needed as a dependency. deps <- purrr::map_chr(deps, function(name) glue::glue("library({name})")) deps <- c( diff --git a/R/tools.R b/R/tools.R index 3d5b5646..0330f5a0 100644 --- a/R/tools.R +++ b/R/tools.R @@ -79,6 +79,10 @@ check_paths <- function(paths) { #' with the `legacy_max_lint_r_errors` option in `rhino.yml`. #' This can be useful when inheriting legacy code with multiple styling issues. #' +#' The [box.linters::namespaced_function_calls()] linter requires the `{treesitter}` and +#' `{treesitter.r}` packages. These require R >= 4.3.0. `lint_r()` will continue to run and skip +#' `namespaced_function_calls()` if its dependencies are not available. +#' #' @param paths Character vector of directories and files to lint. #' When `NULL` (the default), check `app` and `tests/testthat` directories. #' @@ -87,6 +91,17 @@ check_paths <- function(paths) { #' @export # nolint end lint_r <- function(paths = NULL) { + if (!box.linters::is_treesitter_installed()) { + cli::cli_warn( + c( + "!" = paste( + "`box.linters::namespaced_function_calls()` requires {{treesitter}} and {{treesitter.r}}", + "to be installed." + ), + "i" = "`lintr_r()` will continue to run using the other linter functions." + ) + ) + } if (is.null(paths)) { paths <- c("app", "tests/testthat") } @@ -122,12 +137,30 @@ rhino_style <- function() { #' Format R #' -#' Uses the `{styler}` package to automatically format R sources. +#' Uses the `{styler}` and `{box.linters}` packages to automatically format R sources. As with +#' `styler`, carefully examine the results after running this function. #' #' The code is formatted according to the `styler::tidyverse_style` guide with one adjustment: #' spacing around math operators is not modified to avoid conflicts with `box::use()` statements. #' +#' If available, `box::use()` calls are reformatted by styling functions provided by +#' `{box.linters}`. These include: +#' +#' * Separating `box::use()` calls for packages and local modules +#' * Alphabetically sorting packages, modules, and functions. +#' * Adding trailing commas +#' +#' `box.linters::style_*` functions require the `treesitter` and `treesitter.r` packages. These, in +#' turn, require R >= 4.3.0. `format_r()` will continue to operate without these but will not +#' perform `box::use()` call styling. +#' +#' For more information on `box::use()` call styling please refer to the `{box.linters}` styling +#' functions +#' [documentation](https://appsilon.github.io/box.linters/reference/style_box_use_text.html). +#' #' @param paths Character vector of files and directories to format. +#' @param exclude_files Character vector with regular expressions of files that should be excluded +#' from styling. #' @return None. This function is called for side effects. #' #' @examples @@ -139,11 +172,24 @@ rhino_style <- function() { #' format_r("app/view") #' } #' @export -format_r <- function(paths) { +format_r <- function(paths, exclude_files = NULL) { + style_box_use <- box.linters::style_box_use_dir + if (!box.linters::is_treesitter_installed()) { + style_box_use <- function(path, exclude_files) { } + cli::cli_warn( + c( + "x" = "The packages {{treesitter}} and {{treesitter.r}} are required by `box::use()` styling features of `format_r()`.", #nolint + "i" = "These package require R version >= 4.3.0 to install." + ) + ) + } + for (path in paths) { if (fs::is_dir(path)) { - styler::style_dir(path, style = rhino_style) + style_box_use(path, exclude_files = exclude_files) + styler::style_dir(path, style = rhino_style, exclude_files = exclude_files) } else { + style_box_use(path, exclude_files = exclude_files) styler::style_file(path, style = rhino_style) } } diff --git a/man/format_r.Rd b/man/format_r.Rd index c4d00340..51e82c9c 100644 --- a/man/format_r.Rd +++ b/man/format_r.Rd @@ -4,20 +4,40 @@ \alias{format_r} \title{Format R} \usage{ -format_r(paths) +format_r(paths, exclude_files = NULL) } \arguments{ \item{paths}{Character vector of files and directories to format.} + +\item{exclude_files}{Character vector with regular expressions of files that should be excluded +from styling.} } \value{ None. This function is called for side effects. } \description{ -Uses the \code{{styler}} package to automatically format R sources. +Uses the \code{{styler}} and \code{{box.linters}} packages to automatically format R sources. As with +\code{styler}, carefully examine the results after running this function. } \details{ The code is formatted according to the \code{styler::tidyverse_style} guide with one adjustment: spacing around math operators is not modified to avoid conflicts with \code{box::use()} statements. + +If available, \code{box::use()} calls are reformatted by styling functions provided by +\code{{box.linters}}. These include: +\itemize{ +\item Separating \code{box::use()} calls for packages and local modules +\item Alphabetically sorting packages, modules, and functions. +\item Adding trailing commas +} + +\verb{box.linters::style_*} functions require the \code{treesitter} and \code{treesitter.r} packages. These, in +turn, require R >= 4.3.0. \code{format_r()} will continue to operate without these but will not +perform \code{box::use()} call styling. + +For more information on \code{box::use()} call styling please refer to the \code{{box.linters}} styling +functions +\href{https://appsilon.github.io/box.linters/reference/style_box_use_text.html}{documentation}. } \examples{ if (interactive()) { diff --git a/man/lint_r.Rd b/man/lint_r.Rd index 5a348725..9c6336a9 100644 --- a/man/lint_r.Rd +++ b/man/lint_r.Rd @@ -24,4 +24,8 @@ in the \code{.lintr} file. You can set the maximum number of accepted style errors with the \code{legacy_max_lint_r_errors} option in \code{rhino.yml}. This can be useful when inheriting legacy code with multiple styling issues. + +The \code{\link[box.linters:namespaced_function_calls]{box.linters::namespaced_function_calls()}} linter requires the \code{{treesitter}} and +\code{{treesitter.r}} packages. These require R >= 4.3.0. \code{lint_r()} will continue to run and skip +\code{namespaced_function_calls()} if its dependencies are not available. } diff --git a/tests/e2e/test-lint-r.R b/tests/e2e/test-lint-r.R index da3ff49a..7fa42a91 100644 --- a/tests/e2e/test-lint-r.R +++ b/tests/e2e/test-lint-r.R @@ -1,3 +1,5 @@ +install.packages(c("treesitter", "treesitter.r")) + rhino::lint_r() # Create bad scripts and test if formatting returns the expected result test_file_path <- fs::path("app", "logic", "bad-style.R") diff --git a/vignettes/explanation/rhino-style-guide.Rmd b/vignettes/explanation/rhino-style-guide.Rmd index c3baa44d..9071aed2 100644 --- a/vignettes/explanation/rhino-style-guide.Rmd +++ b/vignettes/explanation/rhino-style-guide.Rmd @@ -129,3 +129,42 @@ box::use( shiny[div, fluidPage, navbarPage, sidebarPanel, sidebarLayout, mainPanel, tabPanel, tabsetPanel, titlePanel], ) ``` + +# Automated Styling of `box::use()` calls + +As of Rhino version 1.10.0, `format_r()` includes styling for `box::use()` calls. This is provided by [`{box.linters}`](https://appsilon.github.io/box.linters/) version >= 0.10.4. As with [`{styler}`](https://styler.r-lib.org/), carefully examine the styling results after performing automated styling on your code. + +Automated styling covers some of the topics described above such as: + +* Separating `box::use()` calls for packages and local modules +* Alphabetically sorting packages, modules, and functions. +* Adding trailing commas + +```r +# Original +box::use(stringr[str_trim, str_pad], dplyr, app/logic/table) + +# Styled +box::use( + dplyr, + stringr[str_pad, str_trim], +) + +box::use( + app/logic/table +) +``` + +## Requirements + +* R version >= 4.3.0 +* `box.linters` version >= 0.10.4 +* `treesitter` package +* `treesitter.r` package + +## How to use + +1. `box::use()` call styling is included when running `format_r()`. +2. It can also be separately executed by running one of the [`box.linters::style_*`](https://appsilon.github.io/box.linters/reference/index.html#box-styling) functions. + +For more information on the abilities of `{{box.linters}}` styling functions refer to the styling functions [documentation](https://appsilon.github.io/box.linters/reference/style_box_use_text.html).