use std::ffi::OsString;
use std::path::{Path, PathBuf};
use std::process::Command;
use std::sync::atomic::{AtomicU64, Ordering};
use testing_conventions::patch_coverage::check;
use testing_conventions::run;
struct TempRepo(PathBuf);
impl TempRepo {
fn new(slug: &str) -> Self {
static COUNTER: AtomicU64 = AtomicU64::new(0);
let root = std::env::temp_dir().join(format!(
"tc-patch-cov-{}-{}-{}",
slug,
std::process::id(),
COUNTER.fetch_add(1, Ordering::Relaxed),
));
std::fs::create_dir_all(&root).unwrap();
git(&root, &["init", "-q"]);
git(&root, &["config", "user.email", "test@example.com"]);
git(&root, &["config", "user.name", "Test"]);
TempRepo(root)
}
fn write(&self, rel: &str, contents: &str) {
let path = self.0.join(rel);
std::fs::create_dir_all(path.parent().unwrap()).unwrap();
std::fs::write(path, contents).unwrap();
}
fn commit(&self, message: &str) {
git(&self.0, &["add", "-A"]);
git(
&self.0,
&["-c", "commit.gpgsign=false", "commit", "-q", "-m", message],
);
}
fn head(&self) -> String {
let out = Command::new("git")
.args(["rev-parse", "HEAD"])
.current_dir(&self.0)
.output()
.expect("git rev-parse should run");
assert!(out.status.success(), "git rev-parse failed");
String::from_utf8(out.stdout).unwrap().trim().to_string()
}
}
impl Drop for TempRepo {
fn drop(&mut self) {
let _ = std::fs::remove_dir_all(&self.0);
}
}
fn git(dir: &Path, args: &[&str]) {
let status = Command::new("git")
.args(args)
.current_dir(dir)
.status()
.expect("git should run");
assert!(status.success(), "git {args:?} failed");
}
fn uncovered(repo: &TempRepo, base: &str) -> Vec<String> {
check(&repo.0, base, &[])
.expect("checking a readable repo should succeed")
.iter()
.map(|u| format!("{}:{}", u.file, u.line))
.collect()
}
fn run_patch_coverage(
repo: &TempRepo,
language: &str,
base: &str,
config: Option<&str>,
) -> anyhow::Result<i32> {
let mut argv: Vec<OsString> = vec![
"testing-conventions".into(),
"unit".into(),
"patch-coverage".into(),
repo.0.clone().into_os_string(),
"--language".into(),
language.into(),
"--base".into(),
base.into(),
];
if let Some(name) = config {
argv.push("--config".into());
argv.push(repo.0.join(name).into_os_string());
}
run(argv)
}
fn run_cli(args: &[&str]) -> anyhow::Result<i32> {
let argv: Vec<OsString> = std::iter::once(OsString::from("testing-conventions"))
.chain(args.iter().copied().map(OsString::from))
.collect();
run(argv)
}
const WIDGET_PY: &str = r#"def widget(n):
if n > 0:
return "pos"
return "neg"
"#;
const WIDGET_TEST_PY: &str = r#"from widget import widget
def test_widget():
assert widget(1) == "pos"
assert widget(-1) == "neg"
"#;
const WIDGET_PY_UNCOVERED: &str = r#"def widget(n):
if n > 0:
return "pos"
if n == 42:
return "answer"
return "neg"
"#;
#[test]
fn python_uncovered_changed_line_is_reported() {
let repo = TempRepo::new("uncovered");
repo.write("widget.py", WIDGET_PY);
repo.write("widget_test.py", WIDGET_TEST_PY);
repo.commit("base");
let base = repo.head();
repo.write("widget.py", WIDGET_PY_UNCOVERED);
repo.commit("add an untested branch");
let reported = uncovered(&repo, &base);
assert!(
reported.contains(&"widget.py:5".to_string()),
"the uncovered new line should be reported; got: {reported:?}"
);
assert!(
reported.contains(&"widget.py:4".to_string()),
"the uncovered new branch source should be reported; got: {reported:?}"
);
}
#[test]
fn python_covered_change_is_clean() {
let repo = TempRepo::new("covered");
repo.write("widget.py", WIDGET_PY);
repo.write("widget_test.py", WIDGET_TEST_PY);
repo.commit("base");
let base = repo.head();
repo.write(
"widget.py",
r#"def widget(n):
if n > 0:
return "positive"
return "neg"
"#,
);
repo.write(
"widget_test.py",
r#"from widget import widget
def test_widget():
assert widget(1) == "positive"
assert widget(-1) == "neg"
"#,
);
repo.commit("reword a covered line and update its test");
assert!(
uncovered(&repo, &base).is_empty(),
"a change to a covered line is clean"
);
}
#[test]
fn a_change_touching_no_python_is_clean() {
let repo = TempRepo::new("no-py");
repo.write("widget.py", WIDGET_PY);
repo.write("widget_test.py", WIDGET_TEST_PY);
repo.write("README.md", "# project\n");
repo.commit("base");
let base = repo.head();
repo.write("README.md", "# project\n\nnow with docs\n");
repo.commit("docs only");
assert!(
uncovered(&repo, &base).is_empty(),
"a change that touches no Python source is clean"
);
}
#[test]
fn python_added_untested_file_changed_lines_are_uncovered() {
let repo = TempRepo::new("added");
repo.write("widget.py", WIDGET_PY);
repo.write("widget_test.py", WIDGET_TEST_PY);
repo.commit("base");
let base = repo.head();
repo.write("lonely.py", "def lonely():\n return 41\n");
repo.commit("add a brand-new untested source");
let reported = uncovered(&repo, &base);
assert!(
reported.contains(&"lonely.py:1".to_string()),
"the added file's uncovered lines should be reported; got: {reported:?}"
);
}
#[test]
fn python_changed_comment_line_is_not_a_subject() {
let repo = TempRepo::new("comment");
repo.write("widget.py", WIDGET_PY);
repo.write("widget_test.py", WIDGET_TEST_PY);
repo.commit("base");
let base = repo.head();
repo.write(
"widget.py",
r#"def widget(n):
# branch on the sign of n
if n > 0:
return "pos"
return "neg"
"#,
);
repo.commit("add an explanatory comment");
assert!(
uncovered(&repo, &base).is_empty(),
"a changed comment line has nothing to cover"
);
}
#[test]
fn an_unknown_base_ref_is_an_error() {
let repo = TempRepo::new("bad-base");
repo.write("widget.py", WIDGET_PY);
repo.write("widget_test.py", WIDGET_TEST_PY);
repo.commit("base");
assert!(
check(&repo.0, "no-such-ref", &[]).is_err(),
"an unresolvable base ref must error"
);
}
#[test]
fn python_subcommand_exits_nonzero_on_an_uncovered_changed_line() {
let repo = TempRepo::new("cli-red");
repo.write("widget.py", WIDGET_PY);
repo.write("widget_test.py", WIDGET_TEST_PY);
repo.commit("base");
let base = repo.head();
repo.write("widget.py", WIDGET_PY_UNCOVERED);
repo.commit("add an untested branch");
assert_eq!(run_patch_coverage(&repo, "python", &base, None).unwrap(), 1);
}
#[test]
fn python_subcommand_exits_zero_when_the_change_is_covered() {
let repo = TempRepo::new("cli-clean");
repo.write("widget.py", WIDGET_PY);
repo.write("widget_test.py", WIDGET_TEST_PY);
repo.commit("base");
let base = repo.head();
repo.write(
"widget.py",
r#"def widget(n):
if n > 0:
return "positive"
return "neg"
"#,
);
repo.write(
"widget_test.py",
r#"from widget import widget
def test_widget():
assert widget(1) == "positive"
assert widget(-1) == "neg"
"#,
);
repo.commit("reword a covered line and update its test");
assert_eq!(run_patch_coverage(&repo, "python", &base, None).unwrap(), 0);
}
#[test]
fn python_a_coverage_exemption_lifts_an_uncovered_changed_line() {
let repo = TempRepo::new("exempt");
repo.write(
"testing-conventions.toml",
"[[python.exempt]]\npath = \"shim.py\"\nrules = [\"coverage\"]\n\
reason = \"thin launcher; logic lives in tested modules\"\n",
);
repo.write("widget.py", WIDGET_PY);
repo.write("widget_test.py", WIDGET_TEST_PY);
repo.write("shim.py", "def shim():\n return 0\n");
repo.commit("base");
let base = repo.head();
repo.write("shim.py", "def shim():\n return 1\n");
repo.commit("edit the untested launcher");
assert_eq!(run_patch_coverage(&repo, "python", &base, None).unwrap(), 1);
assert_eq!(
run_patch_coverage(&repo, "python", &base, Some("testing-conventions.toml")).unwrap(),
0
);
}
#[test]
fn patch_coverage_requires_language() {
let err = run_cli(&["unit", "patch-coverage", "/tmp", "--base", "HEAD"])
.expect_err("--language is required");
let clap_err = err
.downcast_ref::<clap::Error>()
.expect("a missing required flag should surface as a clap::Error");
assert_eq!(
clap_err.kind(),
clap::error::ErrorKind::MissingRequiredArgument
);
}
#[test]
fn patch_coverage_rejects_rust() {
let repo = TempRepo::new("rust-reject");
repo.write("lib.rs", "pub fn f() {}\n");
repo.commit("base");
let base = repo.head();
let err = run_patch_coverage(&repo, "rust", &base, None).unwrap_err();
assert!(err.to_string().contains("separate item"), "got: {err}");
}
#[test]
fn patch_coverage_rejects_typescript() {
let repo = TempRepo::new("ts-reject");
repo.write("widget.ts", "export const f = () => 1;\n");
repo.commit("base");
let base = repo.head();
let err = run_patch_coverage(&repo, "typescript", &base, None).unwrap_err();
assert!(err.to_string().contains("separate item"), "got: {err}");
}