A portable configuration for the ‘lintr’ R package that includes a handful of custom linters that I commonly use.
Last active
July 11, 2024 08:52
-
-
Save klmr/86752f3bee5ee35048eb90b6c7b2bb7c to your computer and use it in GitHub Desktop.
Alternative R linters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| arrow_assignment_linter = function () { | |
| xpath = '//LEFT_ASSIGN[text() != ":="] | //RIGHT_ASSIGN' | |
| lint_message_fmt = 'Use =, not %s, for assignment.' | |
| lintr::Linter(\(source_expression) { | |
| if (! lintr::is_lint_level(source_expression, 'expression')) { | |
| return(list()) | |
| } | |
| xml = source_expression$xml_parsed_content | |
| bad_expr = xml2::xml_find_all(xml, xpath) | |
| if (length(bad_expr) == 0L) { | |
| return(list()) | |
| } | |
| operator = xml2::xml_text(bad_expr) | |
| lint_message = sprintf(lint_message_fmt, operator) | |
| lintr::xml_nodes_to_lints(bad_expr, source_expression, lint_message, type = 'style') | |
| }) | |
| } |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| linters: { | |
| config_dir = dirname(file) | |
| lapply(dir(config_dir, '\\.[rR]$', full.names = TRUE), source, local = environment()) | |
| lintr::linters_with_defaults( | |
| assignment_linter = arrow_assignment_linter(), | |
| function_left_parentheses_linter = function_definition_linter(), | |
| line_length_linter = line_length_linter(120L), | |
| object_usage_linter = NULL, # unusably buggy | |
| quotes_linter = lintr::quotes_linter("'") | |
| ) | |
| } |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| function_definition_linter = function () { | |
| bad_line_fun_xpath = '(//FUNCTION | //OP-LAMBDA)[@line1 != following-sibling::OP-LEFT-PAREN/@line1]' | |
| bad_line_call_xpath = '//SYMBOL_FUNCTION_CALL[@line1 != parent::expr/following-sibling::OP-LEFT-PAREN/@line1]' | |
| bad_col_fun_xpath = '//FUNCTION[ | |
| @line1 = following-sibling::OP-LEFT-PAREN/@line1 | |
| and @col2 != following-sibling::OP-LEFT-PAREN/@col1 - 2 | |
| ]' | |
| bad_col_call_xpath = '//SYMBOL_FUNCTION_CALL[ | |
| line1 = parent::expr/following-sibling::OP-LEFT-PAREN/@line1 | |
| and @col2 != parent::expr/following-sibling::OP-LEFT-PAREN/@col1 - 1 | |
| ]' | |
| lintr::Linter(\(source_expression) { | |
| if (! lintr::is_lint_level(source_expression, 'expression')) { | |
| return(list()) | |
| } | |
| xml = source_expression$xml_parsed_content | |
| bad_line_fun_exprs = xml2::xml_find_all(xml, bad_line_fun_xpath) | |
| bad_line_fun_lints = lintr::xml_nodes_to_lints( | |
| bad_line_fun_exprs, | |
| source_expression = source_expression, | |
| lint_message = 'Left parenthesis should be on the same line as the \'function\' symbol.' | |
| ) | |
| bad_line_call_exprs = xml2::xml_find_all(xml, bad_line_call_xpath) | |
| bad_line_call_lints = lintr::xml_nodes_to_lints( | |
| bad_line_call_exprs, | |
| source_expression = source_expression, | |
| lint_message = 'Left parenthesis should be on the same line as the function\'s symbol.' | |
| ) | |
| bad_col_fun_exprs = xml2::xml_find_all(xml, bad_col_fun_xpath) | |
| bad_col_fun_lints = lintr::xml_nodes_to_lints( | |
| bad_col_fun_exprs, | |
| source_expression = source_expression, | |
| lint_message = 'Add spaces before the left parenthesis in a function definition.', | |
| range_start_xpath = 'number(./@col2 + 1)', | |
| range_end_xpath = 'number(./following-sibling::OP-LEFT-PAREN/@col1)' | |
| ) | |
| bad_col_call_exprs = xml2::xml_find_all(xml, bad_col_call_xpath) | |
| bad_col_call_lints = lintr::xml_nodes_to_lints( | |
| bad_col_call_exprs, | |
| source_expression = source_expression, | |
| lint_message = 'Remove spaces before the left parenthesis in a function call.', | |
| range_start_xpath = 'number(./@col2 + 1)', | |
| range_end_xpath = 'number(./parent::expr/following-sibling::OP-LEFT-PAREN/@col1 - 1)' | |
| ) | |
| c(bad_line_fun_lints, bad_line_call_lints, bad_col_fun_lints, bad_col_call_lints) | |
| }) | |
| } |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| line_length_linter = function (length = 120L) { | |
| general_msg = paste('Lines should not be more than', length, 'characters.') | |
| lintr::Linter(\(source_expression) { | |
| if (! lintr::is_lint_level(source_expression, 'file')) { | |
| return(list()) | |
| } | |
| # Note that this will handle “comment-looking” lines in multi-line strings incorrectly. But that’s fine. | |
| comment_lines = grep('^\\s*#', source_expression$file_lines) | |
| line_lengths = nchar(source_expression$file_lines) | |
| long_lines = setdiff(which(line_lengths > length), comment_lines) | |
| Map( | |
| \(long_line, line_length) { | |
| lintr::Lint( | |
| filename = source_expression$filename, | |
| line_number = long_line, | |
| column_number = length + 1L, | |
| type = 'style', | |
| message = paste(general_msg, 'This line is', line_length, 'characters.'), | |
| line = source_expression$file_lines[long_line], | |
| ranges = list(c(1L, line_length)) | |
| ) | |
| }, | |
| long_lines, | |
| line_lengths[long_lines] | |
| ) | |
| }) | |
| } |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment