Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

New rep_len_linter #2286

Merged
merged 13 commits into from
Nov 19, 2023
1 change: 1 addition & 0 deletions DESCRIPTION
Original file line number Diff line number Diff line change
Expand Up @@ -154,6 +154,7 @@ Collate:
'redundant_equals_linter.R'
'redundant_ifelse_linter.R'
'regex_subset_linter.R'
'rep_len_linter.R'
'repeat_linter.R'
'routine_registration_linter.R'
'sample_int_linter.R'
Expand Down
1 change: 1 addition & 0 deletions NAMESPACE
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,7 @@ export(quotes_linter)
export(redundant_equals_linter)
export(redundant_ifelse_linter)
export(regex_subset_linter)
export(rep_len_linter)
export(repeat_linter)
export(routine_registration_linter)
export(sample_int_linter)
Expand Down
1 change: 1 addition & 0 deletions NEWS.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
* `stopifnot_all_linter()` discourages tests with `all()` like `stopifnot(all(x > 0))`; `stopifnot()` runs `all()` itself, and uses a better error message (part of #884, @MichaelChirico).
* `comparison_negation_linter()` for discouraging negated comparisons when a direct negation is preferable, e.g. `!(x == y)` could be `x != y` (part of #884, @MichaelChirico).
* `terminal_close_linter()` for discouraging using `close()` to end functions (part of #884, @MichaelChirico). Such usages are not robust to errors, where `close()` will not be run as intended. Put `close()` in an `on.exit()` hook, or use {withr} to manage connections with proper cleanup.
* `rep_len_linter()` for encouraging use of `rep_len()` directly instead of `rep(x, length.out = n)` (part of #884, @MichaelChirico).

### Lint accuracy fixes: removing false positives

Expand Down
36 changes: 36 additions & 0 deletions R/rep_len_linter.R
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
#' Require usage of rep_len(x, n) over rep(x, length.out = n)
#'
#' `rep(x, length.out = n)` calls `rep_len(x, n)` "under the hood". The latter
#' is thus more direct and equally readable.
#'
MichaelChirico marked this conversation as resolved.
Show resolved Hide resolved
#' @examples
#' # will produce lints
#' lint(
#' text = "rep(1:3, length.out = 10)",
#' linters = rep_len_linter()
#' )
#'
#' # okay
#' lint(
#' text = "rep(1:3, each = 2L, length.out = 10L)",
IndrajeetPatil marked this conversation as resolved.
Show resolved Hide resolved
#' linters = rep_len_linter()
#' )
#'
#' @evalRd rd_tags("rep_len_linter")
#' @seealso [linters] for a complete list of linters available in lintr.
#' @export
rep_len_linter <- make_linter_from_xpath(
# count(expr) is for cases using positional matching; see ?rep.
xpath = "
//SYMBOL_FUNCTION_CALL[text() = 'rep']
/parent::expr
/parent::expr[
(
SYMBOL_SUB[text() = 'length.out']
or (not(SYMBOL_SUB) and count(expr) = 4)
)
and not(SYMBOL_SUB[text() = 'each'] or count(expr) = 5)
]
",
lint_message = "Use rep_len(x, n) instead of rep(x, length.out = n)."
)
1 change: 1 addition & 0 deletions inst/lintr/linters.csv
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ quotes_linter,style consistency readability default configurable
redundant_equals_linter,best_practices readability efficiency common_mistakes
redundant_ifelse_linter,best_practices efficiency consistency configurable
regex_subset_linter,best_practices efficiency
rep_len_linter,readability consistency best_practices
repeat_linter,style readability
routine_registration_linter,best_practices efficiency robustness
sample_int_linter,efficiency readability robustness
Expand Down
1 change: 1 addition & 0 deletions man/best_practices_linters.Rd

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

1 change: 1 addition & 0 deletions man/consistency_linters.Rd

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

7 changes: 4 additions & 3 deletions man/linters.Rd

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

1 change: 1 addition & 0 deletions man/readability_linters.Rd

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

32 changes: 32 additions & 0 deletions man/rep_len_linter.Rd

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

36 changes: 36 additions & 0 deletions tests/testthat/test-rep_len_linter.R
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
test_that("rep_len_linter skips allowed usages", {
linter <- rep_len_linter()

# only catch length.out usages
expect_lint("rep(x, y)", NULL, linter)
expect_lint("rep(1:10, 2)", NULL, linter)
expect_lint("rep(1:10, 10:1)", NULL, linter)

# usage of each is not compatible with rep_len; see ?rep.
expect_lint("rep(x, each = 4, length.out = 50)", NULL, linter)
# each is implicitly the 4th positional argument. a very strange usage
# (because length.out is ignored), but doesn't hurt to catch it
expect_lint("rep(a, b, length.out = c, d)", NULL, linter)
# ditto for implicit length.out=
expect_lint("rep(a, b, c, d)", NULL, linter)
})

test_that("rep_len_linter blocks simple disallowed usages", {
linter <- rep_len_linter()
lint_msg <- rex::rex("Use rep_len(x, n) instead of rep(x, length.out = n).")

# only catch length.out usages
expect_lint("rep(x, length.out = 4L)", lint_msg, linter)

# implicit times= argument; length.out has priority over times= (see ?rep),
# so we still lint since it's as if times= is not supplied.
# (notice here that the base behavior is odd -- one might expect output like
# head(rep(1:10, 10:1), 50), but instead we get rep(1:10, length.out = 50))
expect_lint("rep(1:10, 10:1, length.out = 50)", lint_msg, linter)

# ditto for explicit times= argument
expect_lint("rep(1:10, times = 10:1, length.out = 50)", lint_msg, linter)

# implicit usage in third argument
expect_lint("rep(1:10, 10:1, 50)", lint_msg, linter)
})
Loading