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:
parent
fa867e93ea
commit
e253406026
7 changed files with 84 additions and 4 deletions
3
Cargo.lock
generated
3
Cargo.lock
generated
|
@ -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)",
|
||||||
]
|
]
|
||||||
|
|
20
src/du/du.rs
20
src/du/du.rs
|
@ -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
BIN
tests/fixtures/du/subdir/subwords.txt
vendored
Normal file
Binary file not shown.
1
tests/fixtures/du/subdir/subwords2.txt
vendored
Normal file
1
tests/fixtures/du/subdir/subwords2.txt
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
hello
|
1
tests/fixtures/du/words.txt
vendored
Normal file
1
tests/fixtures/du/words.txt
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
hello
|
62
tests/test_du.rs
Normal file
62
tests/test_du.rs
Normal 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);
|
||||||
|
*/
|
|
@ -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;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue