rust/clippy_dev/src/main.rs

287 lines
9.4 KiB
Rust
Raw Normal View History

#![cfg_attr(feature = "deny-warnings", deny(warnings))]
use clap::{App, Arg, SubCommand};
use clippy_dev::*;
2019-06-24 23:43:38 -05:00
mod fmt;
mod new_lint;
mod stderr_length_check;
#[derive(PartialEq)]
enum UpdateMode {
Check,
2018-11-27 14:13:08 -06:00
Change,
}
fn main() {
let matches = App::new("Clippy developer tooling")
2019-06-24 23:43:38 -05:00
.subcommand(
SubCommand::with_name("fmt")
.about("Run rustfmt on all projects and tests")
.arg(
Arg::with_name("check")
.long("check")
.help("Use the rustfmt --check option"),
)
.arg(
Arg::with_name("verbose")
.short("v")
.long("verbose")
.help("Echo commands run"),
),
)
.subcommand(
SubCommand::with_name("update_lints")
.about("Updates lint registration and information from the source code")
.long_about(
2018-11-27 14:13:08 -06:00
"Makes sure that:\n \
* the lint count in README.md is correct\n \
* the changelog contains markdown link references at the bottom\n \
* all lint groups include the correct lints\n \
* lint modules in `clippy_lints/*` are visible in `src/lib.rs` via `pub mod`\n \
* all lints are registered in the lint store",
)
2018-11-27 14:13:08 -06:00
.arg(Arg::with_name("print-only").long("print-only").help(
"Print a table of lints to STDOUT. \
This does not include deprecated and internal lints. \
(Does not modify any files)",
))
.arg(
Arg::with_name("check")
.long("check")
.help("Checks that util/dev update_lints has been run. Used on CI."),
2018-11-27 14:13:08 -06:00
),
)
.subcommand(
SubCommand::with_name("new_lint")
.about("Create new lint and run util/dev update_lints")
.arg(
Arg::with_name("pass")
.short("p")
.long("pass")
.help("Specify whether the lint runs during the early or late pass")
.takes_value(true)
.possible_values(&["early", "late"])
.required(true),
)
.arg(
Arg::with_name("name")
.short("n")
.long("name")
.help("Name of the new lint in snake case, ex: fn_too_long")
.takes_value(true)
.required(true),
)
.arg(
Arg::with_name("category")
.short("c")
.long("category")
.help("What category the lint belongs to")
.default_value("nursery")
.possible_values(&[
"style",
"correctness",
"complexity",
"perf",
"pedantic",
"restriction",
"cargo",
"nursery",
"internal",
"internal_warn",
])
.takes_value(true),
),
)
.arg(
Arg::with_name("limit-stderr-length")
.long("limit-stderr-length")
.help("Ensures that stderr files do not grow longer than a certain amount of lines."),
)
2018-11-27 14:13:08 -06:00
.get_matches();
if matches.is_present("limit-stderr-length") {
stderr_length_check::check();
}
2019-06-24 23:43:38 -05:00
match matches.subcommand() {
("fmt", Some(matches)) => {
fmt::run(matches.is_present("check"), matches.is_present("verbose"));
2019-10-24 04:55:22 -05:00
},
2019-06-24 23:43:38 -05:00
("update_lints", Some(matches)) => {
if matches.is_present("print-only") {
print_lints();
} else if matches.is_present("check") {
update_lints(&UpdateMode::Check);
} else {
update_lints(&UpdateMode::Change);
}
2019-10-24 04:55:22 -05:00
},
("new_lint", Some(matches)) => {
match new_lint::create(
matches.value_of("pass"),
matches.value_of("name"),
matches.value_of("category"),
) {
Ok(_) => update_lints(&UpdateMode::Change),
Err(e) => eprintln!("Unable to create lint: {}", e),
}
},
2019-10-24 04:55:22 -05:00
_ => {},
}
}
fn print_lints() {
let lint_list = gather_all();
let usable_lints: Vec<Lint> = Lint::usable_lints(lint_list).collect();
let lint_count = usable_lints.len();
let grouped_by_lint_group = Lint::by_lint_group(&usable_lints);
for (lint_group, mut lints) in grouped_by_lint_group {
2018-11-27 14:13:08 -06:00
if lint_group == "Deprecated" {
continue;
}
println!("\n## {}", lint_group);
2018-10-15 14:02:38 -05:00
lints.sort_by_key(|l| l.name.clone());
for lint in lints {
2018-11-27 14:13:08 -06:00
println!(
"* [{}]({}#{}) ({})",
lint.name,
clippy_dev::DOCS_LINK.clone(),
lint.name,
lint.desc
);
}
}
println!("there are {} lints", lint_count);
}
2019-10-24 07:33:14 -05:00
#[allow(clippy::too_many_lines)]
2018-11-05 00:11:25 -06:00
fn update_lints(update_mode: &UpdateMode) {
let lint_list: Vec<Lint> = gather_all().collect();
2019-06-08 13:29:27 -05:00
let usable_lints: Vec<Lint> = Lint::usable_lints(lint_list.clone().into_iter()).collect();
let lint_count = usable_lints.len();
let mut sorted_usable_lints = usable_lints.clone();
sorted_usable_lints.sort_by_key(|lint| lint.name.clone());
let mut file_change = replace_region_in_file(
"../src/lintlist/mod.rs",
"begin lint list",
"end lint list",
false,
update_mode == &UpdateMode::Change,
|| {
format!(
"pub const ALL_LINTS: [Lint; {}] = {:#?};",
sorted_usable_lints.len(),
sorted_usable_lints
)
.lines()
.map(ToString::to_string)
.collect::<Vec<_>>()
},
)
.changed;
file_change |= replace_region_in_file(
"../README.md",
r#"\[There are \d+ lints included in this crate!\]\(https://rust-lang.github.io/rust-clippy/master/index.html\)"#,
"",
true,
2018-11-05 00:11:25 -06:00
update_mode == &UpdateMode::Change,
|| {
vec![
format!("[There are {} lints included in this crate!](https://rust-lang.github.io/rust-clippy/master/index.html)", lint_count)
]
}
).changed;
file_change |= replace_region_in_file(
"../CHANGELOG.md",
"<!-- begin autogenerated links to lint list -->",
"<!-- end autogenerated links to lint list -->",
false,
2018-11-05 00:11:25 -06:00
update_mode == &UpdateMode::Change,
2018-11-27 14:13:08 -06:00
|| gen_changelog_lint_list(lint_list.clone()),
)
.changed;
file_change |= replace_region_in_file(
"../clippy_lints/src/lib.rs",
"begin deprecated lints",
"end deprecated lints",
false,
2018-11-05 00:11:25 -06:00
update_mode == &UpdateMode::Change,
2018-11-27 14:13:08 -06:00
|| gen_deprecated(&lint_list),
)
.changed;
2019-10-11 07:33:42 -05:00
file_change |= replace_region_in_file(
"../clippy_lints/src/lib.rs",
"begin register lints",
"end register lints",
false,
update_mode == &UpdateMode::Change,
|| gen_register_lint_list(&lint_list),
)
.changed;
file_change |= replace_region_in_file(
"../clippy_lints/src/lib.rs",
"begin lints modules",
"end lints modules",
false,
2018-11-05 00:11:25 -06:00
update_mode == &UpdateMode::Change,
2018-11-27 14:13:08 -06:00
|| gen_modules_list(lint_list.clone()),
)
.changed;
// Generate lists of lints in the clippy::all lint group
file_change |= replace_region_in_file(
"../clippy_lints/src/lib.rs",
2019-10-11 07:33:42 -05:00
r#"store.register_group\(true, "clippy::all""#,
r#"\]\);"#,
false,
2018-11-05 00:11:25 -06:00
update_mode == &UpdateMode::Change,
|| {
// clippy::all should only include the following lint groups:
2018-11-27 14:13:08 -06:00
let all_group_lints = usable_lints
.clone()
.into_iter()
.filter(|l| {
l.group == "correctness" || l.group == "style" || l.group == "complexity" || l.group == "perf"
})
.collect();
gen_lint_group_list(all_group_lints)
2018-11-27 14:13:08 -06:00
},
)
.changed;
// Generate the list of lints for all other lint groups
for (lint_group, lints) in Lint::by_lint_group(&usable_lints) {
file_change |= replace_region_in_file(
"../clippy_lints/src/lib.rs",
2019-10-11 07:33:42 -05:00
&format!("store.register_group\\(true, \"clippy::{}\"", lint_group),
r#"\]\);"#,
false,
2018-11-05 00:11:25 -06:00
update_mode == &UpdateMode::Change,
2018-11-27 14:13:08 -06:00
|| gen_lint_group_list(lints.clone()),
)
.changed;
}
2018-11-05 00:11:25 -06:00
if update_mode == &UpdateMode::Check && file_change {
2018-11-27 14:13:08 -06:00
println!(
"Not all lints defined properly. \
Please run `util/dev update_lints` to make sure all lints are defined properly."
);
std::process::exit(1);
}
}