]> git.lizzy.rs Git - rust.git/blob - clippy_dev/src/main.rs
Auto merge of #4099 - flip1995:ul_4094, r=oli-obk
[rust.git] / clippy_dev / src / main.rs
1 extern crate clap;
2 extern crate clippy_dev;
3 extern crate regex;
4
5 use clap::{App, Arg, SubCommand};
6 use clippy_dev::*;
7 mod stderr_length_check;
8
9 #[derive(PartialEq)]
10 enum UpdateMode {
11     Check,
12     Change,
13 }
14
15 fn main() {
16     let matches = App::new("Clippy developer tooling")
17         .subcommand(
18             SubCommand::with_name("update_lints")
19                 .about("Updates lint registration and information from the source code")
20                 .long_about(
21                     "Makes sure that:\n \
22                      * the lint count in README.md is correct\n \
23                      * the changelog contains markdown link references at the bottom\n \
24                      * all lint groups include the correct lints\n \
25                      * lint modules in `clippy_lints/*` are visible in `src/lib.rs` via `pub mod`\n \
26                      * all lints are registered in the lint store",
27                 )
28                 .arg(Arg::with_name("print-only").long("print-only").help(
29                     "Print a table of lints to STDOUT. \
30                      This does not include deprecated and internal lints. \
31                      (Does not modify any files)",
32                 ))
33                 .arg(
34                     Arg::with_name("check")
35                         .long("check")
36                         .help("Checks that util/dev update_lints has been run. Used on CI."),
37                 ),
38         )
39         .arg(
40             Arg::with_name("limit-stderr-length")
41                 .long("limit-stderr-length")
42                 .help("Ensures that stderr files do not grow longer than a certain amount of lines."),
43         )
44         .get_matches();
45
46     if matches.is_present("limit-stderr-length") {
47         stderr_length_check::check();
48     }
49     if let Some(matches) = matches.subcommand_matches("update_lints") {
50         if matches.is_present("print-only") {
51             print_lints();
52         } else if matches.is_present("check") {
53             update_lints(&UpdateMode::Check);
54         } else {
55             update_lints(&UpdateMode::Change);
56         }
57     }
58 }
59
60 fn print_lints() {
61     let lint_list = gather_all();
62     let usable_lints: Vec<Lint> = Lint::usable_lints(lint_list).collect();
63     let lint_count = usable_lints.len();
64     let grouped_by_lint_group = Lint::by_lint_group(&usable_lints);
65
66     for (lint_group, mut lints) in grouped_by_lint_group {
67         if lint_group == "Deprecated" {
68             continue;
69         }
70         println!("\n## {}", lint_group);
71
72         lints.sort_by_key(|l| l.name.clone());
73
74         for lint in lints {
75             println!(
76                 "* [{}]({}#{}) ({})",
77                 lint.name,
78                 clippy_dev::DOCS_LINK.clone(),
79                 lint.name,
80                 lint.desc
81             );
82         }
83     }
84
85     println!("there are {} lints", lint_count);
86 }
87
88 fn update_lints(update_mode: &UpdateMode) {
89     let lint_list: Vec<Lint> = gather_all().collect();
90     let usable_lints: Vec<Lint> = Lint::usable_lints(lint_list.clone().into_iter()).collect();
91     let lint_count = usable_lints.len();
92
93     let mut file_change = replace_region_in_file(
94         "../README.md",
95         r#"\[There are \d+ lints included in this crate!\]\(https://rust-lang.github.io/rust-clippy/master/index.html\)"#,
96         "",
97         true,
98         update_mode == &UpdateMode::Change,
99         || {
100             vec![
101                 format!("[There are {} lints included in this crate!](https://rust-lang.github.io/rust-clippy/master/index.html)", lint_count)
102             ]
103         }
104     ).changed;
105
106     file_change |= replace_region_in_file(
107         "../CHANGELOG.md",
108         "<!-- begin autogenerated links to lint list -->",
109         "<!-- end autogenerated links to lint list -->",
110         false,
111         update_mode == &UpdateMode::Change,
112         || gen_changelog_lint_list(lint_list.clone()),
113     )
114     .changed;
115
116     file_change |= replace_region_in_file(
117         "../clippy_lints/src/lib.rs",
118         "begin deprecated lints",
119         "end deprecated lints",
120         false,
121         update_mode == &UpdateMode::Change,
122         || gen_deprecated(&lint_list),
123     )
124     .changed;
125
126     file_change |= replace_region_in_file(
127         "../clippy_lints/src/lib.rs",
128         "begin lints modules",
129         "end lints modules",
130         false,
131         update_mode == &UpdateMode::Change,
132         || gen_modules_list(lint_list.clone()),
133     )
134     .changed;
135
136     // Generate lists of lints in the clippy::all lint group
137     file_change |= replace_region_in_file(
138         "../clippy_lints/src/lib.rs",
139         r#"reg.register_lint_group\("clippy::all""#,
140         r#"\]\);"#,
141         false,
142         update_mode == &UpdateMode::Change,
143         || {
144             // clippy::all should only include the following lint groups:
145             let all_group_lints = usable_lints
146                 .clone()
147                 .into_iter()
148                 .filter(|l| {
149                     l.group == "correctness" || l.group == "style" || l.group == "complexity" || l.group == "perf"
150                 })
151                 .collect();
152
153             gen_lint_group_list(all_group_lints)
154         },
155     )
156     .changed;
157
158     // Generate the list of lints for all other lint groups
159     for (lint_group, lints) in Lint::by_lint_group(&usable_lints) {
160         file_change |= replace_region_in_file(
161             "../clippy_lints/src/lib.rs",
162             &format!("reg.register_lint_group\\(\"clippy::{}\"", lint_group),
163             r#"\]\);"#,
164             false,
165             update_mode == &UpdateMode::Change,
166             || gen_lint_group_list(lints.clone()),
167         )
168         .changed;
169     }
170
171     if update_mode == &UpdateMode::Check && file_change {
172         println!(
173             "Not all lints defined properly. \
174              Please run `util/dev update_lints` to make sure all lints are defined properly."
175         );
176         std::process::exit(1);
177     }
178 }