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(());
};
if x > BigUint::from_u32(1).unwrap() {
if x <= BigUint::from_u64(u64::MAX).unwrap() {
let prime_factors = num_prime::nt_funcs::factorize64(x.clone().to_u64_digits()[0]);
write_result_u64(w, &x, prime_factors, print_exponents)
.map_err_context(|| translate!("factor-error-write-error"))?;
}
else if x <= BigUint::from_u128(u128::MAX).unwrap() {
let rx = num_str.trim().parse::<u128>();
let Ok(x) = rx else {
show_warning!("{}: {}", num_str.maybe_quote(), rx.unwrap_err());
set_exit_code(1);
return Ok(());
};
let prime_factors = num_prime::nt_funcs::factorize128(x);
write_result_u128(w, &x, prime_factors, print_exponents)
.map_err_context(|| translate!("factor-error-write-error"))?;
}
else {
let (prime_factors, remaining) = num_prime::nt_funcs::factors(x.clone(), None);
if let Some(_remaining) = remaining {
return Err(USimpleError::new(
1,
translate!("factor-error-factorization-incomplete"),
));
}
write_result_big_uint(w, &x, prime_factors, print_exponents)
.map_err_context(|| translate!("factor-error-write-error"))?;
}
} else {
let empty_primes: BTreeMap<BigUint, usize> = BTreeMap::new();
write_result_big_uint(w, &x, empty_primes, print_exponents)
.map_err_context(|| translate!("factor-error-write-error"))?;
}
Ok(())
}
fn write_result_u64(
w: &mut io::BufWriter<impl Write>,
x: &BigUint,
factorization: BTreeMap<u64, 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()
}
fn write_result_u128(
w: &mut io::BufWriter<impl Write>,
x: &u128,
factorization: BTreeMap<u128, 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()
}
fn write_result_big_uint(
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),
)
}