use std::collections::BTreeMap;
use std::io::BufRead;
use std::io::{self, Write, stdin, stdout};
use clap::{Arg, ArgAction, Command};
use num_bigint::BigUint;
use num_traits::FromPrimitive;
use uucore::display::Quotable;
use uucore::error::{FromIo, UResult, USimpleError, set_exit_code};
use uucore::translate;
use uucore::{format_usage, show_error, show_warning};
mod options {
pub static EXPONENTS: &str = "exponents";
pub static HELP: &str = "help";
pub static NUMBER: &str = "NUMBER";
}
fn print_factors_str(
num_str: &str,
w: &mut io::BufWriter<impl Write>,
print_exponents: bool,
) -> UResult<()> {
let rx = num_str.trim().parse::<BigUint>();
let Ok(x) = rx else {
show_warning!("{}: {}", num_str.maybe_quote(), rx.unwrap_err());
set_exit_code(1);
return Ok(());
};
let (factorization, remaining) = if x > BigUint::from_u32(1).unwrap() {
num_prime::nt_funcs::factors(x.clone(), None)
} else {
(BTreeMap::new(), None)
};
if let Some(_remaining) = remaining {
return Err(USimpleError::new(
1,
translate!("factor-error-factorization-incomplete"),
));
}
write_result(w, &x, factorization, print_exponents)
.map_err_context(|| translate!("factor-error-write-error"))?;
Ok(())
}
fn write_result(
w: &mut io::BufWriter<impl Write>,
x: &BigUint,
factorization: BTreeMap<BigUint, usize>,
print_exponents: bool,
) -> io::Result<()> {
write!(w, "{x}:")?;
for (factor, n) in factorization {
if print_exponents {
if n > 1 {
write!(w, " {factor}^{n}")?;
} else {
write!(w, " {factor}")?;
}
} else {
w.write_all(format!(" {factor}").repeat(n).as_bytes())?;
}
}
writeln!(w)?;
w.flush()
}
#[uucore::main]
pub fn uumain(args: impl uucore::Args) -> UResult<()> {
let matches = uucore::clap_localization::handle_clap_result(uu_app(), args)?;
let print_exponents = matches.get_flag(options::EXPONENTS);
let stdout = stdout();
let mut w = io::BufWriter::with_capacity(4 * 1024, stdout.lock());
if let Some(values) = matches.get_many::<String>(options::NUMBER) {
for number in values {
print_factors_str(number, &mut w, print_exponents)?;
}
} else {
let stdin = stdin();
let lines = stdin.lock().lines();
for line in lines {
match line {
Ok(line) => {
for number in line.split_whitespace() {
print_factors_str(number, &mut w, print_exponents)?;
}
}
Err(e) => {
set_exit_code(1);
show_error!("{}", translate!("factor-error-reading-input", "error" => e));
return Ok(());
}
}
}
}
if let Err(e) = w.flush() {
show_error!("{e}");
}
Ok(())
}
pub fn uu_app() -> Command {
Command::new(uucore::util_name())
.version(uucore::crate_version!())
.help_template(uucore::localized_help_template(uucore::util_name()))
.about(translate!("factor-about"))
.override_usage(format_usage(&translate!("factor-usage")))
.infer_long_args(true)
.disable_help_flag(true)
.args_override_self(true)
.arg(Arg::new(options::NUMBER).action(ArgAction::Append))
.arg(
Arg::new(options::EXPONENTS)
.short('h')
.long(options::EXPONENTS)
.help(translate!("factor-help-exponents"))
.action(ArgAction::SetTrue),
)
.arg(
Arg::new(options::HELP)
.long(options::HELP)
.help(translate!("factor-help-help"))
.action(ArgAction::Help),
)
}