Significantly more glorious marks

This commit is contained in:
Aleksey Kladov 2020-05-20 00:19:59 +02:00
parent 4d3fd62f89
commit d18d1c0594
4 changed files with 86 additions and 8 deletions

View file

@ -1,11 +1,11 @@
use ra_ide_db::defs::{classify_name_ref, Definition, NameRefClass}; use ra_ide_db::defs::{classify_name_ref, Definition, NameRefClass};
use ra_syntax::{ast, AstNode, SyntaxKind, T}; use ra_syntax::{ast, AstNode, SyntaxKind, T};
use test_utils::mark;
use crate::{ use crate::{
assist_context::{AssistContext, Assists}, assist_context::{AssistContext, Assists},
AssistId, AssistId,
}; };
use test_utils::tested_by;
// Assist: add_turbo_fish // Assist: add_turbo_fish
// //
@ -28,7 +28,7 @@ pub(crate) fn add_turbo_fish(acc: &mut Assists, ctx: &AssistContext) -> Option<(
let ident = ctx.find_token_at_offset(SyntaxKind::IDENT)?; let ident = ctx.find_token_at_offset(SyntaxKind::IDENT)?;
let next_token = ident.next_token()?; let next_token = ident.next_token()?;
if next_token.kind() == T![::] { if next_token.kind() == T![::] {
tested_by!(add_turbo_fish_one_fish_is_enough); mark::hit!(add_turbo_fish_one_fish_is_enough);
return None; return None;
} }
let name_ref = ast::NameRef::cast(ident.parent())?; let name_ref = ast::NameRef::cast(ident.parent())?;
@ -42,7 +42,7 @@ pub(crate) fn add_turbo_fish(acc: &mut Assists, ctx: &AssistContext) -> Option<(
}; };
let generics = hir::GenericDef::Function(fun).params(ctx.sema.db); let generics = hir::GenericDef::Function(fun).params(ctx.sema.db);
if generics.is_empty() { if generics.is_empty() {
tested_by!(add_turbo_fish_non_generic); mark::hit!(add_turbo_fish_non_generic);
return None; return None;
} }
acc.add(AssistId("add_turbo_fish"), "Add `::<>`", ident.text_range(), |builder| { acc.add(AssistId("add_turbo_fish"), "Add `::<>`", ident.text_range(), |builder| {
@ -58,7 +58,7 @@ mod tests {
use crate::tests::{check_assist, check_assist_not_applicable}; use crate::tests::{check_assist, check_assist_not_applicable};
use super::*; use super::*;
use test_utils::covers; use test_utils::mark;
#[test] #[test]
fn add_turbo_fish_function() { fn add_turbo_fish_function() {
@ -106,7 +106,7 @@ fn main() {
#[test] #[test]
fn add_turbo_fish_one_fish_is_enough() { fn add_turbo_fish_one_fish_is_enough() {
covers!(add_turbo_fish_one_fish_is_enough); mark::check!(add_turbo_fish_one_fish_is_enough);
check_assist_not_applicable( check_assist_not_applicable(
add_turbo_fish, add_turbo_fish,
r#" r#"
@ -120,7 +120,7 @@ fn main() {
#[test] #[test]
fn add_turbo_fish_non_generic() { fn add_turbo_fish_non_generic() {
covers!(add_turbo_fish_non_generic); mark::check!(add_turbo_fish_non_generic);
check_assist_not_applicable( check_assist_not_applicable(
add_turbo_fish, add_turbo_fish,
r#" r#"

View file

@ -10,6 +10,4 @@ test_utils::marks![
test_not_applicable_if_variable_unused test_not_applicable_if_variable_unused
change_visibility_field_false_positive change_visibility_field_false_positive
test_add_from_impl_already_exists test_add_from_impl_already_exists
add_turbo_fish_one_fish_is_enough
add_turbo_fish_non_generic
]; ];

View file

@ -8,6 +8,8 @@
#[macro_use] #[macro_use]
pub mod marks; pub mod marks;
#[macro_use]
pub mod mark;
use std::{ use std::{
fs, fs,

View file

@ -0,0 +1,78 @@
//! This module implements manually tracked test coverage, which is useful for
//! quickly finding a test responsible for testing a particular bit of code.
//!
//! See <https://matklad.github.io/2018/06/18/a-trick-for-test-maintenance.html>
//! for details, but the TL;DR is that you write your test as
//!
//! ```
//! #[test]
//! fn test_foo() {
//! mark::check!(test_foo);
//! }
//! ```
//!
//! and in the code under test you write
//!
//! ```
//! # use test_utils::mark;
//! # fn some_condition() -> bool { true }
//! fn foo() {
//! if some_condition() {
//! mark::hit!(test_foo);
//! }
//! }
//! ```
//!
//! This module then checks that executing the test indeed covers the specified
//! function. This is useful if you come back to the `foo` function ten years
//! later and wonder where the test are: now you can grep for `test_foo`.
use std::sync::atomic::{AtomicUsize, Ordering};
#[macro_export]
macro_rules! _hit {
($ident:ident) => {{
#[cfg(test)]
{
extern "C" {
#[no_mangle]
static $ident: std::sync::atomic::AtomicUsize;
}
unsafe {
$ident.fetch_add(1, std::sync::atomic::Ordering::SeqCst);
}
}
}};
}
pub use _hit as hit;
#[macro_export]
macro_rules! _check {
($ident:ident) => {
#[no_mangle]
static $ident: std::sync::atomic::AtomicUsize = std::sync::atomic::AtomicUsize::new(0);
let _checker = $crate::mark::MarkChecker::new(&$ident);
};
}
pub use _check as check;
pub struct MarkChecker {
mark: &'static AtomicUsize,
value_on_entry: usize,
}
impl MarkChecker {
pub fn new(mark: &'static AtomicUsize) -> MarkChecker {
let value_on_entry = mark.load(Ordering::SeqCst);
MarkChecker { mark, value_on_entry }
}
}
impl Drop for MarkChecker {
fn drop(&mut self) {
if std::thread::panicking() {
return;
}
let value_on_exit = self.mark.load(Ordering::SeqCst);
assert!(value_on_exit > self.value_on_entry, "mark was not hit")
}
}