completely rewrote test suite for top level src files and all builtin files
This commit is contained in:
@@ -173,20 +173,24 @@ pub fn complete_builtin(node: Node) -> ShResult<()> {
|
||||
|
||||
if comp_opts.flags.contains(CompFlags::PRINT) {
|
||||
if argv.is_empty() {
|
||||
read_meta(|m| {
|
||||
read_meta(|m| -> ShResult<()> {
|
||||
let specs = m.comp_specs().values();
|
||||
for spec in specs {
|
||||
println!("{}", spec.source());
|
||||
let stdout = borrow_fd(STDOUT_FILENO);
|
||||
write(stdout, spec.source().as_bytes())?;
|
||||
}
|
||||
})
|
||||
Ok(())
|
||||
})?;
|
||||
} else {
|
||||
read_meta(|m| {
|
||||
read_meta(|m| -> ShResult<()> {
|
||||
for (cmd, _) in &argv {
|
||||
if let Some(spec) = m.comp_specs().get(cmd) {
|
||||
println!("{}", spec.source());
|
||||
let stdout = borrow_fd(STDOUT_FILENO);
|
||||
write(stdout, spec.source().as_bytes())?;
|
||||
}
|
||||
}
|
||||
})
|
||||
Ok(())
|
||||
})?;
|
||||
}
|
||||
|
||||
state::set_status(0);
|
||||
@@ -309,3 +313,318 @@ pub fn get_comp_opts(opts: Vec<Opt>) -> ShResult<CompOpts> {
|
||||
|
||||
Ok(comp_opts)
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use std::fs;
|
||||
use tempfile::TempDir;
|
||||
use crate::state::{self, read_meta, write_vars, VarFlags, VarKind};
|
||||
use crate::testutil::{TestGuard, test_input};
|
||||
|
||||
// ===================== complete: Registration =====================
|
||||
|
||||
#[test]
|
||||
fn complete_register_wordlist() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -W 'foo bar baz' mycmd").unwrap();
|
||||
|
||||
let spec = read_meta(|m| m.get_comp_spec("mycmd"));
|
||||
assert!(spec.is_some());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_register_files() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -f mycmd").unwrap();
|
||||
|
||||
let spec = read_meta(|m| m.get_comp_spec("mycmd"));
|
||||
assert!(spec.is_some());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_register_dirs() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -d mycmd").unwrap();
|
||||
|
||||
let spec = read_meta(|m| m.get_comp_spec("mycmd"));
|
||||
assert!(spec.is_some());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_register_multiple_commands() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -W 'x y' cmd1 cmd2").unwrap();
|
||||
|
||||
assert!(read_meta(|m| m.get_comp_spec("cmd1")).is_some());
|
||||
assert!(read_meta(|m| m.get_comp_spec("cmd2")).is_some());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_register_function() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -F _my_comp mycmd").unwrap();
|
||||
|
||||
let spec = read_meta(|m| m.get_comp_spec("mycmd"));
|
||||
assert!(spec.is_some());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_register_combined_flags() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -f -d -v mycmd").unwrap();
|
||||
|
||||
let spec = read_meta(|m| m.get_comp_spec("mycmd"));
|
||||
assert!(spec.is_some());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_overwrite_spec() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -W 'old' mycmd").unwrap();
|
||||
test_input("complete -W 'new' mycmd").unwrap();
|
||||
|
||||
let spec = read_meta(|m| m.get_comp_spec("mycmd"));
|
||||
assert!(spec.is_some());
|
||||
// Verify the source reflects the latest registration
|
||||
assert!(spec.unwrap().source().contains("new"));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_no_command_fails() {
|
||||
let _g = TestGuard::new();
|
||||
let result = test_input("complete -W 'foo'");
|
||||
assert!(result.is_err());
|
||||
}
|
||||
|
||||
// ===================== complete -r: Removal =====================
|
||||
|
||||
#[test]
|
||||
fn complete_remove_spec() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -W 'foo' mycmd").unwrap();
|
||||
assert!(read_meta(|m| m.get_comp_spec("mycmd")).is_some());
|
||||
|
||||
test_input("complete -r mycmd").unwrap();
|
||||
assert!(read_meta(|m| m.get_comp_spec("mycmd")).is_none());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_remove_multiple() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -W 'a' cmd1").unwrap();
|
||||
test_input("complete -W 'b' cmd2").unwrap();
|
||||
|
||||
test_input("complete -r cmd1 cmd2").unwrap();
|
||||
assert!(read_meta(|m| m.get_comp_spec("cmd1")).is_none());
|
||||
assert!(read_meta(|m| m.get_comp_spec("cmd2")).is_none());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_remove_nonexistent_is_ok() {
|
||||
let _g = TestGuard::new();
|
||||
// Removing a spec that doesn't exist should not error
|
||||
test_input("complete -r nosuchcmd").unwrap();
|
||||
assert_eq!(state::get_status(), 0);
|
||||
}
|
||||
|
||||
// ===================== complete -p: Print =====================
|
||||
|
||||
#[test]
|
||||
fn complete_print_specific() {
|
||||
let guard = TestGuard::new();
|
||||
test_input("complete -W 'alpha beta' mycmd").unwrap();
|
||||
guard.read_output();
|
||||
|
||||
test_input("complete -p mycmd").unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("mycmd"));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_print_all() {
|
||||
let guard = TestGuard::new();
|
||||
// Clear any existing specs and register two
|
||||
test_input("complete -W 'a' cmd1").unwrap();
|
||||
test_input("complete -W 'b' cmd2").unwrap();
|
||||
guard.read_output();
|
||||
|
||||
test_input("complete -p").unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("cmd1"));
|
||||
assert!(out.contains("cmd2"));
|
||||
}
|
||||
|
||||
// ===================== complete -o: Option flags =====================
|
||||
|
||||
#[test]
|
||||
fn complete_option_default() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -o default -W 'foo' mycmd").unwrap();
|
||||
assert_eq!(state::get_status(), 0);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_option_dirnames() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -o dirnames -W 'foo' mycmd").unwrap();
|
||||
assert_eq!(state::get_status(), 0);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn complete_option_invalid() {
|
||||
let _g = TestGuard::new();
|
||||
let result = test_input("complete -o bogus -W 'foo' mycmd");
|
||||
assert!(result.is_err());
|
||||
}
|
||||
|
||||
// ===================== compgen -W: Word list =====================
|
||||
|
||||
#[test]
|
||||
fn compgen_wordlist_no_prefix() {
|
||||
let guard = TestGuard::new();
|
||||
test_input("compgen -W 'alpha beta gamma'").unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("alpha"));
|
||||
assert!(out.contains("beta"));
|
||||
assert!(out.contains("gamma"));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn compgen_wordlist_with_prefix() {
|
||||
let guard = TestGuard::new();
|
||||
test_input("compgen -W 'apple banana avocado' a").unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("apple"));
|
||||
assert!(out.contains("avocado"));
|
||||
assert!(!out.contains("banana"));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn compgen_wordlist_no_match() {
|
||||
let guard = TestGuard::new();
|
||||
test_input("compgen -W 'foo bar baz' z").unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.trim().is_empty());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn compgen_wordlist_exact_match() {
|
||||
let guard = TestGuard::new();
|
||||
test_input("compgen -W 'hello help helm' hel").unwrap();
|
||||
let out = guard.read_output();
|
||||
let lines: Vec<&str> = out.lines().collect();
|
||||
assert_eq!(lines.len(), 3);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn compgen_wordlist_single_match() {
|
||||
let guard = TestGuard::new();
|
||||
test_input("compgen -W 'alpha beta gamma' g").unwrap();
|
||||
let out = guard.read_output();
|
||||
let lines: Vec<&str> = out.lines().collect();
|
||||
assert_eq!(lines.len(), 1);
|
||||
assert_eq!(lines[0], "gamma");
|
||||
}
|
||||
|
||||
// ===================== compgen -v: Variables =====================
|
||||
|
||||
#[test]
|
||||
fn compgen_variables() {
|
||||
let guard = TestGuard::new();
|
||||
write_vars(|v| v.set_var("TESTCOMPVAR", VarKind::Str("x".into()), VarFlags::NONE)).unwrap();
|
||||
|
||||
test_input("compgen -v TESTCOMP").unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("TESTCOMPVAR"));
|
||||
}
|
||||
|
||||
// ===================== compgen -a: Aliases =====================
|
||||
|
||||
#[test]
|
||||
fn compgen_aliases() {
|
||||
let guard = TestGuard::new();
|
||||
test_input("alias testcompalias='echo hi'").unwrap();
|
||||
guard.read_output();
|
||||
|
||||
test_input("compgen -a testcomp").unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("testcompalias"));
|
||||
}
|
||||
|
||||
// ===================== compgen -d: Directories =====================
|
||||
|
||||
#[test]
|
||||
fn compgen_dirs() {
|
||||
let guard = TestGuard::new();
|
||||
let tmp = TempDir::new().unwrap();
|
||||
let sub = tmp.path().join("subdir");
|
||||
fs::create_dir(&sub).unwrap();
|
||||
|
||||
let prefix = format!("{}/", tmp.path().display());
|
||||
test_input(format!("compgen -d {prefix}")).unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("subdir"));
|
||||
}
|
||||
|
||||
// ===================== compgen -f: Files =====================
|
||||
|
||||
#[test]
|
||||
fn compgen_files() {
|
||||
let guard = TestGuard::new();
|
||||
let tmp = TempDir::new().unwrap();
|
||||
fs::write(tmp.path().join("testfile.txt"), "").unwrap();
|
||||
fs::create_dir(tmp.path().join("testdir")).unwrap();
|
||||
|
||||
let prefix = format!("{}/test", tmp.path().display());
|
||||
test_input(format!("compgen -f {prefix}")).unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("testfile.txt"));
|
||||
assert!(out.contains("testdir"));
|
||||
}
|
||||
|
||||
// ===================== compgen -F: Completion function =====================
|
||||
|
||||
#[test]
|
||||
fn compgen_function() {
|
||||
let guard = TestGuard::new();
|
||||
// Define a completion function that sets COMPREPLY
|
||||
test_input("_mycomp() { COMPREPLY=(opt1 opt2 opt3); }").unwrap();
|
||||
guard.read_output();
|
||||
|
||||
test_input("compgen -F _mycomp").unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("opt1"));
|
||||
assert!(out.contains("opt2"));
|
||||
assert!(out.contains("opt3"));
|
||||
}
|
||||
|
||||
// ===================== compgen: combined flags =====================
|
||||
|
||||
#[test]
|
||||
fn compgen_wordlist_and_aliases() {
|
||||
let guard = TestGuard::new();
|
||||
test_input("alias testcga='true'").unwrap();
|
||||
guard.read_output();
|
||||
|
||||
test_input("compgen -W 'testcgw' -a testcg").unwrap();
|
||||
let out = guard.read_output();
|
||||
assert!(out.contains("testcgw"));
|
||||
assert!(out.contains("testcga"));
|
||||
}
|
||||
|
||||
// ===================== Status =====================
|
||||
|
||||
#[test]
|
||||
fn complete_status_zero() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("complete -W 'x' mycmd").unwrap();
|
||||
assert_eq!(state::get_status(), 0);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn compgen_status_zero() {
|
||||
let _g = TestGuard::new();
|
||||
test_input("compgen -W 'hello'").unwrap();
|
||||
assert_eq!(state::get_status(), 0);
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user