From c90a1c759a0bfd27c62a77d96a55ba9f4cf075a0 Mon Sep 17 00:00:00 2001 From: Daniel Hofstetter Date: Sun, 22 May 2022 16:54:51 +0200 Subject: [PATCH] df: fix rounding behavior in humanreadable mode Fixes #3422 --- Cargo.lock | 1 - src/uu/df/Cargo.toml | 1 - src/uu/df/src/blocks.rs | 221 +++++++++++++++++++--------------------- src/uu/df/src/df.rs | 14 +-- src/uu/df/src/table.rs | 59 +++-------- 5 files changed, 129 insertions(+), 167 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 40fdcfd2af8..7253d58a9ab 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2219,7 +2219,6 @@ name = "uu_df" version = "0.0.14" dependencies = [ "clap 3.1.18", - "number_prefix", "unicode-width", "uucore", ] diff --git a/src/uu/df/Cargo.toml b/src/uu/df/Cargo.toml index 8e4fc969840..ccbdc803b2b 100644 --- a/src/uu/df/Cargo.toml +++ b/src/uu/df/Cargo.toml @@ -16,7 +16,6 @@ path = "src/df.rs" [dependencies] clap = { version = "3.1", features = ["wrap_help", "cargo"] } -number_prefix = "0.4" uucore = { version=">=0.0.11", package="uucore", path="../../uucore", features=["libc", "fsext"] } unicode-width = "0.1.9" diff --git a/src/uu/df/src/blocks.rs b/src/uu/df/src/blocks.rs index 88190b5c187..2593c13dd54 100644 --- a/src/uu/df/src/blocks.rs +++ b/src/uu/df/src/blocks.rs @@ -26,9 +26,7 @@ const IEC_BASES: [u128; 10] = [ 1_237_940_039_285_380_274_899_124_224, ]; -/// Suffixes for the first nine multi-byte unit suffixes. -const SUFFIXES: [char; 9] = ['B', 'K', 'M', 'G', 'T', 'P', 'E', 'Z', 'Y']; - +/// The first ten powers of 1000. const SI_BASES: [u128; 10] = [ 1, 1_000, @@ -42,96 +40,71 @@ const SI_BASES: [u128; 10] = [ 1_000_000_000_000_000_000_000_000_000, ]; -// we use "kB" instead of "KB" because of GNU df -const SI_SUFFIXES: [&str; 9] = ["B", "kB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB"]; +/// A SuffixType determines whether the suffixes are 1000 or 1024 based, and whether they are +/// intended for HumanReadable mode or not. +#[derive(Clone, Copy)] +pub(crate) enum SuffixType { + Iec, + Si, + HumanReadable(HumanReadable), +} -/// Convert a multiple of 1024 into a string like "12K" or "34M". -/// -/// # Examples -/// -/// Powers of 1024 become "1K", "1M", "1G", etc. -/// -/// ```rust,ignore -/// assert_eq!(to_magnitude_and_suffix_1024(1024).unwrap(), "1K"); -/// assert_eq!(to_magnitude_and_suffix_1024(1024 * 1024).unwrap(), "1M"); -/// assert_eq!(to_magnitude_and_suffix_1024(1024 * 1024 * 1024).unwrap(), "1G"); -/// ``` -/// -/// Multiples of those powers affect the magnitude part of the -/// returned string: -/// -/// ```rust,ignore -/// assert_eq!(to_magnitude_and_suffix_1024(123 * 1024).unwrap(), "123K"); -/// assert_eq!(to_magnitude_and_suffix_1024(456 * 1024 * 1024).unwrap(), "456M"); -/// assert_eq!(to_magnitude_and_suffix_1024(789 * 1024 * 1024 * 1024).unwrap(), "789G"); -/// ``` -fn to_magnitude_and_suffix_1024(n: u128) -> Result { - // Find the smallest power of 1024 that is larger than `n`. That - // number indicates which units and suffix to use. - for i in 0..IEC_BASES.len() - 1 { - if n < IEC_BASES[i + 1] { - return Ok(format!("{}{}", n / IEC_BASES[i], SUFFIXES[i])); +impl SuffixType { + /// The first ten powers of 1024 and 1000, respectively. + fn bases(&self) -> [u128; 10] { + match self { + Self::Iec | Self::HumanReadable(HumanReadable::Binary) => IEC_BASES, + Self::Si | Self::HumanReadable(HumanReadable::Decimal) => SI_BASES, + } + } + + /// Suffixes for the first nine multi-byte unit suffixes. + fn suffixes(&self) -> [&'static str; 9] { + match self { + // we use "kB" instead of "KB", same as GNU df + Self::Si => ["B", "kB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB"], + Self::Iec => ["B", "K", "M", "G", "T", "P", "E", "Z", "Y"], + Self::HumanReadable(HumanReadable::Binary) => { + ["", "K", "M", "G", "T", "P", "E", "Z", "Y"] + } + Self::HumanReadable(HumanReadable::Decimal) => { + ["", "k", "M", "G", "T", "P", "E", "Z", "Y"] + } } } - Err(()) } -/// Convert a number into a string like "12kB" or "34MB". -/// -/// Powers of 1000 become "1kB", "1MB", "1GB", etc. +/// Convert a number into a magnitude and a multi-byte unit suffix. /// /// The returned string has a maximum length of 5 chars, for example: "1.1kB", "999kB", "1MB". -fn to_magnitude_and_suffix_not_powers_of_1024(n: u128) -> Result { +pub(crate) fn to_magnitude_and_suffix(n: u128, suffix_type: SuffixType) -> String { + let bases = suffix_type.bases(); + let suffixes = suffix_type.suffixes(); let mut i = 0; - while SI_BASES[i + 1] - SI_BASES[i] < n && i < SI_SUFFIXES.len() { + while bases[i + 1] - bases[i] < n && i < suffixes.len() { i += 1; } - let quot = n / SI_BASES[i]; - let rem = n % SI_BASES[i]; - let suffix = SI_SUFFIXES[i]; + let quot = n / bases[i]; + let rem = n % bases[i]; + let suffix = suffixes[i]; if rem == 0 { - Ok(format!("{}{}", quot, suffix)) + format!("{}{}", quot, suffix) } else { - let tenths_place = rem / (SI_BASES[i] / 10); + let tenths_place = rem / (bases[i] / 10); - if rem % (SI_BASES[i] / 10) == 0 { - Ok(format!("{}.{}{}", quot, tenths_place, suffix)) + if rem % (bases[i] / 10) == 0 { + format!("{}.{}{}", quot, tenths_place, suffix) } else if tenths_place + 1 == 10 || quot >= 10 { - Ok(format!("{}{}", quot + 1, suffix)) + format!("{}{}", quot + 1, suffix) } else { - Ok(format!("{}.{}{}", quot, tenths_place + 1, suffix)) + format!("{}.{}{}", quot, tenths_place + 1, suffix) } } } -/// Convert a number into a magnitude and a multi-byte unit suffix. -/// -/// # Errors -/// -/// If the number is too large to represent. -fn to_magnitude_and_suffix(n: u128) -> Result { - if n % 1024 == 0 && n % 1000 != 0 { - to_magnitude_and_suffix_1024(n) - } else { - to_magnitude_and_suffix_not_powers_of_1024(n) - } -} - -/// A mode to use in condensing the display of a large number of bytes. -pub(crate) enum SizeFormat { - HumanReadable(HumanReadable), - StaticBlockSize, -} - -impl Default for SizeFormat { - fn default() -> Self { - Self::StaticBlockSize - } -} - /// A mode to use in condensing the human readable display of a large number /// of bytes. /// @@ -207,10 +180,15 @@ pub(crate) fn block_size_from_matches(matches: &ArgMatches) -> Result fmt::Result { match self { - Self::Bytes(n) => match to_magnitude_and_suffix(*n as u128) { - Ok(s) => write!(f, "{}", s), - Err(_) => Err(fmt::Error), - }, + Self::Bytes(n) => { + let s = if n % 1024 == 0 && n % 1000 != 0 { + to_magnitude_and_suffix(*n as u128, SuffixType::Iec) + } else { + to_magnitude_and_suffix(*n as u128, SuffixType::Si) + }; + + write!(f, "{}", s) + } } } } @@ -220,56 +198,64 @@ mod tests { use std::env; - use crate::blocks::{to_magnitude_and_suffix, BlockSize}; + use crate::blocks::{to_magnitude_and_suffix, BlockSize, SuffixType}; #[test] fn test_to_magnitude_and_suffix_powers_of_1024() { - assert_eq!(to_magnitude_and_suffix(1024).unwrap(), "1K"); - assert_eq!(to_magnitude_and_suffix(2048).unwrap(), "2K"); - assert_eq!(to_magnitude_and_suffix(4096).unwrap(), "4K"); - assert_eq!(to_magnitude_and_suffix(1024 * 1024).unwrap(), "1M"); - assert_eq!(to_magnitude_and_suffix(2 * 1024 * 1024).unwrap(), "2M"); - assert_eq!(to_magnitude_and_suffix(1024 * 1024 * 1024).unwrap(), "1G"); + assert_eq!(to_magnitude_and_suffix(1024, SuffixType::Iec), "1K"); + assert_eq!(to_magnitude_and_suffix(2048, SuffixType::Iec), "2K"); + assert_eq!(to_magnitude_and_suffix(4096, SuffixType::Iec), "4K"); + assert_eq!(to_magnitude_and_suffix(1024 * 1024, SuffixType::Iec), "1M"); assert_eq!( - to_magnitude_and_suffix(34 * 1024 * 1024 * 1024).unwrap(), + to_magnitude_and_suffix(2 * 1024 * 1024, SuffixType::Iec), + "2M" + ); + assert_eq!( + to_magnitude_and_suffix(1024 * 1024 * 1024, SuffixType::Iec), + "1G" + ); + assert_eq!( + to_magnitude_and_suffix(34 * 1024 * 1024 * 1024, SuffixType::Iec), "34G" ); } #[test] fn test_to_magnitude_and_suffix_not_powers_of_1024() { - assert_eq!(to_magnitude_and_suffix(1).unwrap(), "1B"); - assert_eq!(to_magnitude_and_suffix(999).unwrap(), "999B"); - - assert_eq!(to_magnitude_and_suffix(1000).unwrap(), "1kB"); - assert_eq!(to_magnitude_and_suffix(1001).unwrap(), "1.1kB"); - assert_eq!(to_magnitude_and_suffix(1023).unwrap(), "1.1kB"); - assert_eq!(to_magnitude_and_suffix(1025).unwrap(), "1.1kB"); - assert_eq!(to_magnitude_and_suffix(10_001).unwrap(), "11kB"); - assert_eq!(to_magnitude_and_suffix(999_000).unwrap(), "999kB"); - - assert_eq!(to_magnitude_and_suffix(999_001).unwrap(), "1MB"); - assert_eq!(to_magnitude_and_suffix(999_999).unwrap(), "1MB"); - assert_eq!(to_magnitude_and_suffix(1_000_000).unwrap(), "1MB"); - assert_eq!(to_magnitude_and_suffix(1_000_001).unwrap(), "1.1MB"); - assert_eq!(to_magnitude_and_suffix(1_100_000).unwrap(), "1.1MB"); - assert_eq!(to_magnitude_and_suffix(1_100_001).unwrap(), "1.2MB"); - assert_eq!(to_magnitude_and_suffix(1_900_000).unwrap(), "1.9MB"); - assert_eq!(to_magnitude_and_suffix(1_900_001).unwrap(), "2MB"); - assert_eq!(to_magnitude_and_suffix(9_900_000).unwrap(), "9.9MB"); - assert_eq!(to_magnitude_and_suffix(9_900_001).unwrap(), "10MB"); - assert_eq!(to_magnitude_and_suffix(999_000_000).unwrap(), "999MB"); - - assert_eq!(to_magnitude_and_suffix(999_000_001).unwrap(), "1GB"); - assert_eq!(to_magnitude_and_suffix(1_000_000_000).unwrap(), "1GB"); - assert_eq!(to_magnitude_and_suffix(1_000_000_001).unwrap(), "1.1GB"); - } + assert_eq!(to_magnitude_and_suffix(1, SuffixType::Si), "1B"); + assert_eq!(to_magnitude_and_suffix(999, SuffixType::Si), "999B"); + + assert_eq!(to_magnitude_and_suffix(1000, SuffixType::Si), "1kB"); + assert_eq!(to_magnitude_and_suffix(1001, SuffixType::Si), "1.1kB"); + assert_eq!(to_magnitude_and_suffix(1023, SuffixType::Si), "1.1kB"); + assert_eq!(to_magnitude_and_suffix(1025, SuffixType::Si), "1.1kB"); + assert_eq!(to_magnitude_and_suffix(10_001, SuffixType::Si), "11kB"); + assert_eq!(to_magnitude_and_suffix(999_000, SuffixType::Si), "999kB"); + + assert_eq!(to_magnitude_and_suffix(999_001, SuffixType::Si), "1MB"); + assert_eq!(to_magnitude_and_suffix(999_999, SuffixType::Si), "1MB"); + assert_eq!(to_magnitude_and_suffix(1_000_000, SuffixType::Si), "1MB"); + assert_eq!(to_magnitude_and_suffix(1_000_001, SuffixType::Si), "1.1MB"); + assert_eq!(to_magnitude_and_suffix(1_100_000, SuffixType::Si), "1.1MB"); + assert_eq!(to_magnitude_and_suffix(1_100_001, SuffixType::Si), "1.2MB"); + assert_eq!(to_magnitude_and_suffix(1_900_000, SuffixType::Si), "1.9MB"); + assert_eq!(to_magnitude_and_suffix(1_900_001, SuffixType::Si), "2MB"); + assert_eq!(to_magnitude_and_suffix(9_900_000, SuffixType::Si), "9.9MB"); + assert_eq!(to_magnitude_and_suffix(9_900_001, SuffixType::Si), "10MB"); + assert_eq!( + to_magnitude_and_suffix(999_000_000, SuffixType::Si), + "999MB" + ); - #[test] - fn test_to_magnitude_and_suffix_multiples_of_1000_and_1024() { - assert_eq!(to_magnitude_and_suffix(128_000).unwrap(), "128kB"); - assert_eq!(to_magnitude_and_suffix(1000 * 1024).unwrap(), "1.1MB"); - assert_eq!(to_magnitude_and_suffix(1_000_000_000_000).unwrap(), "1TB"); + assert_eq!(to_magnitude_and_suffix(999_000_001, SuffixType::Si), "1GB"); + assert_eq!( + to_magnitude_and_suffix(1_000_000_000, SuffixType::Si), + "1GB" + ); + assert_eq!( + to_magnitude_and_suffix(1_000_000_001, SuffixType::Si), + "1.1GB" + ); } #[test] @@ -279,6 +265,13 @@ mod tests { assert_eq!(format!("{}", BlockSize::Bytes(3 * 1024 * 1024)), "3M"); } + #[test] + fn test_block_size_display_multiples_of_1000_and_1024() { + assert_eq!(format!("{}", BlockSize::Bytes(128_000)), "128kB"); + assert_eq!(format!("{}", BlockSize::Bytes(1000 * 1024)), "1.1MB"); + assert_eq!(format!("{}", BlockSize::Bytes(1_000_000_000_000)), "1TB"); + } + #[test] fn test_default_block_size() { assert_eq!(BlockSize::Bytes(1024), BlockSize::default()); diff --git a/src/uu/df/src/df.rs b/src/uu/df/src/df.rs index 7c95e693808..6e8482ffee1 100644 --- a/src/uu/df/src/df.rs +++ b/src/uu/df/src/df.rs @@ -11,7 +11,7 @@ mod columns; mod filesystem; mod table; -use blocks::{HumanReadable, SizeFormat}; +use blocks::HumanReadable; use table::HeaderMode; use uucore::display::Quotable; use uucore::error::{UError, UResult, USimpleError}; @@ -71,7 +71,7 @@ static OUTPUT_FIELD_LIST: [&str; 12] = [ struct Options { show_local_fs: bool, show_all_fs: bool, - size_format: SizeFormat, + human_readable: Option, block_size: BlockSize, header_mode: HeaderMode, @@ -100,7 +100,7 @@ impl Default for Options { show_local_fs: Default::default(), show_all_fs: Default::default(), block_size: Default::default(), - size_format: Default::default(), + human_readable: Default::default(), header_mode: Default::default(), include: Default::default(), exclude: Default::default(), @@ -200,13 +200,13 @@ impl Options { HeaderMode::Default } }, - size_format: { + human_readable: { if matches.is_present(OPT_HUMAN_READABLE_BINARY) { - SizeFormat::HumanReadable(HumanReadable::Binary) + Some(HumanReadable::Binary) } else if matches.is_present(OPT_HUMAN_READABLE_DECIMAL) { - SizeFormat::HumanReadable(HumanReadable::Decimal) + Some(HumanReadable::Decimal) } else { - SizeFormat::StaticBlockSize + None } }, include, diff --git a/src/uu/df/src/table.rs b/src/uu/df/src/table.rs index fac2ca66395..22d3a4437ec 100644 --- a/src/uu/df/src/table.rs +++ b/src/uu/df/src/table.rs @@ -7,10 +7,9 @@ //! //! A table ([`Table`]) comprises a header row ([`Header`]) and a //! collection of data rows ([`Row`]), one per filesystem. -use number_prefix::NumberPrefix; use unicode_width::UnicodeWidthStr; -use crate::blocks::{HumanReadable, SizeFormat}; +use crate::blocks::{to_magnitude_and_suffix, SuffixType}; use crate::columns::{Alignment, Column}; use crate::filesystem::Filesystem; use crate::{BlockSize, Options}; @@ -213,28 +212,15 @@ impl<'a> RowFormatter<'a> { Self { row, options } } - /// Get a human readable string giving the scaled version of the input number. - fn scaled_human_readable(&self, size: u64, human_readable: HumanReadable) -> String { - let number_prefix = match human_readable { - HumanReadable::Decimal => NumberPrefix::decimal(size as f64), - HumanReadable::Binary => NumberPrefix::binary(size as f64), - }; - match number_prefix { - NumberPrefix::Standalone(bytes) => bytes.to_string(), - NumberPrefix::Prefixed(prefix, bytes) => format!("{:.1}{}", bytes, prefix.symbol()), - } - } - /// Get a string giving the scaled version of the input number. /// /// The scaling factor is defined in the `options` field. fn scaled_bytes(&self, size: u64) -> String { - match self.options.size_format { - SizeFormat::HumanReadable(h) => self.scaled_human_readable(size, h), - SizeFormat::StaticBlockSize => { - let BlockSize::Bytes(d) = self.options.block_size; - (size as f64 / d as f64).ceil().to_string() - } + if let Some(h) = self.options.human_readable { + to_magnitude_and_suffix(size.into(), SuffixType::HumanReadable(h)) + } else { + let BlockSize::Bytes(d) = self.options.block_size; + (size as f64 / d as f64).ceil().to_string() } } @@ -242,9 +228,10 @@ impl<'a> RowFormatter<'a> { /// /// The scaling factor is defined in the `options` field. fn scaled_inodes(&self, size: u64) -> String { - match self.options.size_format { - SizeFormat::HumanReadable(h) => self.scaled_human_readable(size, h), - SizeFormat::StaticBlockSize => size.to_string(), + if let Some(h) = self.options.human_readable { + to_magnitude_and_suffix(size.into(), SuffixType::HumanReadable(h)) + } else { + size.to_string() } } @@ -450,7 +437,7 @@ impl fmt::Display for Table { #[cfg(test)] mod tests { - use crate::blocks::{HumanReadable, SizeFormat}; + use crate::blocks::HumanReadable; use crate::columns::Column; use crate::table::{Header, HeaderMode, Row, RowFormatter}; use crate::{BlockSize, Options}; @@ -715,7 +702,7 @@ mod tests { #[test] fn test_row_formatter_with_human_readable_si() { let options = Options { - size_format: SizeFormat::HumanReadable(HumanReadable::Decimal), + human_readable: Some(HumanReadable::Decimal), columns: COLUMNS_WITH_FS_TYPE.to_vec(), ..Default::default() }; @@ -734,22 +721,14 @@ mod tests { let fmt = RowFormatter::new(&row, &options); assert_eq!( fmt.get_values(), - vec!( - "my_device", - "my_type", - "4.0k", - "1.0k", - "3.0k", - "25%", - "my_mount" - ) + vec!("my_device", "my_type", "4k", "1k", "3k", "25%", "my_mount") ); } #[test] fn test_row_formatter_with_human_readable_binary() { let options = Options { - size_format: SizeFormat::HumanReadable(HumanReadable::Binary), + human_readable: Some(HumanReadable::Binary), columns: COLUMNS_WITH_FS_TYPE.to_vec(), ..Default::default() }; @@ -768,15 +747,7 @@ mod tests { let fmt = RowFormatter::new(&row, &options); assert_eq!( fmt.get_values(), - vec!( - "my_device", - "my_type", - "4.0Ki", - "1.0Ki", - "3.0Ki", - "25%", - "my_mount" - ) + vec!("my_device", "my_type", "4K", "1K", "3K", "25%", "my_mount") ); }