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

Linter for explicit/implicit returns #2271

Merged
merged 47 commits into from
Nov 24, 2023
Merged
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
47 commits
Select commit Hold shift + click to select a range
bc9f197
feat: Add `return_linter`
MEO265 Nov 8, 2023
2a54612
feat: Dont lint deterministically returning control statements
MEO265 Nov 9, 2023
293e4a2
test: Accept false negatives
MEO265 Nov 9, 2023
842a006
feat: Do not lint stop
MEO265 Nov 9, 2023
cda5823
feat: Refined lint of `switch`
MEO265 Nov 9, 2023
c091d1f
test: Add line tests
MEO265 Nov 10, 2023
067a33d
doc: Mark as configurable
MEO265 Nov 10, 2023
94955ed
mnt: Add terminal new lines
MEO265 Nov 10, 2023
8a27435
Merge branch 'main' into feature/return_linter
MEO265 Nov 10, 2023
ae5304f
incorporate new test cases, improve lint message, begin work to recon…
MichaelChirico Nov 20, 2023
336e9a5
drop in the other XPath
MichaelChirico Nov 20, 2023
a9c8801
catch OP-LAMBDA, typos, fix lint metadata
MichaelChirico Nov 20, 2023
a4a18e3
remove vestigial, clean up repeated var usage
MichaelChirico Nov 20, 2023
23b0fdb
Merge remote-tracking branch 'origin/main' into feature/return_linter
MichaelChirico Nov 20, 2023
63741c3
progress on rectifying disagreements
MichaelChirico Nov 20, 2023
580b935
more progress, simplifying XPath
MichaelChirico Nov 20, 2023
48eaab0
more test+logic adjustments, now passing tests
MichaelChirico Nov 20, 2023
c6b11be
simplify implicit XPath
MichaelChirico Nov 20, 2023
ee5eed3
test code style
MichaelChirico Nov 20, 2023
0826721
Merge branch 'main' into feature/return_linter
MichaelChirico Nov 20, 2023
bafe717
add simple examples
MichaelChirico Nov 20, 2023
f58d408
set as a default linter
MichaelChirico Nov 20, 2023
514927d
test-defaults
MichaelChirico Nov 20, 2023
b3ac25f
style guide ref in doc
MichaelChirico Nov 20, 2023
f8d958d
finish TODOs
MichaelChirico Nov 21, 2023
de3e9ce
NEWS entry
MichaelChirico Nov 21, 2023
95edca4
Merge branch 'main' into feature/return_linter
MichaelChirico Nov 21, 2023
3810968
Merge branch 'main' into feature/return_linter
MichaelChirico Nov 21, 2023
9bb7d3a
fix merge
MichaelChirico Nov 21, 2023
414257b
feat: Add parameters for exceptions
MEO265 Nov 21, 2023
8da36f4
feat: Add parameter for Runit
MEO265 Nov 21, 2023
5d67b25
feat: Lint `warning`, `message`, and `stopifnot`
MEO265 Nov 21, 2023
0b95fc7
mnt: Add terminal newline to tests
MEO265 Nov 21, 2023
4b09754
doc: Fix doc of `additional_side_effect_func`
MEO265 Nov 21, 2023
ca945bc
Merge branch 'main' into feature/return_linter
MichaelChirico Nov 23, 2023
f7afa54
Merge branch 'feature/return_linter' of github.com:MEO265/lintr into …
MichaelChirico Nov 23, 2023
91820bd
drop runit support for now
MichaelChirico Nov 23, 2023
e1961d5
style
MichaelChirico Nov 23, 2023
5248af9
rename parameter to accept "implicit"/"explicit"
MichaelChirico Nov 23, 2023
8612b70
rename other parameters
MichaelChirico Nov 23, 2023
aff7765
corresponding changes to tests
MichaelChirico Nov 23, 2023
28b51f4
dont link R4.0+ tryInvokeRestart, which is in linked page already anyway
MichaelChirico Nov 23, 2023
cb2d9c4
Merge branch 'main' into feature/return_linter
MEO265 Nov 23, 2023
8a0ea77
Merge branch 'main' into MEO265-feature/return_linter
AshesITR Nov 23, 2023
63eba24
review and fixes
AshesITR Nov 23, 2023
f89c8bc
document()
AshesITR Nov 23, 2023
325205c
Merge branch 'main' into feature/return_linter
AshesITR Nov 23, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
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_ifelse_linter.R'
'regex_subset_linter.R'
'repeat_linter.R'
'return_linter.R'
'routine_registration_linter.R'
'scalar_in_linter.R'
'semicolon_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(redundant_equals_linter)
export(redundant_ifelse_linter)
export(regex_subset_linter)
export(repeat_linter)
export(return_linter)
export(routine_registration_linter)
export(sarif_output)
export(scalar_in_linter)
Expand Down
73 changes: 73 additions & 0 deletions R/return_linter.R
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
#' Return linter
#'
#' This linter checks for explicit [return()] at the end of a function
MichaelChirico marked this conversation as resolved.
Show resolved Hide resolved
#'
#' @param use_implicit_returns Whether to use implicit or explicit returns
#'
#' @evalRd rd_tags("return_linter")
#' @seealso [linters] for a complete list of linters available in lintr.
#'
#' @export
return_linter <- function(use_implicit_returns = TRUE) {

if (use_implicit_returns) {
xpath <- "
(//FUNCTION | //OP-LAMBDA)[following-sibling::expr[1]/*[1][self::OP-LEFT-BRACE]]
MichaelChirico marked this conversation as resolved.
Show resolved Hide resolved
/following-sibling::expr[1]/
expr[last()][
expr[1][
not(OP-DOLLAR or OP-AT)
and SYMBOL_FUNCTION_CALL[text() = 'return']
]
]
"
msg <- "An explicit return at the end of a function is not required"
} else {
xpath <- "
(//FUNCTION | //OP-LAMBDA)[following-sibling::expr[1]/*[1][self::OP-LEFT-BRACE]]
/following-sibling::expr[1]/
expr[last()][
expr[1][not(
(not(OP-DOLLAR or OP-AT) and SYMBOL_FUNCTION_CALL[text() = 'return' or text() = 'stop'])
)] and
expr[1][
not(
not(OP-DOLLAR or OP-AT) and SYMBOL_FUNCTION_CALL[text() = 'switch'] and
not(following-sibling::expr[position()>1][
not(descendant::SYMBOL_FUNCTION_CALL[text() = 'return' or text() = 'stop'])
]) and
following-sibling::expr[position()>1][preceding-sibling::*[1][self::OP-COMMA]]
)
] and
not(
REPEAT or IF[
following-sibling::ELSE[
following-sibling::expr[1][
descendant::SYMBOL_FUNCTION_CALL[text() = 'return' or text() = 'stop']
]
]
]
)
]
"
msg <- "An explicit return at the end of a function is desired"
}

Linter(function(source_expression) {
if (!is_lint_level(source_expression, "expression")) {
return(list())
}

xml <- source_expression$xml_parsed_content

xml_nodes <- xml_find_all(xml, xpath)

xml_nodes_to_lints(
xml_nodes,
source_expression = source_expression,
lint_message = msg,
type = "style"
)

})
}
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 @@ redundant_equals_linter,best_practices readability efficiency common_mistakes
redundant_ifelse_linter,best_practices efficiency consistency configurable
regex_subset_linter,best_practices efficiency
repeat_linter,style readability
return_linter,style configurable
routine_registration_linter,best_practices efficiency robustness
scalar_in_linter,readability consistency best_practices efficiency
semicolon_linter,style readability default configurable
Expand Down
1 change: 1 addition & 0 deletions man/configurable_linters.Rd

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

5 changes: 3 additions & 2 deletions man/linters.Rd

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

20 changes: 20 additions & 0 deletions man/return_linter.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/style_linters.Rd

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

Loading
Loading