1
Fork 0
mirror of https://github.com/RGBCube/uutils-coreutils synced 2025-07-27 11:07:44 +00:00

du: Fix incorrect block size assumption.

du and other tools like stat assume a 512 byte block. ls is the only
tool to use 1024.

Add Simple set of tests
This commit is contained in:
bootandy 2018-03-20 12:10:05 -04:00
parent fa867e93ea
commit e253406026
7 changed files with 84 additions and 4 deletions

3
Cargo.lock generated
View file

@ -791,6 +791,8 @@ version = "0.0.1"
dependencies = [ dependencies = [
"getopts 0.2.17 (registry+https://github.com/rust-lang/crates.io-index)", "getopts 0.2.17 (registry+https://github.com/rust-lang/crates.io-index)",
"nix 0.8.1 (registry+https://github.com/rust-lang/crates.io-index)", "nix 0.8.1 (registry+https://github.com/rust-lang/crates.io-index)",
"redox_syscall 0.1.37 (registry+https://github.com/rust-lang/crates.io-index)",
"redox_termios 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
"uucore 0.0.1", "uucore 0.0.1",
] ]
@ -1430,6 +1432,7 @@ dependencies = [
"getopts 0.2.17 (registry+https://github.com/rust-lang/crates.io-index)", "getopts 0.2.17 (registry+https://github.com/rust-lang/crates.io-index)",
"kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", "kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)",
"libc 0.2.39 (registry+https://github.com/rust-lang/crates.io-index)", "libc 0.2.39 (registry+https://github.com/rust-lang/crates.io-index)",
"redox_syscall 0.1.37 (registry+https://github.com/rust-lang/crates.io-index)",
"uucore 0.0.1", "uucore 0.0.1",
"winapi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)", "winapi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)",
] ]

View file

@ -27,7 +27,7 @@ const LONG_HELP: &'static str = "
Display values are in units of the first available SIZE from Display values are in units of the first available SIZE from
--block-size, and the DU_BLOCK_SIZE, BLOCK_SIZE and BLOCKSIZE environ --block-size, and the DU_BLOCK_SIZE, BLOCK_SIZE and BLOCKSIZE environ
ment variables. Otherwise, units default to 1024 bytes (or 512 if ment variables. Otherwise, units default to 1024 bytes (or 512 if
POSIXLY_CORRECT is set). POSIXLY_CORRECT is set or if OS is Apple).
SIZE is an integer and optional unit (example: 10M is 10*1024*1024). SIZE is an integer and optional unit (example: 10M is 10*1024*1024).
Units are K, M, G, T, P, E, Z, Y (powers of 1024) or KB, MB, ... (pow Units are K, M, G, T, P, E, Z, Y (powers of 1024) or KB, MB, ... (pow
@ -69,6 +69,16 @@ impl Stat {
} }
} }
#[cfg(target_os = "macos")]
fn get_default_blocks() -> u64 {
512
}
#[cfg(not(target_os = "macos"))]
fn get_default_blocks() -> u64 {
1024
}
// this takes `my_stat` to avoid having to stat files multiple times. // this takes `my_stat` to avoid having to stat files multiple times.
// XXX: this should use the impl Trait return type when it is stabilized // XXX: this should use the impl Trait return type when it is stabilized
fn du(mut my_stat: Stat, options: &Options, depth: usize) -> Box<DoubleEndedIterator<Item = Stat>> { fn du(mut my_stat: Stat, options: &Options, depth: usize) -> Box<DoubleEndedIterator<Item = Stat>> {
@ -285,7 +295,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
}; };
number * multiple number * multiple
} }
None => 1024, None => get_default_blocks(),
}; };
let convert_size = |size: u64| -> String { let convert_size = |size: u64| -> String {
@ -332,7 +342,7 @@ Try '{} --help' for more information.",
let mut grand_total = 0; let mut grand_total = 0;
for path_str in strs.into_iter() { for path_str in strs.into_iter() {
let path = PathBuf::from(path_str); let path = PathBuf::from(&path_str);
match Stat::new(path) { match Stat::new(path) {
Ok(stat) => { Ok(stat) => {
let iter = du(stat, &options, 0).into_iter(); let iter = du(stat, &options, 0).into_iter();
@ -398,7 +408,9 @@ Try '{} --help' for more information.",
} }
} }
} }
Err(error) => show_error!("{}", error), Err(_) => {
show_info!("{}: {}", path_str, "No such file or directory");
}
} }
} }

BIN
tests/fixtures/du/subdir/subwords.txt vendored Normal file

Binary file not shown.

View file

@ -0,0 +1 @@
hello

1
tests/fixtures/du/words.txt vendored Normal file
View file

@ -0,0 +1 @@
hello

62
tests/test_du.rs Normal file
View file

@ -0,0 +1,62 @@
use common::util::*;
use std::fs::set_permissions;
static SUB_DIR: &str = "subdir";
static SUB_FILE: &str = "subdir/subwords.txt";
static SUB_LINK: &str = "subdir/sublink.txt";
#[test]
fn test_du_basics() {
let (_at, mut ucmd) = at_and_ucmd!();
let result = ucmd.run();
assert!(result.success);
assert_eq!(result.stderr, "");
assert_eq!(result.stdout, "24\t./subdir\n32\t./\n");
}
#[test]
fn test_du_basics_subdir() {
let (_at, mut ucmd) = at_and_ucmd!();
let result = ucmd.arg(SUB_DIR).run();
assert!(result.success);
assert_eq!(result.stderr, "");
assert_eq!(result.stdout, "24\tsubdir\n");
}
#[test]
fn test_du_basics_bad_name() {
let (_at, mut ucmd) = at_and_ucmd!();
let result = ucmd.arg("bad_name").run();
assert_eq!(result.stdout, "");
assert_eq!(result.stderr, "du: bad_name: No such file or directory\n");
}
#[test]
fn test_du_soft_link() {
let ts = TestScenario::new("du");
let link = ts.cmd("ln").arg("-s").arg(SUB_FILE).arg(SUB_LINK).run();
assert!(link.success);
let result = ts.ucmd().arg(SUB_DIR).run();
assert!(result.success);
assert_eq!(result.stderr, "");
assert_eq!(result.stdout, "32\tsubdir\n");
}
// todo:
// du on file with no permissions
// du on soft link
// du on hard link
// du on multi dir with '-d'
//
/*
* let mut permissions = at.make_file(TEST_HELLO_WORLD_DEST)
* .metadata()
* .unwrap()
* .permissions();
* permissions.set_readonly(true);
*/

View file

@ -52,6 +52,7 @@ generic! {
"cut", test_cut; "cut", test_cut;
"dircolors", test_dircolors; "dircolors", test_dircolors;
"dirname", test_dirname; "dirname", test_dirname;
"du", test_du;
"echo", test_echo; "echo", test_echo;
"env", test_env; "env", test_env;
"expr", test_expr; "expr", test_expr;