use clap::Command;
use coreutils::validation;
use itertools::Itertools as _;
use std::cmp;
use std::ffi::OsString;
use std::io::{self, Write};
use std::process;
use uucore::{Args, error::strip_errno};
const VERSION: &str = env!("CARGO_PKG_VERSION");
include!(concat!(env!("OUT_DIR"), "/uutils_map.rs"));
fn usage<T>(utils: &UtilityMap<T>, name: &str) {
let display_list = utils.keys().copied().join(", ");
let width = cmp::min(textwrap::termwidth(), 100) - 8; let indent_list = textwrap::indent(&textwrap::fill(&display_list, width), " ");
#[cfg(feature = "feat_common_core")]
let common_core_string = "
Functions:
'<uutils>' [arguments...]
";
#[cfg(not(feature = "feat_common_core"))]
let common_core_string = "";
let s = format!(
"{name} {VERSION} (multi-call binary)
Usage: {name} [function [arguments...]]
{name} --list
{common_core_string}Options:
--list lists all defined functions, one per row
Currently defined functions:
{indent_list}"
);
if let Err(e) = writeln!(io::stdout(), "{s}")
&& e.kind() != io::ErrorKind::BrokenPipe
{
let _ = writeln!(io::stderr(), "coreutils: {}", strip_errno(&e));
process::exit(1);
}
}
#[allow(clippy::cognitive_complexity)]
fn main() {
let utils = util_map();
let mut args = uucore::args_os();
let binary = validation::binary_path(&mut args);
let binary_as_util = validation::name(&binary).unwrap_or_else(|| {
usage(&utils, "<unknown binary name>");
process::exit(0);
});
let is_coreutils = binary_as_util.ends_with("utils");
let matched_util = utils
.keys()
.filter(|&&u| binary_as_util.ends_with(u) && !is_coreutils)
.max_by_key(|u| u.len());
let util_name = if let Some(&util) = matched_util {
Some(OsString::from(util))
} else if is_coreutils || binary_as_util.ends_with("box") {
uucore::set_utility_is_second_arg();
args.next()
} else {
validation::not_found(&OsString::from(binary_as_util));
};
if let Some(util_os) = util_name {
let Some(util) = util_os.to_str() else {
validation::not_found(&util_os)
};
match util {
"--list" => {
if args.next().is_some() {
let _ = writeln!(io::stderr(), "coreutils: invalid argument");
process::exit(1);
}
let mut out = io::stdout().lock();
for util in utils.keys() {
if let Err(e) = writeln!(out, "{util}")
&& e.kind() != io::ErrorKind::BrokenPipe
{
let _ = writeln!(io::stderr(), "coreutils: {}", strip_errno(&e));
process::exit(1);
}
}
process::exit(0);
}
"--version" | "-V" => {
if let Err(e) = writeln!(io::stdout(), "coreutils {VERSION} (multi-call binary)")
&& e.kind() != io::ErrorKind::BrokenPipe
{
let _ = writeln!(io::stderr(), "coreutils: {}", strip_errno(&e));
process::exit(1);
}
process::exit(0);
}
_ => {}
}
match utils.get(util) {
Some(&(uumain, _)) => {
validation::setup_localization_or_exit(util);
process::exit(uumain(vec![util_os].into_iter().chain(args)));
}
None => {
if util == "--help" || util == "-h" {
if let Some(util_os) = args.next() {
let Some(util) = util_os.to_str() else {
validation::not_found(&util_os)
};
match utils.get(util) {
Some(&(uumain, _)) => {
let code = uumain(
vec![util_os, OsString::from("--help")]
.into_iter()
.chain(args),
);
io::stdout().flush().expect("could not flush stdout");
process::exit(code);
}
None => validation::not_found(&util_os),
}
}
usage(&utils, binary_as_util);
process::exit(0);
} else if util.starts_with('-') {
validation::unrecognized_option(binary_as_util, &util_os);
} else {
validation::not_found(&util_os);
}
}
}
} else {
if binary_as_util.ends_with("box") {
usage(&utils, binary_as_util);
} else {
let _ = writeln!(io::stderr(), "coreutils: missing argument");
}
process::exit(1);
}
}