Files
kebab/crates/kebab-cli/tests/reset_cli.rs
altair823 565caebec6 review(회차1): 회차 1 critical + nit 반영
- (critical) embeddings.rs: truncate_embedding_records 위치 이동.
  mark_embedding_records_committed 함수 위에 끼워 넣었더니 위쪽
  mark_committed 의 14 줄짜리 doc comment (`WHERE status='pending'`
  의 design rationale 등) 가 truncate 의 doc 으로 흡수되고
  mark_committed 자체는 doc 없이 남는 버그. impl block 끝 (mark_committed
  의 닫는 } 다음) 으로 옮겨 plan 의 원래 의도와도 일치.
- (nit) tests/reset_cli.rs: removed_paths 의 idempotency 검증 보강.
  data dir 은 reported, cache dir 은 omit (생성 안 했으니)
  되어야 함을 strict 하게 assert. state dir 은 logging init 의
  side-effect 로 자동 생성되어 둘 다 가능하므로 허용.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 18:43:25 +00:00

157 lines
6.2 KiB
Rust

//! Integration coverage for `kebab reset` — exercises the binary
//! end-to-end against a tempdir-rooted XDG layout. Each test runs the
//! built `kebab` bin in a fresh subprocess so the per-process XDG env
//! overrides don't bleed into sibling tests.
use std::process::Command;
fn kebab_bin() -> std::path::PathBuf {
// The compiled bin is at `target/debug/kebab` relative to the
// workspace root. CARGO_MANIFEST_DIR points at the kebab-cli crate
// dir; the workspace root is two levels above (../../).
let manifest = env!("CARGO_MANIFEST_DIR");
std::path::PathBuf::from(manifest)
.parent()
.unwrap()
.parent()
.unwrap()
.join("target/debug/kebab")
}
#[test]
fn reset_data_only_yes_removes_data_dir_and_keeps_config() {
let tmp = tempfile::tempdir().unwrap();
let xdg_cfg = tmp.path().join("cfg");
let xdg_data = tmp.path().join("data");
let xdg_cache = tmp.path().join("cache");
let xdg_state = tmp.path().join("state");
std::fs::create_dir_all(xdg_cfg.join("kebab")).unwrap();
std::fs::create_dir_all(xdg_data.join("kebab")).unwrap();
std::fs::create_dir_all(xdg_cache.join("kebab")).unwrap();
std::fs::create_dir_all(xdg_state.join("kebab")).unwrap();
// No `config.toml` written — Config::load(None) falls back to
// defaults when the file is absent (see kebab-config). The marker
// file under cfg/kebab/ is what we assert survives.
std::fs::write(xdg_cfg.join("kebab/marker"), b"cfg").unwrap();
std::fs::write(xdg_data.join("kebab/marker"), b"data").unwrap();
let out = Command::new(kebab_bin())
.args(["reset", "--data-only", "--yes"])
.env("XDG_CONFIG_HOME", &xdg_cfg)
.env("XDG_DATA_HOME", &xdg_data)
.env("XDG_CACHE_HOME", &xdg_cache)
.env("XDG_STATE_HOME", &xdg_state)
.output()
.unwrap();
assert!(
out.status.success(),
"stderr: {}",
String::from_utf8_lossy(&out.stderr)
);
assert!(!xdg_data.join("kebab").exists(), "data dir should be gone");
assert!(!xdg_cache.join("kebab").exists(), "cache dir should be gone");
assert!(!xdg_state.join("kebab").exists(), "state dir should be gone");
assert!(xdg_cfg.join("kebab/marker").exists(), "config dir preserved");
}
#[test]
fn reset_no_yes_in_non_tty_aborts_with_exit_2() {
let tmp = tempfile::tempdir().unwrap();
let xdg_data = tmp.path().join("data");
std::fs::create_dir_all(xdg_data.join("kebab")).unwrap();
std::fs::write(xdg_data.join("kebab/marker"), b"d").unwrap();
let out = Command::new(kebab_bin())
.args(["reset", "--data-only"])
.env("XDG_CONFIG_HOME", tmp.path().join("cfg"))
.env("XDG_DATA_HOME", &xdg_data)
.env("XDG_CACHE_HOME", tmp.path().join("cache"))
.env("XDG_STATE_HOME", tmp.path().join("state"))
.output()
.unwrap();
// Non-TTY (Command::output gives no tty) without --yes must abort.
assert!(!out.status.success(), "expected abort, got success");
let code = out.status.code().unwrap_or(-1);
assert_eq!(code, 2, "expected exit 2 (generic error), got {code}");
assert!(
xdg_data.join("kebab").exists(),
"data dir must survive an aborted reset"
);
let stderr = String::from_utf8_lossy(&out.stderr);
assert!(
stderr.contains("non-interactive") || stderr.contains("--yes"),
"expected refusal hint in stderr, got: {stderr}"
);
}
#[test]
fn reset_data_only_yes_json_emits_reset_report_v1() {
let tmp = tempfile::tempdir().unwrap();
let xdg_data = tmp.path().join("data");
std::fs::create_dir_all(xdg_data.join("kebab")).unwrap();
std::fs::write(xdg_data.join("kebab/marker"), b"d").unwrap();
let out = Command::new(kebab_bin())
.args(["--json", "reset", "--data-only", "--yes"])
.env("XDG_CONFIG_HOME", tmp.path().join("cfg"))
.env("XDG_DATA_HOME", &xdg_data)
.env("XDG_CACHE_HOME", tmp.path().join("cache"))
.env("XDG_STATE_HOME", tmp.path().join("state"))
.output()
.unwrap();
assert!(out.status.success(), "stderr: {}", String::from_utf8_lossy(&out.stderr));
let v: serde_json::Value = serde_json::from_slice(&out.stdout).unwrap();
assert_eq!(
v.get("schema_version").and_then(|s| s.as_str()),
Some("reset_report.v1")
);
assert_eq!(v.get("scope").and_then(|s| s.as_str()), Some("data_only"));
// The data dir was created beforehand and must show up in the
// report. The cache dir was NOT created, so it must be omitted —
// proving idempotency ("missing path is treated as already
// removed"). The state dir may or may not appear: kebab-app's
// logging init creates the state dir as a side-effect, which is
// tolerated. We assert the strict invariant (data in, cache out)
// and let the state dir be either way.
let paths: Vec<String> = v
.get("removed_paths")
.and_then(|a| a.as_array())
.expect("removed_paths must be a JSON array")
.iter()
.filter_map(|s| s.as_str().map(str::to_owned))
.collect();
assert!(
paths.iter().any(|p| p.contains("/data/kebab")),
"data dir must be reported as removed, got: {paths:?}"
);
assert!(
!paths.iter().any(|p| p.contains("/cache/kebab")),
"cache dir was never created and must be omitted, got: {paths:?}"
);
}
#[test]
fn reset_mutually_exclusive_scope_flags_rejected() {
// clap's `group = "reset_scope"` should reject --all and
// --data-only together. The bin must exit nonzero with a clap
// usage error before touching any path.
let tmp = tempfile::tempdir().unwrap();
let out = Command::new(kebab_bin())
.args(["reset", "--all", "--data-only", "--yes"])
.env("XDG_CONFIG_HOME", tmp.path().join("cfg"))
.env("XDG_DATA_HOME", tmp.path().join("data"))
.env("XDG_CACHE_HOME", tmp.path().join("cache"))
.env("XDG_STATE_HOME", tmp.path().join("state"))
.output()
.unwrap();
assert!(!out.status.success());
let stderr = String::from_utf8_lossy(&out.stderr);
assert!(
stderr.contains("cannot be used") || stderr.contains("conflicts"),
"expected clap conflict error, got: {stderr}"
);
}