use assert_cmd::assert::OutputAssertExt;
use elaborate::std::{fs::read_to_string_wc, path::PathContext, process::CommandContext};
use regex::Regex;
use std::{env::remove_var, ffi::OsStr, path::Path, process::Command};
use tempfile::tempdir;
use walkdir::WalkDir;
#[ctor::ctor]
fn initialize() {
unsafe {
remove_var("CARGO_TERM_COLOR");
}
}
#[test]
fn clippy() {
let status = Command::new("cargo")
.args([
"+nightly",
"clippy",
"--all-targets",
"--offline",
"--",
"--deny=warnings",
])
.status_wc()
.unwrap();
assert!(status.success());
}
#[test]
fn doctests_are_disabled() {
for dir in ["example", "fixtures"] {
for result in WalkDir::new(Path::new(env!("CARGO_MANIFEST_DIR")).join(dir)) {
let entry = result.unwrap();
if entry.file_name() != OsStr::new("Cargo.toml") {
continue;
}
let path = entry.path();
let manifest_dir = path.parent_wc().unwrap();
if !manifest_dir.join("src/lib.rs").try_exists_wc().unwrap() {
continue;
}
let contents = read_to_string_wc(path).unwrap();
let table = toml::from_str::<toml::Table>(&contents).unwrap();
let doctest = table
.get("lib")
.and_then(toml::Value::as_table)
.and_then(|table| table.get("doctest"))
.and_then(toml::Value::as_bool);
assert_eq!(Some(false), doctest, "failed for `{}`", path.display());
}
}
}
#[test]
fn dylint() {
let assert = Command::new("cargo")
.args(["dylint", "--all", "--", "--all-targets"])
.env("DYLINT_RUSTFLAGS", "--deny warnings")
.assert();
let stderr = String::from_utf8(assert.get_output().stderr.clone()).unwrap();
assert!(assert.try_success().is_ok(), "{}", stderr);
}
#[test]
fn elaborate_disallowed_methods() {
Command::new("cargo")
.args([
"+nightly",
"clippy",
"--all-targets",
"--",
"--deny=warnings",
])
.env("CLIPPY_CONF_DIR", "assets/elaborate")
.assert()
.success();
}
#[test]
fn fixtures_are_unpublishable() {
for result in WalkDir::new(Path::new(env!("CARGO_MANIFEST_DIR")).join("fixtures")) {
let entry = result.unwrap();
if entry.file_name() != OsStr::new("Cargo.toml") {
continue;
}
let path = entry.path();
let contents = read_to_string_wc(path).unwrap();
let table = toml::from_str::<toml::Table>(&contents).unwrap();
let Some(package) = table.get("package").and_then(|value| value.as_table()) else {
continue;
};
let publish = package.get("publish").and_then(toml::Value::as_bool);
assert_eq!(Some(false), publish, "failed for `{}`", path.display());
}
}
#[cfg_attr(target_os = "windows", ignore = "`markdown_link_check` not installed")]
#[test]
fn markdown_link_check() {
let tempdir = tempdir().unwrap();
Command::new("npm")
.args(["install", "markdown-link-check"])
.current_dir(&tempdir)
.assert()
.success();
let readme_md = concat!(env!("CARGO_MANIFEST_DIR"), "/README.md");
Command::new("npx")
.args(["markdown-link-check", readme_md])
.current_dir(&tempdir)
.assert()
.success();
}
#[test]
fn msrv() {
let status = Command::new("cargo")
.args(["msrv", "verify"])
.status_wc()
.unwrap();
assert!(status.success());
}
#[test]
fn readme_reference_links_are_sorted() {
let re = Regex::new(r"^\[[^^\]]*\]:").unwrap();
let readme = read_to_string_wc("README.md").unwrap();
let links = readme
.lines()
.filter(|line| re.is_match(line))
.collect::<Vec<_>>();
let mut links_sorted = links.clone();
links_sorted.sort_unstable();
assert!(
links_sorted == links,
"contents of README.md are not what was expected:\n{}",
links_sorted.join("\n")
);
}
#[cfg(all(unix, not(feature = "__disable_supply_chain_test")))]
#[test]
fn supply_chain() {
use similar_asserts::SimpleDiff;
use std::{process::ExitStatus, str::FromStr};
let mut command = Command::new("cargo");
command.args(["supply-chain", "update", "--cache-max-age=0s"]);
let _: ExitStatus = command.status_wc().unwrap();
let mut command = Command::new("cargo");
command.args(["supply-chain", "json", "--no-dev"]);
let assert = command.assert().success();
let stdout_actual = std::str::from_utf8(&assert.get_output().stdout).unwrap();
let mut value = serde_json::Value::from_str(stdout_actual).unwrap();
remove_avatars(&mut value);
let stdout_normalized = serde_json::to_string_pretty(&value).unwrap();
let path_buf = Path::new(env!("CARGO_MANIFEST_DIR")).join("tests/supply_chain.json");
if enabled("BLESS") {
use elaborate::std::fs::write_wc;
write_wc(path_buf, stdout_normalized).unwrap();
} else {
let stdout_expected = read_to_string_wc(&path_buf).unwrap();
assert!(
stdout_expected == stdout_normalized,
"{}",
SimpleDiff::from_str(&stdout_expected, &stdout_normalized, "left", "right")
);
}
}
#[cfg(all(unix, not(feature = "__disable_supply_chain_test")))]
fn remove_avatars(value: &mut serde_json::Value) {
match value {
serde_json::Value::Null
| serde_json::Value::Bool(_)
| serde_json::Value::Number(_)
| serde_json::Value::String(_) => {}
serde_json::Value::Array(array) => {
for value in array {
remove_avatars(value);
}
}
serde_json::Value::Object(object) => {
object.retain(|key, value| {
if key == "avatar" {
return false;
}
remove_avatars(value);
true
});
}
}
}
#[cfg(all(unix, not(feature = "__disable_supply_chain_test")))]
fn enabled(key: &str) -> bool {
use elaborate::std::env::var_wc;
var_wc(key).is_ok_and(|value| value != "0")
}