lint_r_code.R 5.05 KB
Newer Older
1
2
3
4
5
6

library(lintr)

args <- commandArgs(
    trailingOnly = TRUE
)
7
SOURCE_DIR <- args[[1L]]
8
9
10

FILES_TO_LINT <- list.files(
    path = SOURCE_DIR
11
    , pattern = "\\.r$|\\.rmd$"
12
13
14
15
16
17
18
    , all.files = TRUE
    , ignore.case = TRUE
    , full.names = TRUE
    , recursive = TRUE
    , include.dirs = FALSE
)

19
20
21
22
23
24
25
26
27
28
29
30
# text to use for pipe operators from packages like 'magrittr'
pipe_text <- paste0(
    "For consistency and the sake of being explicit, this project's code "
    , "does not use the pipe operator."
)

# text to use for functions that should only be called interactively
interactive_text <- paste0(
    "Functions like '?', 'help', and 'install.packages()' should only be used "
    , "interactively, not in package code."
)

31
LINTERS_TO_USE <- list(
32
    "absolute_path"          = lintr::absolute_path_linter()
33
34
    , "any_duplicated"       = lintr::any_duplicated_linter()
    , "any_is_na"            = lintr::any_is_na_linter()
35
36
    , "assignment"           = lintr::assignment_linter()
    , "braces"               = lintr::brace_linter()
37
    , "class_equals_linter"  = lintr::class_equals_linter()
38
39
40
41
42
    , "commas"               = lintr::commas_linter()
    , "equals_na"            = lintr::equals_na_linter()
    , "function_left"        = lintr::function_left_parentheses_linter()
    , "implicit_integers"    = lintr::implicit_integer_linter()
    , "infix_spaces"         = lintr::infix_spaces_linter()
43
44
    , "inner_combine"        = lintr::inner_combine_linter()
    , "literal_coercion"     = lintr::literal_coercion_linter()
45
    , "long_lines"           = lintr::line_length_linter(length = 120L)
46
    , "missing_argument"     = lintr::missing_argument_linter()
47
48
    , "no_tabs"              = lintr::no_tab_linter()
    , "non_portable_path"    = lintr::nonportable_path_linter()
49
50
51
52
    , "numeric_leading_zero" = lintr::numeric_leading_zero_linter()
    , "outer_negation"       = lintr::outer_negation_linter()
    , "package_hooks"        = lintr::package_hooks_linter()
    , "regex_subset"         = lintr::regex_subset_linter()
53
54
55
56
57
    , "semicolon"            = lintr::semicolon_linter()
    , "seq"                  = lintr::seq_linter()
    , "single_quotes"        = lintr::single_quotes_linter()
    , "spaces_inside"        = lintr::spaces_inside_linter()
    , "spaces_left_parens"   = lintr::spaces_left_parentheses_linter()
58
    , "sprintf"              = lintr::sprintf_linter()
59
    , "todo_comments"        = lintr::todo_comment_linter(c("todo", "fixme", "to-do"))
60
61
62
    , "trailing_blank"       = lintr::trailing_blank_lines_linter()
    , "trailing_white"       = lintr::trailing_whitespace_linter()
    , "true_false"           = lintr::T_and_F_symbol_linter()
63
64
    , "unreachable_code"     = lintr::unreachable_code_linter()
    , "vector_logic"         = lintr::vector_logic_linter()
65
66
    , "undesirable_function" = lintr::undesirable_function_linter(
        fun = c(
67
68
            "cat" = "CRAN forbids the use of cat() in packages except in special cases. Use message() or warning()."
            , "cbind" = paste0(
69
70
71
                "cbind is an unsafe way to build up a data frame. merge() or direct "
                , "column assignment is preferred."
            )
72
73
            , "dyn.load" = "Directly loading or unloading .dll or .so files in package code should not be necessary."
            , "dyn.unload" = "Directly loading or unloading .dll or .so files in package code should not be necessary."
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
            , "help" = interactive_text
            , "ifelse" = "The use of ifelse() is dangerous because it will silently allow mixing types."
            , "install.packages" = interactive_text
            , "is.list" = paste0(
                "This project uses data.table, and is.list(x) is TRUE for a data.table. "
                , "identical(class(x), 'list') is a safer way to check that something is an R list object."
            )
            , "rbind" = "data.table::rbindlist() is faster and safer than rbind(), and is preferred in this project."
            , "require" = paste0(
                "library() is preferred to require() because it will raise an error immediately "
                , "if a package is missing."
            )
        )
    )
    , "undesirable_operator" = lintr::undesirable_operator_linter(
        op = c(
            "%>%" = pipe_text
            , "%.%" = pipe_text
            , "%..%" = pipe_text
            , "?" = interactive_text
            , "??" = interactive_text
        )
    )
97
    , "unneeded_concatenation" = lintr::unneeded_concatenation_linter()
98
99
)

100
noquote(paste0(length(FILES_TO_LINT), " R files need linting"))
101

102
results <- NULL
103

104
for (r_file in FILES_TO_LINT) {
105
106
107
108
109
110
111

    this_result <- lintr::lint(
        filename = r_file
        , linters = LINTERS_TO_USE
        , cache = FALSE
    )

112
113
114
115
116
117
118
119
    print(
        sprintf(
            "Found %i linting errors in %s"
            , length(this_result)
            , r_file
        )
        , quote = FALSE
    )
120
121
122
123
124
125
126

    results <- c(results, this_result)

}

issues_found <- length(results)

127
128
noquote(paste0("Total linting issues found: ", issues_found))

129
if (issues_found > 0L) {
130
131
132
133
    print(results)
}

quit(save = "no", status = issues_found)