internal: check that coverage marks are always paired

This commit is contained in:
Aleksey Kladov 2021-06-13 12:33:08 +03:00
parent 0eafc88079
commit 546be18e3a
3 changed files with 54 additions and 6 deletions

View file

@ -163,14 +163,14 @@ fn legacy_macro_items() {
// correctly. // correctly.
check_at( check_at(
r#" r#"
macro_rules! hit { macro_rules! mark {
() => { () => {
struct Hit {} struct Hit {}
} }
} }
fn f() { fn f() {
hit!(); mark!();
$0 $0
} }
"#, "#,
@ -193,20 +193,20 @@ use core::cov_mark;
fn f() { fn f() {
fn nested() { fn nested() {
cov_mark::hit!(Hit); cov_mark::mark!(Hit);
$0 $0
} }
} }
//- /core.rs crate:core //- /core.rs crate:core
pub mod cov_mark { pub mod cov_mark {
#[macro_export] #[macro_export]
macro_rules! _hit { macro_rules! _mark {
($name:ident) => { ($name:ident) => {
struct $name {} struct $name {}
} }
} }
pub use crate::_hit as hit; pub use crate::_mark as mark;
} }
"#, "#,
expect![[r#" expect![[r#"

View file

@ -1007,6 +1007,7 @@ fn go(world: &WorldSnapshot) { go(w$0) }
#[test] #[test]
fn too_many_arguments() { fn too_many_arguments() {
cov_mark::check!(too_many_arguments);
check_relevance( check_relevance(
r#" r#"
struct Foo; struct Foo;

View file

@ -1,4 +1,7 @@
use std::path::{Path, PathBuf}; use std::{
collections::HashSet,
path::{Path, PathBuf},
};
use xshell::{cmd, pushd, pushenv, read_file}; use xshell::{cmd, pushd, pushenv, read_file};
@ -81,6 +84,7 @@ Please adjust docs/dev/lsp-extensions.md.
#[test] #[test]
fn rust_files_are_tidy() { fn rust_files_are_tidy() {
let mut tidy_docs = TidyDocs::default(); let mut tidy_docs = TidyDocs::default();
let mut tidy_marks = TidyMarks::default();
for path in rust_files() { for path in rust_files() {
let text = read_file(&path).unwrap(); let text = read_file(&path).unwrap();
check_todo(&path, &text); check_todo(&path, &text);
@ -88,8 +92,10 @@ fn rust_files_are_tidy() {
check_trailing_ws(&path, &text); check_trailing_ws(&path, &text);
deny_clippy(&path, &text); deny_clippy(&path, &text);
tidy_docs.visit(&path, &text); tidy_docs.visit(&path, &text);
tidy_marks.visit(&path, &text);
} }
tidy_docs.finish(); tidy_docs.finish();
tidy_marks.finish();
} }
#[test] #[test]
@ -408,6 +414,39 @@ fn is_exclude_dir(p: &Path, dirs_to_exclude: &[&str]) -> bool {
.any(|it| dirs_to_exclude.contains(&it)) .any(|it| dirs_to_exclude.contains(&it))
} }
#[derive(Default)]
struct TidyMarks {
hits: HashSet<String>,
checks: HashSet<String>,
}
impl TidyMarks {
fn visit(&mut self, _path: &Path, text: &str) {
for line in text.lines() {
if let Some(mark) = find_mark(line, "hit") {
self.hits.insert(mark.to_string());
}
if let Some(mark) = find_mark(line, "check") {
self.checks.insert(mark.to_string());
}
if let Some(mark) = find_mark(line, "check_count") {
self.checks.insert(mark.to_string());
}
}
}
fn finish(self) {
assert!(!self.hits.is_empty());
let diff: Vec<_> =
self.hits.symmetric_difference(&self.checks).map(|it| it.as_str()).collect();
if !diff.is_empty() {
panic!("unpaired marks: {:?}", diff)
}
}
}
#[allow(deprecated)] #[allow(deprecated)]
fn stable_hash(text: &str) -> u64 { fn stable_hash(text: &str) -> u64 {
use std::hash::{Hash, Hasher, SipHasher}; use std::hash::{Hash, Hasher, SipHasher};
@ -417,3 +456,11 @@ fn stable_hash(text: &str) -> u64 {
text.hash(&mut hasher); text.hash(&mut hasher);
hasher.finish() hasher.finish()
} }
fn find_mark<'a>(text: &'a str, mark: &'static str) -> Option<&'a str> {
let idx = text.find(mark)?;
let text = text[idx + mark.len()..].strip_prefix("!(")?;
let idx = text.find(|c: char| !(c.is_alphanumeric() || c == '_'))?;
let text = &text[..idx];
Some(text)
}