#![allow(clippy::unwrap_used)]
use std::process::Command;
fn cli() -> Command {
Command::new(env!("CARGO_BIN_EXE_fraiseql-cli"))
}
fn fixture(name: &str) -> String {
format!("{}/tests/fixtures/{name}", env!("CARGO_MANIFEST_DIR"))
}
#[test]
fn lint_on_empty_schema_exits_zero() {
let out = cli().args(["lint", &fixture("empty_schema.json")]).output().unwrap();
assert!(out.status.success(), "lint on empty schema must exit 0, got: {:?}", out.status);
}
#[test]
fn lint_on_minimal_schema_exits_zero() {
let out = cli().args(["lint", &fixture("minimal_schema.json")]).output().unwrap();
assert!(out.status.success(), "lint on minimal valid schema must exit 0");
}
#[test]
fn lint_on_missing_file_exits_nonzero() {
let out = cli().args(["lint", "does_not_exist.json"]).output().unwrap();
assert!(!out.status.success(), "lint on missing file must exit non-zero");
}
#[test]
fn lint_json_output_is_valid_json() {
let out = cli().args(["lint", &fixture("empty_schema.json"), "--json"]).output().unwrap();
assert!(out.status.success());
let stdout = String::from_utf8_lossy(&out.stdout);
let parsed: serde_json::Value = serde_json::from_str(&stdout)
.unwrap_or_else(|e| panic!("lint --json output must be valid JSON: {e}\ngot: {stdout}"));
assert!(parsed.is_object(), "JSON output must be an object");
}
#[test]
fn lint_json_output_contains_expected_fields() {
let out = cli().args(["lint", &fixture("empty_schema.json"), "--json"]).output().unwrap();
let stdout = String::from_utf8_lossy(&out.stdout);
let parsed: serde_json::Value = serde_json::from_str(&stdout).unwrap();
let data = parsed.get("data").unwrap_or(&parsed);
assert!(
data.get("overall_score").is_some(),
"JSON output must contain `overall_score`, got: {parsed}"
);
assert!(
data.get("severity_counts").is_some(),
"JSON output must contain `severity_counts`, got: {parsed}"
);
}
#[test]
fn lint_empty_schema_scores_100() {
let out = cli().args(["lint", &fixture("empty_schema.json"), "--json"]).output().unwrap();
let stdout = String::from_utf8_lossy(&out.stdout);
let parsed: serde_json::Value = serde_json::from_str(&stdout).unwrap();
let data = parsed.get("data").unwrap_or(&parsed);
let score = data["overall_score"].as_u64().unwrap_or(0);
assert_eq!(score, 100, "empty schema must score 100; got: {parsed}");
}
#[test]
fn lint_fail_on_critical_clean_schema_exits_zero() {
let out = cli()
.args(["lint", &fixture("empty_schema.json"), "--fail-on-critical"])
.output()
.unwrap();
assert!(out.status.success(), "lint --fail-on-critical on clean schema must exit 0");
}
#[test]
fn lint_federation_filter_exits_zero() {
let out = cli()
.args(["lint", &fixture("minimal_schema.json"), "--federation"])
.output()
.unwrap();
assert!(out.status.success(), "`lint --federation` must exit 0 on valid schema");
}
#[test]
fn lint_cost_filter_exits_zero() {
let out = cli()
.args(["lint", &fixture("minimal_schema.json"), "--cost"])
.output()
.unwrap();
assert!(out.status.success(), "`lint --cost` must exit 0 on valid schema");
}
#[test]
fn lint_cache_filter_exits_zero() {
let out = cli()
.args(["lint", &fixture("minimal_schema.json"), "--cache"])
.output()
.unwrap();
assert!(out.status.success(), "`lint --cache` must exit 0 on valid schema");
}
#[test]
fn lint_exit_code_zero_on_success() {
let out = cli().args(["lint", &fixture("minimal_schema.json")]).output().unwrap();
let code = out.status.code().unwrap_or(-1);
assert_eq!(code, 0, "lint on valid schema must exit with code 0, got {code}");
}
#[test]
fn lint_exit_code_one_on_file_not_found() {
let out = cli().args(["lint", "missing_file.json"]).output().unwrap();
let code = out.status.code().unwrap_or(-1);
assert_eq!(code, 1, "file-not-found must exit with code 1, got {code}");
}