mirror of
https://github.com/uutils/coreutils
synced 2024-12-14 15:22:38 +00:00
303 lines
9.1 KiB
Rust
303 lines
9.1 KiB
Rust
// This file is part of the uutils coreutils package.
|
|
//
|
|
// For the full copyright and license information, please view the LICENSE
|
|
// file that was distributed with this source code.
|
|
|
|
use libc::{close, dup, dup2, pipe, STDERR_FILENO, STDOUT_FILENO};
|
|
use rand::prelude::SliceRandom;
|
|
use rand::Rng;
|
|
use std::ffi::OsString;
|
|
use std::io;
|
|
use std::io::Write;
|
|
use std::os::fd::RawFd;
|
|
use std::process::Command;
|
|
use std::sync::atomic::Ordering;
|
|
use std::sync::{atomic::AtomicBool, Once};
|
|
|
|
/// Represents the result of running a command, including its standard output,
|
|
/// standard error, and exit code.
|
|
pub struct CommandResult {
|
|
/// The standard output (stdout) of the command as a string.
|
|
pub stdout: String,
|
|
|
|
/// The standard error (stderr) of the command as a string.
|
|
pub stderr: String,
|
|
|
|
/// The exit code of the command.
|
|
pub exit_code: i32,
|
|
}
|
|
|
|
static CHECK_GNU: Once = Once::new();
|
|
static IS_GNU: AtomicBool = AtomicBool::new(false);
|
|
|
|
pub fn is_gnu_cmd(cmd_path: &str) -> Result<(), std::io::Error> {
|
|
CHECK_GNU.call_once(|| {
|
|
let version_output = Command::new(cmd_path).arg("--version").output().unwrap();
|
|
|
|
println!("version_output {:#?}", version_output);
|
|
|
|
let version_str = String::from_utf8_lossy(&version_output.stdout).to_string();
|
|
if version_str.contains("GNU coreutils") {
|
|
IS_GNU.store(true, Ordering::Relaxed);
|
|
}
|
|
});
|
|
|
|
if IS_GNU.load(Ordering::Relaxed) {
|
|
Ok(())
|
|
} else {
|
|
panic!("Not the GNU implementation");
|
|
}
|
|
}
|
|
|
|
pub fn generate_and_run_uumain<F>(args: &[OsString], uumain_function: F) -> CommandResult
|
|
where
|
|
F: FnOnce(std::vec::IntoIter<OsString>) -> i32,
|
|
{
|
|
// Duplicate the stdout and stderr file descriptors
|
|
let original_stdout_fd = unsafe { dup(STDOUT_FILENO) };
|
|
let original_stderr_fd = unsafe { dup(STDERR_FILENO) };
|
|
if original_stdout_fd == -1 || original_stderr_fd == -1 {
|
|
return CommandResult {
|
|
stdout: "Failed to duplicate STDOUT_FILENO or STDERR_FILENO".to_string(),
|
|
stderr: "".to_string(),
|
|
exit_code: -1,
|
|
};
|
|
}
|
|
println!("Running test {:?}", &args[0..]);
|
|
let mut pipe_stdout_fds = [-1; 2];
|
|
let mut pipe_stderr_fds = [-1; 2];
|
|
|
|
// Create pipes for stdout and stderr
|
|
if unsafe { pipe(pipe_stdout_fds.as_mut_ptr()) } == -1
|
|
|| unsafe { pipe(pipe_stderr_fds.as_mut_ptr()) } == -1
|
|
{
|
|
return CommandResult {
|
|
stdout: "Failed to create pipes".to_string(),
|
|
stderr: "".to_string(),
|
|
exit_code: -1,
|
|
};
|
|
}
|
|
|
|
// Redirect stdout and stderr to their respective pipes
|
|
if unsafe { dup2(pipe_stdout_fds[1], STDOUT_FILENO) } == -1
|
|
|| unsafe { dup2(pipe_stderr_fds[1], STDERR_FILENO) } == -1
|
|
{
|
|
unsafe {
|
|
close(pipe_stdout_fds[0]);
|
|
close(pipe_stdout_fds[1]);
|
|
close(pipe_stderr_fds[0]);
|
|
close(pipe_stderr_fds[1]);
|
|
}
|
|
return CommandResult {
|
|
stdout: "Failed to redirect STDOUT_FILENO or STDERR_FILENO".to_string(),
|
|
stderr: "".to_string(),
|
|
exit_code: -1,
|
|
};
|
|
}
|
|
|
|
let uumain_exit_status = uumain_function(args.to_owned().into_iter());
|
|
|
|
io::stdout().flush().unwrap();
|
|
io::stderr().flush().unwrap();
|
|
|
|
// Restore the original stdout and stderr
|
|
if unsafe { dup2(original_stdout_fd, STDOUT_FILENO) } == -1
|
|
|| unsafe { dup2(original_stderr_fd, STDERR_FILENO) } == -1
|
|
{
|
|
return CommandResult {
|
|
stdout: "Failed to restore the original STDOUT_FILENO or STDERR_FILENO".to_string(),
|
|
stderr: "".to_string(),
|
|
exit_code: -1,
|
|
};
|
|
}
|
|
unsafe {
|
|
close(original_stdout_fd);
|
|
close(original_stderr_fd);
|
|
|
|
close(pipe_stdout_fds[1]);
|
|
close(pipe_stderr_fds[1]);
|
|
}
|
|
|
|
let captured_stdout = read_from_fd(pipe_stdout_fds[0]).trim().to_string();
|
|
let captured_stderr = read_from_fd(pipe_stderr_fds[0]).to_string();
|
|
let captured_stderr = captured_stderr
|
|
.split_once(':')
|
|
.map(|x| x.1)
|
|
.unwrap_or("")
|
|
.trim()
|
|
.to_string();
|
|
|
|
CommandResult {
|
|
stdout: captured_stdout,
|
|
stderr: captured_stderr,
|
|
exit_code: uumain_exit_status,
|
|
}
|
|
}
|
|
|
|
fn read_from_fd(fd: RawFd) -> String {
|
|
let mut captured_output = Vec::new();
|
|
let mut read_buffer = [0; 1024];
|
|
loop {
|
|
let bytes_read = unsafe {
|
|
libc::read(
|
|
fd,
|
|
read_buffer.as_mut_ptr() as *mut libc::c_void,
|
|
read_buffer.len(),
|
|
)
|
|
};
|
|
|
|
if bytes_read == -1 {
|
|
eprintln!("Failed to read from the pipe");
|
|
break;
|
|
}
|
|
if bytes_read == 0 {
|
|
break;
|
|
}
|
|
captured_output.extend_from_slice(&read_buffer[..bytes_read as usize]);
|
|
}
|
|
|
|
unsafe { libc::close(fd) };
|
|
|
|
String::from_utf8_lossy(&captured_output).into_owned()
|
|
}
|
|
|
|
pub fn run_gnu_cmd(
|
|
cmd_path: &str,
|
|
args: &[OsString],
|
|
check_gnu: bool,
|
|
) -> Result<CommandResult, CommandResult> {
|
|
if check_gnu {
|
|
match is_gnu_cmd(cmd_path) {
|
|
Ok(_) => {} // if the check passes, do nothing
|
|
Err(e) => {
|
|
// Convert the io::Error into the function's error type
|
|
return Err(CommandResult {
|
|
stdout: String::new(),
|
|
stderr: e.to_string(),
|
|
exit_code: -1,
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
let mut command = Command::new(cmd_path);
|
|
for arg in args {
|
|
command.arg(arg);
|
|
}
|
|
|
|
let output = match command.output() {
|
|
Ok(output) => output,
|
|
Err(e) => {
|
|
return Err(CommandResult {
|
|
stdout: String::new(),
|
|
stderr: e.to_string(),
|
|
exit_code: -1,
|
|
});
|
|
}
|
|
};
|
|
let exit_code = output.status.code().unwrap_or(-1);
|
|
|
|
// Here we get stdout and stderr as Strings
|
|
let stdout = String::from_utf8_lossy(&output.stdout).to_string();
|
|
let stderr = String::from_utf8_lossy(&output.stderr).to_string();
|
|
let stderr = stderr
|
|
.split_once(':')
|
|
.map(|x| x.1)
|
|
.unwrap_or("")
|
|
.trim()
|
|
.to_string();
|
|
|
|
if output.status.success() || !check_gnu {
|
|
Ok(CommandResult {
|
|
stdout,
|
|
stderr,
|
|
exit_code,
|
|
})
|
|
} else {
|
|
Err(CommandResult {
|
|
stdout,
|
|
stderr,
|
|
exit_code,
|
|
})
|
|
}
|
|
}
|
|
|
|
/// Compare results from two different implementations of a command.
|
|
///
|
|
/// # Arguments
|
|
/// * `test_type` - The command.
|
|
/// * `input` - The input provided to the command.
|
|
/// * `rust_result` - The result of running the command with the Rust implementation.
|
|
/// * `gnu_result` - The result of running the command with the GNU implementation.
|
|
/// * `fail_on_stderr_diff` - Whether to fail the test if there is a difference in stderr output.
|
|
pub fn compare_result(
|
|
test_type: &str,
|
|
input: &str,
|
|
rust_result: &CommandResult,
|
|
gnu_result: &CommandResult,
|
|
fail_on_stderr_diff: bool,
|
|
) {
|
|
println!("Test Type: {}", test_type);
|
|
println!("Input: {}", input);
|
|
|
|
let mut discrepancies = Vec::new();
|
|
let mut should_panic = false;
|
|
|
|
if rust_result.stdout.trim() != gnu_result.stdout.trim() {
|
|
discrepancies.push("stdout differs");
|
|
println!("Rust stdout: {}", rust_result.stdout);
|
|
println!("GNU stdout: {}", gnu_result.stdout);
|
|
should_panic = true;
|
|
}
|
|
if rust_result.stderr.trim() != gnu_result.stderr.trim() {
|
|
discrepancies.push("stderr differs");
|
|
println!("Rust stderr: {}", rust_result.stderr);
|
|
println!("GNU stderr: {}", gnu_result.stderr);
|
|
if fail_on_stderr_diff {
|
|
should_panic = true;
|
|
}
|
|
}
|
|
if rust_result.exit_code != gnu_result.exit_code {
|
|
discrepancies.push("exit code differs");
|
|
println!("Rust exit code: {}", rust_result.exit_code);
|
|
println!("GNU exit code: {}", gnu_result.exit_code);
|
|
should_panic = true;
|
|
}
|
|
|
|
if discrepancies.is_empty() {
|
|
println!("All outputs and exit codes matched.");
|
|
} else {
|
|
println!("Discrepancy detected: {}", discrepancies.join(", "));
|
|
if should_panic {
|
|
panic!("Test failed for {}: {}", test_type, input);
|
|
} else {
|
|
println!(
|
|
"Test completed with discrepancies for {}: {}",
|
|
test_type, input
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn generate_random_string(max_length: usize) -> String {
|
|
let mut rng = rand::thread_rng();
|
|
let valid_utf8: Vec<char> = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"
|
|
.chars()
|
|
.collect();
|
|
let invalid_utf8 = [0xC3, 0x28]; // Invalid UTF-8 sequence
|
|
let mut result = String::new();
|
|
|
|
for _ in 0..rng.gen_range(1..=max_length) {
|
|
if rng.gen_bool(0.9) {
|
|
let ch = valid_utf8.choose(&mut rng).unwrap();
|
|
result.push(*ch);
|
|
} else {
|
|
let ch = invalid_utf8.choose(&mut rng).unwrap();
|
|
if let Some(c) = char::from_u32(*ch as u32) {
|
|
result.push(c);
|
|
}
|
|
}
|
|
}
|
|
|
|
result
|
|
}
|