KeiSeiKit-1.0/_primitives/_rust/kei-provision/tests/backend_smoke.rs
Parfii-bot 0be354a920 KeiSeiKit-public — clean state
Single-commit clean baseline after security scrub of niche-tells,
project codenames, internal jargon, and contributor-email leaks.

Contents:
- 100 Rust crates (_primitives/_rust/)
- 37 agent manifests (_manifests/) + generated specs (_generated/)
- 67 user-invocable skills (skills/)
- 33 hooks (hooks/)
- Composition blocks (_blocks/)
- Documentation (docs/, README.md)
- TS adapter packages (_ts_packages/)
- Assembler (_assembler/)
- Roles (_roles/)
- Templates (_templates/)
- Forgejo CI (.forgejo/)

Author: Denis Parfionovich <info@greendragon.info>

License: see LICENSE.
2026-05-01 12:09:03 +08:00

264 lines
9 KiB
Rust

//! Smoke tests for kei-provision backends.
//!
//! Strategy: no real cloud calls. We inject a tempdir onto PATH containing
//! fake `hcloud` / `vultr-cli` shell scripts that echo canned JSON matching
//! the real v1 / v3 CLI output shapes. The Backend impls then parse these
//! exactly as they would production output.
use kei_provision::{exec, resolve, CreateOpts};
use std::fs;
use std::os::unix::fs::PermissionsExt;
use std::path::Path;
use std::sync::Mutex;
use tempfile::TempDir;
// Process-global PATH + env vars are not thread-safe across the parallel
// `cargo test` runner. Serialize tests that mutate env.
static ENV_LOCK: Mutex<()> = Mutex::new(());
/// Create a fake CLI script at `<dir>/<bin>` that emits `stdout` verbatim
/// (regardless of arguments) and exits 0.
fn install_fake(dir: &Path, bin: &str, stdout: &str) {
let path = dir.join(bin);
// printf with escaped % for shell robustness — none of our fixtures
// need printf interpolation, so use `cat <<'EOF'`.
let script = format!("#!/usr/bin/env bash\ncat <<'EOF'\n{stdout}\nEOF\n");
fs::write(&path, script).expect("write fake");
let mut perms = fs::metadata(&path).unwrap().permissions();
perms.set_mode(0o755);
fs::set_permissions(&path, perms).unwrap();
}
/// Install a fake that always exits non-zero (simulates "server absent").
fn install_fake_fail(dir: &Path, bin: &str) {
let path = dir.join(bin);
let script = "#!/usr/bin/env bash\nexit 1\n";
fs::write(&path, script).expect("write fake");
let mut perms = fs::metadata(&path).unwrap().permissions();
perms.set_mode(0o755);
fs::set_permissions(&path, perms).unwrap();
}
/// Install a fake that emits `stderr_text` to stderr then exits non-zero.
/// Used to test error-redaction paths in `run_void` / `run_json_strict`.
fn install_fake_stderr(dir: &Path, bin: &str, stderr_text: &str) {
let path = dir.join(bin);
// `cat <<'EOF' 1>&2` preserves literal text including URLs + secrets.
let script = format!(
"#!/usr/bin/env bash\ncat <<'EOF' 1>&2\n{stderr_text}\nEOF\nexit 1\n"
);
fs::write(&path, script).expect("write fake");
let mut perms = fs::metadata(&path).unwrap().permissions();
perms.set_mode(0o755);
fs::set_permissions(&path, perms).unwrap();
}
/// Prepend tempdir to PATH so the fake binary wins, but keep the rest of
/// PATH so `#!/usr/bin/env bash` can still find `bash`.
fn prep_env(dir: &Path, token_var: &str) {
let old = std::env::var("PATH").unwrap_or_default();
let new = format!("{}:{}", dir.display(), old);
std::env::set_var("PATH", new);
std::env::set_var(token_var, "fake-token-for-tests");
}
const HETZNER_DESCRIBE: &str = r#"{
"id": 42,
"name": "test-srv-a",
"status": "running",
"public_net": { "ipv4": { "ip": "1.2.3.4" } },
"server_type": { "name": "cx22" },
"datacenter": { "location": { "name": "fsn1" } }
}"#;
const HETZNER_LIST: &str = r#"[
{
"id": 42,
"name": "test-srv-a",
"status": "running",
"public_net": { "ipv4": { "ip": "1.2.3.4" } }
},
{
"id": 43,
"name": "test-srv-b",
"status": "running",
"public_net": { "ipv4": { "ip": "5.6.7.8" } }
}
]"#;
const VULTR_LIST: &str = r#"{
"instances": [
{
"id": "abc-123",
"label": "test-vultr",
"status": "active",
"power_status": "running",
"main_ip": "9.8.7.6",
"region": "ams",
"plan": "vc2-1c-1gb"
}
]
}"#;
#[test]
fn hetzner_status_parses_ipv4_and_id() {
let _guard = ENV_LOCK.lock().unwrap_or_else(|p| p.into_inner());
let dir = TempDir::new().unwrap();
install_fake(dir.path(), "hcloud", HETZNER_DESCRIBE);
prep_env(dir.path(), "HCLOUD_TOKEN");
let b = resolve("hetzner").unwrap();
let info = b.status("test-srv-a").unwrap().expect("server present");
assert_eq!(info.name, "test-srv-a");
assert_eq!(info.id, "42");
assert_eq!(info.ipv4.as_deref(), Some("1.2.3.4"));
assert_eq!(info.status, "running");
}
#[test]
fn hetzner_status_absent_returns_none() {
let _guard = ENV_LOCK.lock().unwrap_or_else(|p| p.into_inner());
let dir = TempDir::new().unwrap();
install_fake_fail(dir.path(), "hcloud");
prep_env(dir.path(), "HCLOUD_TOKEN");
let b = resolve("hetzner").unwrap();
assert!(b.status("nonexistent").unwrap().is_none());
}
#[test]
fn hetzner_list_parses_array() {
let _guard = ENV_LOCK.lock().unwrap_or_else(|p| p.into_inner());
let dir = TempDir::new().unwrap();
install_fake(dir.path(), "hcloud", HETZNER_LIST);
prep_env(dir.path(), "HCLOUD_TOKEN");
let b = resolve("hetzner").unwrap();
let servers = b.list().unwrap();
assert_eq!(servers.len(), 2);
assert_eq!(servers[0].name, "test-srv-a");
assert_eq!(servers[1].ipv4.as_deref(), Some("5.6.7.8"));
}
#[test]
fn vultr_status_matches_label() {
let _guard = ENV_LOCK.lock().unwrap_or_else(|p| p.into_inner());
let dir = TempDir::new().unwrap();
install_fake(dir.path(), "vultr-cli", VULTR_LIST);
prep_env(dir.path(), "VULTR_API_KEY");
let b = resolve("vultr").unwrap();
let info = b.status("test-vultr").unwrap().expect("found");
assert_eq!(info.id, "abc-123");
assert_eq!(info.ipv4.as_deref(), Some("9.8.7.6"));
assert_eq!(info.status, "active");
}
#[test]
fn vultr_status_absent_when_label_missing() {
let _guard = ENV_LOCK.lock().unwrap_or_else(|p| p.into_inner());
let dir = TempDir::new().unwrap();
install_fake(dir.path(), "vultr-cli", VULTR_LIST);
prep_env(dir.path(), "VULTR_API_KEY");
let b = resolve("vultr").unwrap();
assert!(b.status("not-in-list").unwrap().is_none());
}
#[test]
fn vultr_destroy_absent_is_idempotent() {
let _guard = ENV_LOCK.lock().unwrap_or_else(|p| p.into_inner());
let dir = TempDir::new().unwrap();
install_fake(dir.path(), "vultr-cli", VULTR_LIST);
prep_env(dir.path(), "VULTR_API_KEY");
let b = resolve("vultr").unwrap();
// "ghost" is not in VULTR_LIST → destroy must succeed no-op
b.destroy("ghost", true).unwrap();
}
#[test]
fn unknown_backend_errors_out() {
let err = match resolve("gcp") {
Ok(_) => panic!("gcp should not resolve"),
Err(e) => e.to_string(),
};
assert!(err.contains("unknown backend"), "got: {err}");
}
#[test]
fn create_opts_default_is_none_everywhere() {
let o = CreateOpts::default();
assert!(o.server_type.is_none());
assert!(o.location.is_none());
assert!(o.image.is_none());
assert!(o.ssh_key.is_none());
assert!(o.firewall.is_none());
assert!(o.user_data_path.is_none());
}
// ---- security: error-message redaction (MEDIUM info-disclosure) ----
#[test]
fn error_redacts_full_argv_out_of_message() {
// If a future refactor ever routes `--api-key sk-live-ABCDEF...` as
// an inline arg, the failure path must NOT leak it into logs.
let _guard = ENV_LOCK.lock().unwrap_or_else(|p| p.into_inner());
let dir = TempDir::new().unwrap();
install_fake_stderr(dir.path(), "fakebin", "boom");
prep_env(dir.path(), "IGNORED_TOKEN");
let err = exec::run_void(
"fakebin",
&["--api-key", "sk-live-SECRET-DO-NOT-LEAK", "--region", "fsn1"],
)
.unwrap_err()
.to_string();
// No full arg list; no secret-looking token.
assert!(!err.contains("sk-live-SECRET-DO-NOT-LEAK"), "leaked: {err}");
assert!(!err.contains("--api-key"), "leaked flag: {err}");
// But the count + binary name MUST remain for debugging.
assert!(err.contains("fakebin"), "missing bin: {err}");
assert!(err.contains("4 args"), "missing count: {err}");
}
#[test]
fn error_truncates_long_stderr_to_200_chars_plus_marker() {
let _guard = ENV_LOCK.lock().unwrap_or_else(|p| p.into_inner());
let dir = TempDir::new().unwrap();
// 400 ASCII chars — must truncate.
let long: String = "A".repeat(400);
install_fake_stderr(dir.path(), "fakebin2", &long);
prep_env(dir.path(), "IGNORED_TOKEN");
let err = exec::run_void("fakebin2", &["--op", "create"])
.unwrap_err()
.to_string();
assert!(err.contains("(truncated)"), "no marker: {err}");
// Full 400-byte blob must not be present.
assert!(!err.contains(&"A".repeat(400)), "not truncated: {err}");
// 200 retained is allowed.
assert!(err.contains(&"A".repeat(200)), "kept too little: {err}");
}
#[test]
fn error_truncation_is_utf8_safe_on_cyrillic() {
// Cyrillic chars are multi-byte — a naive byte-slice would panic
// on a non-char-boundary. Assert the helper survives + produces
// well-formed UTF-8.
let _guard = ENV_LOCK.lock().unwrap_or_else(|p| p.into_inner());
let dir = TempDir::new().unwrap();
let cyr: String = "Ошибка".repeat(80); // ~480 chars, multi-byte
install_fake_stderr(dir.path(), "fakebin3", &cyr);
prep_env(dir.path(), "IGNORED_TOKEN");
let err = exec::run_void("fakebin3", &["--any"])
.unwrap_err()
.to_string();
// Didn't panic, is valid UTF-8 (it's a Rust String), contains marker.
assert!(err.contains("(truncated)"), "no marker: {err}");
assert!(err.contains("Ошибка"), "lost cyrillic: {err}");
}