]> git.lizzy.rs Git - rust.git/blob - clippy_dev/src/bless.rs
Auto merge of #6513 - nahuakang:fix/empty_enum_lint_never_type, r=flip1995
[rust.git] / clippy_dev / src / bless.rs
1 //! `bless` updates the reference files in the repo with changed output files
2 //! from the last test run.
3
4 use std::env;
5 use std::ffi::OsStr;
6 use std::fs;
7 use std::lazy::SyncLazy;
8 use std::path::{Path, PathBuf};
9 use walkdir::WalkDir;
10
11 use crate::clippy_project_root;
12
13 // NOTE: this is duplicated with tests/cargo/mod.rs What to do?
14 pub static CARGO_TARGET_DIR: SyncLazy<PathBuf> = SyncLazy::new(|| match env::var_os("CARGO_TARGET_DIR") {
15     Some(v) => v.into(),
16     None => env::current_dir().unwrap().join("target"),
17 });
18
19 static CLIPPY_BUILD_TIME: SyncLazy<Option<std::time::SystemTime>> = SyncLazy::new(|| {
20     let profile = env::var("PROFILE").unwrap_or_else(|_| "debug".to_string());
21     let mut path = PathBuf::from(&**CARGO_TARGET_DIR);
22     path.push(profile);
23     path.push("cargo-clippy");
24     fs::metadata(path).ok()?.modified().ok()
25 });
26
27 pub fn bless(ignore_timestamp: bool) {
28     let test_suite_dirs = [
29         clippy_project_root().join("tests").join("ui"),
30         clippy_project_root().join("tests").join("ui-internal"),
31         clippy_project_root().join("tests").join("ui-toml"),
32         clippy_project_root().join("tests").join("ui-cargo"),
33     ];
34     for test_suite_dir in &test_suite_dirs {
35         WalkDir::new(test_suite_dir)
36             .into_iter()
37             .filter_map(Result::ok)
38             .filter(|f| f.path().extension() == Some(OsStr::new("rs")))
39             .for_each(|f| {
40                 let test_name = f.path().strip_prefix(test_suite_dir).unwrap();
41                 for &ext in &["stdout", "stderr", "fixed"] {
42                     update_reference_file(
43                         f.path().with_extension(ext),
44                         test_name.with_extension(ext),
45                         ignore_timestamp,
46                     );
47                 }
48             });
49     }
50 }
51
52 fn update_reference_file(reference_file_path: PathBuf, test_name: PathBuf, ignore_timestamp: bool) {
53     let test_output_path = build_dir().join(test_name);
54     let relative_reference_file_path = reference_file_path.strip_prefix(clippy_project_root()).unwrap();
55
56     // If compiletest did not write any changes during the test run,
57     // we don't have to update anything
58     if !test_output_path.exists() {
59         return;
60     }
61
62     // If the test output was not updated since the last clippy build, it may be outdated
63     if !ignore_timestamp && !updated_since_clippy_build(&test_output_path).unwrap_or(true) {
64         return;
65     }
66
67     let test_output_file = fs::read(&test_output_path).expect("Unable to read test output file");
68     let reference_file = fs::read(&reference_file_path).unwrap_or_default();
69
70     if test_output_file != reference_file {
71         // If a test run caused an output file to change, update the reference file
72         println!("updating {}", &relative_reference_file_path.display());
73         fs::copy(test_output_path, &reference_file_path).expect("Could not update reference file");
74
75         // We need to re-read the file now because it was potentially updated from copying
76         let reference_file = fs::read(&reference_file_path).unwrap_or_default();
77
78         if reference_file.is_empty() {
79             // If we copied over an empty output file, we remove the now empty reference file
80             println!("removing {}", &relative_reference_file_path.display());
81             fs::remove_file(reference_file_path).expect("Could not remove reference file");
82         }
83     }
84 }
85
86 fn updated_since_clippy_build(path: &Path) -> Option<bool> {
87     let clippy_build_time = (*CLIPPY_BUILD_TIME)?;
88     let modified = fs::metadata(path).ok()?.modified().ok()?;
89     Some(modified >= clippy_build_time)
90 }
91
92 fn build_dir() -> PathBuf {
93     let profile = env::var("PROFILE").unwrap_or_else(|_| "debug".to_string());
94     let mut path = PathBuf::new();
95     path.push(CARGO_TARGET_DIR.clone());
96     path.push(profile);
97     path.push("test_build_base");
98     path
99 }