use predicates::prelude::*;
use serial_test::serial;
use std::sync::OnceLock;
#[macro_use]
mod common;
use common::{
ensure_test_project, get_function_address, get_function_addresses, DaemonTestHarness,
};
const TEST_PROJECT: &str = "ci-test";
const TEST_PROGRAM: &str = "sample_binary";
static HARNESS: OnceLock<DaemonTestHarness> = OnceLock::new();
fn harness() -> &'static DaemonTestHarness {
HARNESS.get_or_init(|| {
ensure_test_project(TEST_PROJECT, TEST_PROGRAM);
DaemonTestHarness::new(TEST_PROJECT, TEST_PROGRAM).expect("Failed to start daemon")
})
}
#[test]
#[serial]
fn test_symbol_list() {
require_ghidra!();
let _harness = harness();
let output = assert_cmd::cargo::cargo_bin_cmd!("ghidra")
.arg("symbol")
.arg("list")
.arg("--project")
.arg(TEST_PROJECT)
.arg("--program")
.arg(TEST_PROGRAM)
.output()
.expect("Failed to run command");
assert!(output.status.success(), "symbol list should succeed");
let stdout = String::from_utf8_lossy(&output.stdout);
assert!(
stdout.contains("main") || stdout.contains("_main"),
"symbol list should contain main. Output: {}",
stdout
);
}
#[test]
#[serial]
fn test_symbol_create_and_get() {
require_ghidra!();
let harness = harness();
let addr = get_function_address(harness, TEST_PROJECT, TEST_PROGRAM, "main");
assert_cmd::cargo::cargo_bin_cmd!("ghidra")
.arg("symbol")
.arg("create")
.arg(&addr)
.arg("test_symbol")
.arg("--project")
.arg(TEST_PROJECT)
.arg("--program")
.arg(TEST_PROGRAM)
.assert()
.success();
assert_cmd::cargo::cargo_bin_cmd!("ghidra")
.arg("symbol")
.arg("get")
.arg("test_symbol")
.arg("--project")
.arg(TEST_PROJECT)
.arg("--program")
.arg(TEST_PROGRAM)
.assert()
.success()
.stdout(predicate::str::contains("test_symbol"));
}
#[test]
#[serial]
fn test_symbol_rename() {
require_ghidra!();
let harness = harness();
let addrs = get_function_addresses(harness, TEST_PROJECT, TEST_PROGRAM, 2);
let addr = &addrs[1];
let old_name = format!("old_sym_{}", std::process::id());
let new_name = format!("new_sym_{}", std::process::id());
assert_cmd::cargo::cargo_bin_cmd!("ghidra")
.arg("symbol")
.arg("create")
.arg(addr)
.arg(&old_name)
.arg("--project")
.arg(TEST_PROJECT)
.arg("--program")
.arg(TEST_PROGRAM)
.assert()
.success();
assert_cmd::cargo::cargo_bin_cmd!("ghidra")
.arg("symbol")
.arg("rename")
.arg(&old_name)
.arg(&new_name)
.arg("--project")
.arg(TEST_PROJECT)
.arg("--program")
.arg(TEST_PROGRAM)
.assert()
.success();
assert_cmd::cargo::cargo_bin_cmd!("ghidra")
.arg("symbol")
.arg("get")
.arg(&new_name)
.arg("--project")
.arg(TEST_PROJECT)
.arg("--program")
.arg(TEST_PROGRAM)
.assert()
.success()
.stdout(predicate::str::contains(&*new_name));
}
#[test]
#[serial]
fn test_symbol_get_nonexistent() {
require_ghidra!();
let _harness = harness();
assert_cmd::cargo::cargo_bin_cmd!("ghidra")
.arg("symbol")
.arg("get")
.arg("nonexistent_symbol_12345")
.arg("--project")
.arg(TEST_PROJECT)
.arg("--program")
.arg(TEST_PROGRAM)
.assert()
.failure();
}