#[cfg(not(unix))]
fn main() {
eprintln!("whoami: only available on Unix");
std::process::exit(1);
}
#[cfg(unix)]
use std::ffi::CStr;
#[cfg(unix)]
use std::io::Write;
#[cfg(unix)]
use std::process;
#[cfg(unix)]
const TOOL_NAME: &str = "whoami";
#[cfg(unix)]
const VERSION: &str = env!("CARGO_PKG_VERSION");
#[cfg(unix)]
fn main() {
coreutils_rs::common::reset_sigpipe();
if let Some(arg) = std::env::args_os().nth(1) {
let arg = arg.to_string_lossy();
match arg.as_ref() {
"--help" => {
println!("Usage: {}", TOOL_NAME);
println!(" or: {} OPTION", TOOL_NAME);
println!("Print the user name associated with the current effective user ID.");
println!("Same as id -un.");
println!();
println!(" --help display this help and exit");
println!(" --version output version information and exit");
return;
}
"--version" => {
println!("{} (fcoreutils) {}", TOOL_NAME, VERSION);
return;
}
_ => {
eprintln!("{}: extra operand '{}'", TOOL_NAME, arg);
eprintln!("Try '{} --help' for more information.", TOOL_NAME);
process::exit(1);
}
}
}
let uid = unsafe { libc::geteuid() };
let pw = unsafe { libc::getpwuid(uid) };
if pw.is_null() {
eprintln!("{}: cannot find name for user ID {}", TOOL_NAME, uid);
process::exit(1);
}
let name = unsafe { CStr::from_ptr((*pw).pw_name) };
let bytes = name.to_bytes();
let stdout = std::io::stdout();
let mut out = stdout.lock();
let _ = out.write_all(bytes);
let _ = out.write_all(b"\n");
}
#[cfg(all(test, unix))]
mod tests {
use std::process::Command;
fn cmd() -> Command {
let mut path = std::env::current_exe().unwrap();
path.pop();
path.pop();
path.push("fwhoami");
Command::new(path)
}
#[test]
fn test_whoami_prints_username() {
let output = cmd().output().unwrap();
assert_eq!(output.status.code(), Some(0));
let stdout = String::from_utf8_lossy(&output.stdout);
let name = stdout.trim();
assert!(!name.is_empty(), "whoami should output a non-empty name");
}
#[test]
fn test_whoami_matches_gnu() {
let gnu = Command::new("whoami").output();
if let Ok(gnu) = gnu {
let ours = cmd().output().unwrap();
assert_eq!(ours.stdout, gnu.stdout, "STDOUT mismatch");
assert_eq!(ours.status.code(), gnu.status.code(), "Exit code mismatch");
}
}
#[test]
fn test_whoami_no_args_accepted() {
let output = cmd().arg("extra").output().unwrap();
assert_eq!(output.status.code(), Some(1));
}
#[test]
fn test_whoami_basic() {
let output = cmd().output().unwrap();
assert!(output.status.success());
let stdout = String::from_utf8_lossy(&output.stdout);
assert!(!stdout.trim().is_empty());
}
#[test]
fn test_whoami_matches_id() {
let id_out = Command::new("id").arg("-un").output();
if let Ok(id_out) = id_out {
let ours = cmd().output().unwrap();
assert_eq!(
String::from_utf8_lossy(&ours.stdout).trim(),
String::from_utf8_lossy(&id_out.stdout).trim()
);
}
}
#[test]
fn test_whoami_single_line() {
let output = cmd().output().unwrap();
assert!(output.status.success());
let stdout = String::from_utf8_lossy(&output.stdout);
assert_eq!(stdout.lines().count(), 1);
}
}