Compare commits

...

12 Commits

Author SHA1 Message Date
andy.boot ba04d4c42c Increment version 2018-04-27 12:44:24 +01:00
andy boot 6842526d2c Merge pull request #15 from bootandy/depth2
Add support for -d depth flag
2018-04-27 12:28:15 +01:00
andy.boot 4ac85d7dc9 Simplify tests
dust sort is now more predictable as it orders first by size and second
by name.

Walkdir still gives different iteration orders on different systems so
all output is not entirely predictable
2018-04-27 11:35:47 +01:00
andy.boot 8170a07886 Increase default to 20 from 15
Clean up README a bit.
2018-04-27 10:23:20 +01:00
andy.boot 0f1f823736 Add long options to cmd line params 2018-04-27 10:15:30 +01:00
andy.boot e6c777fb8b Add support for -d depth flag
Following a user request the option '-d N' allows a user to output N
levels of sub directories.

Fixed bug: so that trailing slashes are now removed.
2018-04-27 10:01:41 +01:00
andy boot 0bded9698a Merge pull request #14 from bootandy/fix_mac
Fix tests on mac
2018-04-24 17:09:30 +01:00
bootandy c7f0ea59f0 Fix tests on mac
Macos does not appear to have a predictable iteration order of the files
2018-04-24 16:50:37 +01:00
andy boot 6d62cfb9ae Merge pull request #13 from bootandy/walkdir2
Rewrite to use walkdir instead of recursion
2018-04-24 15:41:51 +01:00
andy.boot 803934d84b By default only print last leaf of path
Fixes #9
https://github.com/bootandy/dust/issues/9

Instead of printing the all sub tree leaves only the last leaf is now
printed. See readme change for example

The flag '-p' was added to print the sub tree the old way
2018-04-24 14:54:11 +01:00
andy.boot 24c97ef92f Rewrite to use walkdir instead of recursion
Advised to use walkdir by burntsushi as using recursion on file systems
can blow the stack.

walkdir is slower but allows the code to be cleaner and more reliable

Also experimented with ignore but locking the hashmap resulted in
similar performance to walkdir but with much uglier code.
2018-04-24 14:53:47 +01:00
andy boot 270edf0a76 Update README.md 2018-04-18 13:49:05 +01:00
10 changed files with 416 additions and 303 deletions
+1
View File
@@ -5,3 +5,4 @@
# These are backup files generated by rustfmt # These are backup files generated by rustfmt
**/*.rs.bk **/*.rs.bk
*.swp *.swp
.vscode/*
Generated
+20
View File
@@ -127,6 +127,7 @@ dependencies = [
"assert_cli 0.5.4 (registry+https://github.com/rust-lang/crates.io-index)", "assert_cli 0.5.4 (registry+https://github.com/rust-lang/crates.io-index)",
"clap 2.31.1 (registry+https://github.com/rust-lang/crates.io-index)", "clap 2.31.1 (registry+https://github.com/rust-lang/crates.io-index)",
"tempfile 3.0.0 (registry+https://github.com/rust-lang/crates.io-index)", "tempfile 3.0.0 (registry+https://github.com/rust-lang/crates.io-index)",
"walkdir 2.1.4 (registry+https://github.com/rust-lang/crates.io-index)",
] ]
[[package]] [[package]]
@@ -259,6 +260,14 @@ dependencies = [
"winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)",
] ]
[[package]]
name = "same-file"
version = "1.0.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
dependencies = [
"winapi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)",
]
[[package]] [[package]]
name = "semver" name = "semver"
version = "0.8.0" version = "0.8.0"
@@ -403,6 +412,15 @@ dependencies = [
"winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)",
] ]
[[package]]
name = "walkdir"
version = "2.1.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
dependencies = [
"same-file 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
"winapi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)",
]
[[package]] [[package]]
name = "winapi" name = "winapi"
version = "0.2.8" version = "0.2.8"
@@ -467,6 +485,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
"checksum remove_dir_all 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)" = "dfc5b3ce5d5ea144bb04ebd093a9e14e9765bcfec866aecda9b6dec43b3d1e24" "checksum remove_dir_all 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)" = "dfc5b3ce5d5ea144bb04ebd093a9e14e9765bcfec866aecda9b6dec43b3d1e24"
"checksum rustc-demangle 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)" = "11fb43a206a04116ffd7cfcf9bcb941f8eb6cc7ff667272246b0a1c74259a3cb" "checksum rustc-demangle 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)" = "11fb43a206a04116ffd7cfcf9bcb941f8eb6cc7ff667272246b0a1c74259a3cb"
"checksum same-file 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)" = "d931a44fdaa43b8637009e7632a02adc4f2b2e0733c08caa4cf00e8da4a117a7" "checksum same-file 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)" = "d931a44fdaa43b8637009e7632a02adc4f2b2e0733c08caa4cf00e8da4a117a7"
"checksum same-file 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)" = "cfb6eded0b06a0b512c8ddbcf04089138c9b4362c2f696f3c3d76039d68f3637"
"checksum semver 0.8.0 (registry+https://github.com/rust-lang/crates.io-index)" = "bee2bc909ab2d8d60dab26e8cad85b25d795b14603a0dcb627b78b9d30b6454b" "checksum semver 0.8.0 (registry+https://github.com/rust-lang/crates.io-index)" = "bee2bc909ab2d8d60dab26e8cad85b25d795b14603a0dcb627b78b9d30b6454b"
"checksum semver-parser 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)" = "388a1df253eca08550bef6c72392cfe7c30914bf41df5269b68cbd6ff8f570a3" "checksum semver-parser 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)" = "388a1df253eca08550bef6c72392cfe7c30914bf41df5269b68cbd6ff8f570a3"
"checksum serde 1.0.37 (registry+https://github.com/rust-lang/crates.io-index)" = "d3bcee660dcde8f52c3765dd9ca5ee36b4bf35470a738eb0bd5a8752b0389645" "checksum serde 1.0.37 (registry+https://github.com/rust-lang/crates.io-index)" = "d3bcee660dcde8f52c3765dd9ca5ee36b4bf35470a738eb0bd5a8752b0389645"
@@ -484,6 +503,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
"checksum unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "fc72304796d0818e357ead4e000d19c9c174ab23dc11093ac919054d20a6a7fc" "checksum unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "fc72304796d0818e357ead4e000d19c9c174ab23dc11093ac919054d20a6a7fc"
"checksum vec_map 0.8.0 (registry+https://github.com/rust-lang/crates.io-index)" = "887b5b631c2ad01628bbbaa7dd4c869f80d3186688f8d0b6f58774fbe324988c" "checksum vec_map 0.8.0 (registry+https://github.com/rust-lang/crates.io-index)" = "887b5b631c2ad01628bbbaa7dd4c869f80d3186688f8d0b6f58774fbe324988c"
"checksum walkdir 1.0.7 (registry+https://github.com/rust-lang/crates.io-index)" = "bb08f9e670fab86099470b97cd2b252d6527f0b3cc1401acdb595ffc9dd288ff" "checksum walkdir 1.0.7 (registry+https://github.com/rust-lang/crates.io-index)" = "bb08f9e670fab86099470b97cd2b252d6527f0b3cc1401acdb595ffc9dd288ff"
"checksum walkdir 2.1.4 (registry+https://github.com/rust-lang/crates.io-index)" = "63636bd0eb3d00ccb8b9036381b526efac53caf112b7783b730ab3f8e44da369"
"checksum winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)" = "167dc9d6949a9b857f3451275e911c3f44255842c1f7a76f33c55103a909087a" "checksum winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)" = "167dc9d6949a9b857f3451275e911c3f44255842c1f7a76f33c55103a909087a"
"checksum winapi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)" = "04e3bd221fcbe8a271359c04f21a76db7d0c6028862d1bb5512d85e1e2eb5bb3" "checksum winapi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)" = "04e3bd221fcbe8a271359c04f21a76db7d0c6028862d1bb5512d85e1e2eb5bb3"
"checksum winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "2d315eee3b34aca4797b2da6b13ed88266e6d612562a0c46390af8299fc699bc" "checksum winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "2d315eee3b34aca4797b2da6b13ed88266e6d612562a0c46390af8299fc699bc"
+2 -1
View File
@@ -1,7 +1,7 @@
[package] [package]
name = "du-dust" name = "du-dust"
description = "A more intuitive version of du" description = "A more intuitive version of du"
version = "0.2.4" version = "0.3.0"
authors = ["bootandy <bootandy@gmail.com>", "nebkor <code@ardent.nebcorp.com>"] authors = ["bootandy <bootandy@gmail.com>", "nebkor <code@ardent.nebcorp.com>"]
documentation = "https://github.com/bootandy/dust" documentation = "https://github.com/bootandy/dust"
@@ -24,4 +24,5 @@ ansi_term = "0.11"
clap = "2.31" clap = "2.31"
assert_cli = "0.5" assert_cli = "0.5"
tempfile = "3" tempfile = "3"
walkdir = "2"
+41 -27
View File
@@ -1,58 +1,72 @@
[![Build Status](https://travis-ci.org/bootandy/dust.svg?branch=master)](https://travis-ci.org/bootandy/dust) [![Build Status](https://travis-ci.org/bootandy/dust.svg?branch=master)](https://travis-ci.org/bootandy/dust)
# Dust # Dust
du + rust = dust. Like du but more intuitive du + rust = dust. Like du but more intuitive
## Install ## Install
* Download linux / mac binary from [Releases](https://github.com/bootandy/dust/releases) ### Cargo
* unzip file: tar -xvf _downloaded_file.tar.gz_
* move file to executable path: sudo mv dust /usr/local/bin/ * cargo install du-dust
### Download
* Download linux / mac binary from [Releases](https://github.com/bootandy/dust/releases)
* unzip file: tar -xvf _downloaded_file.tar.gz_
* move file to executable path: sudo mv dust /usr/local/bin/
## Overview ## Overview
Dust is meant to give you an instant overview of which directories are using disk space without requiring sort or head. Dust will print a maximum of 1 'Did not have permissions message'. Dust is meant to give you an instant overview of which directories are using disk space without requiring sort or head. Dust will print a maximum of 1 'Did not have permissions message'.
Dust will list the 15 biggest sub directories or files and will smartly recurse down the tree to find the larger ones. There is no need for a '-d' flag or a '-h' flag. The largest sub directory will have its size shown in *red* Dust will list the 20 biggest sub directories or files and will smartly recurse down the tree to find the larger ones. There is no need for a '-d' flag or a '-h' flag. The largest sub directory will have its size shown in *red*
## Why? ## Why?
du has a number of ways of showing you what it finds, in terms of disk consumption, but really, there are only one or two ways you invoke it: with -h for “human readable” units, like 100G or 89k, or with -b for “bytes”. The former is generally used for a quick survey of a directory with a small number of things in it, and the latter for when you have a bunch and need to sort the output numerically, and youre obligated to either further pass it into something like awk to turn bytes into the appropriate human-friendly unit like mega or gigabytes, or you just do some rough math in your head and use the ordering to sanity check. Then once you have the top offenders, you recurse down into the largest one and repeat the process until youve found your cruft or gems and can move on.
du has a number of ways of showing you what it finds, in terms of disk consumption, but really, there are only one or two ways you invoke it: with -h for “human readable” units, like 100G or 89k, or with -b for “bytes”. The former is generally used for a quick survey of a directory with a small number of things in it, and the latter for when you have a bunch and need to sort the output numerically, and youre obligated to either further pass it into something like awk to turn bytes into the appropriate human-friendly unit like mega or gigabytes, or pipe thru sort and head while remembering the '-h' flag. Then once you have the top offenders, you recurse down into the largest one and repeat the process until youve found your cruft or gems and can move on.
Dust assumes thats what you wanted to do in the first place, and takes care of tracking the largest offenders in terms of actual size, and showing them to you with human-friendly units and in-context within the filetree. Dust assumes thats what you wanted to do in the first place, and takes care of tracking the largest offenders in terms of actual size, and showing them to you with human-friendly units and in-context within the filetree.
## Usage ## Usage
``` ```
Usage: dust <dir> Usage: dust <dir>
Usage: dust <dir> <another_dir> <and_more> Usage: dust <dir> <another_dir> <and_more>
Usage: dust -p <dir> (full-path - does not shorten the path of the subdirectories)
Usage: dust -s <dir> (apparent-size - shows the length of the file as opposed to the amount of disk space it uses) Usage: dust -s <dir> (apparent-size - shows the length of the file as opposed to the amount of disk space it uses)
Usage: dust -n 30 <dir> (Shows 30 directories not 15) Usage: dust -n 30 <dir> (Shows 30 directories not 20)
Usage: dust -d 3 <dir> (Shows 3 levels of subdirectories)
``` ```
``` ```
djin:git/dust> dust djin:git/dust> dust
65M . 1.2G target
65M ─┬ ./target 622M ─┬ debug
49M ├─┬ ./target/debug 445M ├── deps
26M │ ├─┬ ./target/debug/deps 70M │ ├── incremental
21M │ │ └── ./target/debug/deps/libclap-9e6625ac8ff074ad.rlib 56M │ └── build
13M │ ├── ./target/debug/dust 262M ├─┬ rls
8.9M │ └─┬ ./target/debug/incremental 262M │ └─┬ debug
6.7M │ ├─┬ ./target/debug/incremental/dust-2748eiei2tcnp 203M │ ├── deps
6.7M │ │ └─┬ ./target/debug/incremental/dust-2748eiei2tcnp/s-ezd6jnik5u-163pyem-1aab9ncf5glum 56M │ └── build
3.0M │ │ └── ./target/debug/incremental/dust-2748eiei2tcnp/s-ezd6jnik5u-163pyem-1aab9ncf5glum/dep-graph.bin 165M ├─┬ package
2.2M │ └─┬ ./target/debug/incremental/dust-1dlon65p8m3vl 165M │ └─┬ du-dust-0.2.4
2.2M │ └── ./target/debug/incremental/dust-1dlon65p8m3vl/s-ezd6jncecv-1xsnfd0-4dw9l1r2th2t 165M │ └─target
15M └─┬ ./target/release 165M └─┬ debug
9.2M ├─┬ ./target/release/deps 131M └── deps
6.7M │ └── ./target/release/deps/libclap-87bc2534ea57f044.rlib 165M └─┬ release
5.9M └── ./target/release/dust 124M └── deps
``` ```
## Performance ## Performance
dust is currently about 4 times slower than du.
Dust is currently about 4 times slower than du.
## Alternatives ## Alternatives
* [NCDU](https://dev.yorhel.nl/ncdu)
* du -d 1 -h | sort -h * [NCDU](https://dev.yorhel.nl/ncdu)
* du -d 1 -h | sort -h
Note: Apparent-size is calculated slightly differently in dust to gdu. In dust each hard link is counted as using file_length space. In gdu only the first entry is counted. Note: Apparent-size is calculated slightly differently in dust to gdu. In dust each hard link is counted as using file_length space. In gdu only the first entry is counted.
+96 -35
View File
@@ -2,42 +2,63 @@ extern crate ansi_term;
use self::ansi_term::Colour::Fixed; use self::ansi_term::Colour::Fixed;
use lib::Node;
static UNITS: [char; 4] = ['T', 'G', 'M', 'K']; static UNITS: [char; 4] = ['T', 'G', 'M', 'K'];
pub fn draw_it(permissions: bool, heads: &Vec<Node>, to_display: &Vec<&Node>) -> () { pub fn draw_it(
permissions: bool,
short_paths: bool,
depth: Option<u64>,
base_dirs: Vec<String>,
to_display: Vec<(String, u64)>,
) -> () {
if !permissions { if !permissions {
eprintln!("Did not have permissions for all directories"); eprintln!("Did not have permissions for all directories");
} }
for d in to_display { for f in base_dirs {
if heads.contains(d) { display_node(f, &to_display, true, short_paths, depth, "")
display_node(d, &to_display, true, "")
}
} }
} }
fn get_size(nodes: &Vec<(String, u64)>, node_to_print: &String) -> Option<u64> {
for &(ref k, ref v) in nodes.iter() {
if *k == *node_to_print {
return Some(*v);
}
}
None
}
fn display_node<S: Into<String>>( fn display_node<S: Into<String>>(
node_to_print: &Node, node_to_print: String,
to_display: &Vec<&Node>, to_display: &Vec<(String, u64)>,
is_first: bool, is_biggest: bool,
short_paths: bool,
depth: Option<u64>,
indentation_str: S, indentation_str: S,
) { ) {
let new_depth = match depth {
None => None,
Some(0) => return,
Some(d) => Some(d - 1),
};
match get_size(to_display, &node_to_print) {
None => println!("Can not find path: {}", node_to_print),
Some(size) => {
let mut is = indentation_str.into(); let mut is = indentation_str.into();
print_this_node(node_to_print, is_first, is.as_ref()); let ntp: &str = node_to_print.as_ref();
print_this_node(ntp, size, is_biggest, short_paths, is.as_ref());
is = is.replace("└─┬", " "); is = is.replace("└─┬", " ");
is = is.replace("└──", " "); is = is.replace("└──", " ");
is = is.replace("├──", ""); is = is.replace("├──", "");
is = is.replace("├─┬", ""); is = is.replace("├─┬", "");
let printable_node_slashes = node_to_print.name().matches('/').count(); let printable_node_slashes = node_to_print.matches('/').count();
let mut num_siblings = to_display.iter().fold(0, |a, b| { let mut num_siblings = to_display.iter().fold(0, |a, b| {
if node_to_print.children().contains(b) if b.0.starts_with(ntp) && b.0.matches('/').count() == printable_node_slashes + 1 {
&& b.name().matches('/').count() == printable_node_slashes + 1
{
a + 1 a + 1
} else { } else {
a a
@@ -45,17 +66,38 @@ fn display_node<S: Into<String>>(
}); });
let mut is_biggest = true; let mut is_biggest = true;
for node in to_display { for &(ref k, _) in to_display.iter() {
if node_to_print.children().contains(node) { if k.starts_with(ntp) && k.matches('/').count() == printable_node_slashes + 1 {
let has_display_children = node.children()
.iter()
.fold(false, |has_kids, n| has_kids || to_display.contains(&n));
let has_children = node.children().len() > 0 && has_display_children;
if node.name().matches('/').count() == printable_node_slashes + 1 {
num_siblings -= 1; num_siblings -= 1;
let tree_chars = { let mut has_children = false;
if new_depth.is_none() || new_depth.unwrap() != 1 {
for &(ref k2, _) in to_display.iter() {
let kk: &str = k.as_ref();
if k2.starts_with(kk)
&& k2.matches('/').count() == printable_node_slashes + 2
{
has_children = true;
}
}
}
display_node(
k.to_string(),
to_display,
is_biggest,
short_paths,
new_depth,
is.to_string() + get_tree_chars(num_siblings, has_children),
);
is_biggest = false;
}
}
}
}
}
fn get_tree_chars(num_siblings: u64, has_children: bool) -> &'static str {
if num_siblings == 0 { if num_siblings == 0 {
if has_children { if has_children {
"└─┬" "└─┬"
@@ -69,23 +111,42 @@ fn display_node<S: Into<String>>(
"├──" "├──"
} }
} }
};
display_node(&node, to_display, is_biggest, is.to_string() + tree_chars);
is_biggest = false;
}
}
}
} }
fn print_this_node(node: &Node, is_biggest: bool, indentation: &str) { fn print_this_node(
let pretty_size = format!("{:>5}", human_readable_number(node.size()),); node_name: &str,
size: u64,
is_biggest: bool,
short_paths: bool,
indentation: &str,
) {
let pretty_size = format!("{:>5}", human_readable_number(size),);
println!( println!(
"{}", "{}",
format_string(node.name(), is_biggest, pretty_size.as_ref(), indentation) format_string(
node_name,
is_biggest,
short_paths,
pretty_size.as_ref(),
indentation
)
) )
} }
pub fn format_string(dir_name: &str, is_biggest: bool, size: &str, indentation: &str) -> String { pub fn format_string(
dir_name: &str,
is_biggest: bool,
short_paths: bool,
size: &str,
indentation: &str,
) -> String {
let printable_name = {
if short_paths && dir_name.contains('/') {
dir_name.split('/').last().unwrap()
} else {
dir_name
}
};
format!( format!(
"{} {} {}", "{} {} {}",
if is_biggest { if is_biggest {
@@ -94,7 +155,7 @@ pub fn format_string(dir_name: &str, is_biggest: bool, size: &str, indentation:
Fixed(7).paint(size) Fixed(7).paint(size)
}, },
indentation, indentation,
dir_name, printable_name,
) )
} }
-68
View File
@@ -1,68 +0,0 @@
use std::cmp::{Eq, Ord, Ordering, PartialEq, PartialOrd};
#[derive(Clone, Debug)]
pub struct Node {
name: String,
size: u64,
children: Vec<Node>,
}
impl Node {
pub fn new<S: Into<String>>(name: S, size: u64, children: Vec<Node>) -> Self {
Node {
children: children,
name: name.into(),
size: size,
}
}
pub fn children(&self) -> &Vec<Node> {
&self.children
}
pub fn name(&self) -> &String {
&self.name
}
pub fn size(&self) -> u64 {
self.size
}
}
impl Ord for Node {
fn cmp(&self, other: &Self) -> Ordering {
if self.size > other.size {
Ordering::Less
} else if self.size < other.size {
Ordering::Greater
} else {
let my_slashes = self.name.matches('/').count();
let other_slashes = other.name.matches('/').count();
if my_slashes > other_slashes {
Ordering::Greater
} else if my_slashes < other_slashes {
Ordering::Less
} else {
if self.name < other.name {
Ordering::Less
} else if self.name > other.name {
Ordering::Greater
} else {
Ordering::Equal
}
}
}
}
}
impl PartialOrd for Node {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
Some(self.cmp(other))
}
}
impl PartialEq for Node {
fn eq(&self, other: &Self) -> bool {
(&self.name, self.size) == (&other.name, other.size)
}
}
impl Eq for Node {}
+59 -10
View File
@@ -1,31 +1,46 @@
#[macro_use] #[macro_use]
extern crate clap; extern crate clap;
extern crate assert_cli; extern crate assert_cli;
extern crate walkdir;
use self::display::draw_it; use self::display::draw_it;
use clap::{App, AppSettings, Arg}; use clap::{App, AppSettings, Arg};
use utils::{find_big_ones, get_dir_tree}; use std::io::{self, Write};
use utils::{find_big_ones, get_dir_tree, sort};
mod display; mod display;
mod utils; mod utils;
mod lib;
static DEFAULT_NUMBER_OF_LINES: &'static str = "15"; static DEFAULT_NUMBER_OF_LINES: &'static str = "20";
fn main() { fn main() {
let options = App::new("Dust") let options = App::new("Dust")
.setting(AppSettings::TrailingVarArg) .setting(AppSettings::TrailingVarArg)
.arg(
Arg::with_name("depth")
.short("d")
.long("depth")
.help("Depth to show")
.takes_value(true),
)
.arg( .arg(
Arg::with_name("number_of_lines") Arg::with_name("number_of_lines")
.short("n") .short("n")
.long("number-of-lines")
.help("Number of lines of output to show") .help("Number of lines of output to show")
.takes_value(true) .takes_value(true)
.default_value(DEFAULT_NUMBER_OF_LINES), .default_value(DEFAULT_NUMBER_OF_LINES),
) )
.arg( .arg(
Arg::with_name("use_apparent_size") Arg::with_name("display_full_paths")
.short("p")
.long("full-paths")
.help("If set sub directories will not have their path shortened"),
)
.arg(
Arg::with_name("display_apparent_size")
.short("s") .short("s")
.long("apparent-size")
.help("If set will use file length. Otherwise we use blocks"), .help("If set will use file length. Otherwise we use blocks"),
) )
.arg(Arg::with_name("inputs").multiple(true)) .arg(Arg::with_name("inputs").multiple(true))
@@ -37,12 +52,46 @@ fn main() {
Some(r) => r.collect(), Some(r) => r.collect(),
} }
}; };
let number_of_lines = value_t!(options.value_of("number_of_lines"), usize).unwrap();
let use_apparent_size = options.is_present("use_apparent_size");
let (permissions, node_per_top_level_dir) = get_dir_tree(&filenames, use_apparent_size); let number_of_lines = value_t!(options.value_of("number_of_lines"), usize).unwrap();
let slice_it = find_big_ones(&node_per_top_level_dir, number_of_lines); let depth = {
draw_it(permissions, &node_per_top_level_dir, &slice_it); if options.is_present("depth") {
match value_t!(options.value_of("depth"), u64) {
Ok(v) => Some(v + 1),
Err(_) => None,
}
} else {
None
}
};
if options.is_present("depth")
&& options.value_of("number_of_lines").unwrap() != DEFAULT_NUMBER_OF_LINES
{
io::stderr()
.write(b"Use either -n for number of directories to show. Or -d for depth. Not both")
.expect("Error writing to stderr. Oh the irony!");
return;
}
let use_apparent_size = options.is_present("display_apparent_size");
let use_full_path = options.is_present("display_full_paths");
let (permissions, nodes, top_level_names) = get_dir_tree(&filenames, use_apparent_size);
let sorted_data = sort(nodes);
let biggest_ones = {
if depth.is_none() {
find_big_ones(sorted_data, number_of_lines)
} else {
sorted_data
}
};
draw_it(
permissions,
!use_full_path,
depth,
top_level_names,
biggest_ones,
);
} }
#[cfg(test)] #[cfg(test)]
+104 -44
View File
@@ -15,52 +15,82 @@ pub fn test_main() {
assert_cli::Assert::main_binary() assert_cli::Assert::main_binary()
.with_args(&["src/test_dir"]) .with_args(&["src/test_dir"])
.stdout() .stdout()
.is(main_output()) .is(main_output(true))
.unwrap();
}
#[test]
pub fn test_main_long_paths() {
assert_cli::Assert::main_binary()
.with_args(&["-p", "src/test_dir"])
.stdout()
.is(main_output(false))
.unwrap(); .unwrap();
} }
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]
fn main_output() -> String { fn main_output(short_paths: bool) -> String {
format!( format!(
"{} "{}
{} {}
{} {}
{}", {}",
format_string("src/test_dir", true, " 4.0K", ""), format_string("src/test_dir", true, short_paths, " 4.0K", ""),
format_string("src/test_dir/many", true, " 4.0K", "└─┬",), format_string("src/test_dir/many", true, short_paths, " 4.0K", "└─┬",),
format_string("src/test_dir/many/hello_file", true, " 4.0K", " ├──",), format_string(
format_string("src/test_dir/many/a_file", false, " 0B", " └──",), "src/test_dir/many/hello_file",
true,
short_paths,
" 4.0K",
" ├──",
),
format_string(
"src/test_dir/many/a_file",
false,
short_paths,
" 0B",
" └──",
),
) )
} }
#[cfg(target_os = "linux")] #[cfg(target_os = "linux")]
fn main_output() -> String { fn main_output(short_paths: bool) -> String {
format!( format!(
"{} "{}
{} {}
{} {}
{}", {}",
format_string("src/test_dir", true, " 8.0K", ""), format_string("src/test_dir", true, short_paths, " 12K", ""),
format_string("src/test_dir/many", true, " 4.0K", "└─┬",), format_string("src/test_dir/many", true, short_paths, " 8.0K", "└─┬",),
format_string("src/test_dir/many/hello_file", true, " 4.0K", " ├──",), format_string(
format_string("src/test_dir/many/a_file", false, " 0B", " └──",), "src/test_dir/many/hello_file",
true,
short_paths,
" 4.0K",
" ├──",
),
format_string(
"src/test_dir/many/a_file",
false,
short_paths,
" 0B",
" └──",
),
) )
} }
#[test]
pub fn test_main_extra_slash() {
assert_cli::Assert::main_binary()
.with_args(&["src/test_dir/"])
.stdout()
.is(main_output())
.unwrap();
}
#[test] #[test]
pub fn test_apparent_size() { pub fn test_apparent_size() {
let r = format!( let r = format!(
"{}", "{}",
format_string("src/test_dir/many/hello_file", true, " 6B", " ├──",), format_string(
"src/test_dir/many/hello_file",
true,
true,
" 6B",
" ├──",
),
); );
assert_cli::Assert::main_binary() assert_cli::Assert::main_binary()
@@ -93,10 +123,14 @@ pub fn test_soft_sym_link() {
.output(); .output();
assert!(c.is_ok()); assert!(c.is_ok());
let r = soft_sym_link_output(dir_s, file_path_s, link_name_s);
// We cannot guarantee which version will appear first.
// TODO: Consider adding predictable itteration order (sort file entries by name?)
assert_cli::Assert::main_binary() assert_cli::Assert::main_binary()
.with_args(&[dir_s]) .with_args(&[dir_s])
.stdout() .stdout()
.contains(soft_sym_link_output(dir_s, file_path_s, link_name_s)) .contains(r)
.unwrap(); .unwrap();
} }
@@ -106,20 +140,21 @@ fn soft_sym_link_output(dir: &str, file_path: &str, link_name: &str) -> String {
"{} "{}
{} {}
{}", {}",
format_string(dir, true, " 8.0K", ""), format_string(dir, true, true, " 8.0K", ""),
format_string(file_path, true, " 4.0K", "├──",), format_string(file_path, true, true, " 4.0K", "├──",),
format_string(link_name, false, " 4.0K", "└──",), format_string(link_name, false, true, " 4.0K", "└──",),
) )
} }
#[cfg(target_os = "linux")] #[cfg(target_os = "linux")]
fn soft_sym_link_output(dir: &str, file_path: &str, link_name: &str) -> String { fn soft_sym_link_output(dir: &str, file_path: &str, link_name: &str) -> String {
format!( format!(
"{} "{}
{} {}
{}", {}",
format_string(dir, true, " 4.0K", ""), format_string(dir, true, true, " 8.0K", ""),
format_string(file_path, true, " 4.0K", "├──",), format_string(file_path, true, true, " 4.0K", "├──",),
format_string(link_name, false, " 0B", "└──",), format_string(link_name, false, true, " 0B", "└──",),
) )
} }
@@ -139,18 +174,7 @@ pub fn test_hard_sym_link() {
.output(); .output();
assert!(c.is_ok()); assert!(c.is_ok());
let r = format!( let (r, r2) = hard_link_output(dir_s, file_path_s, link_name_s);
"{}
{}",
format_string(dir_s, true, " 4.0K", ""),
format_string(file_path_s, true, " 4.0K", "└──")
);
let r2 = format!(
"{}
{}",
format_string(dir_s, true, " 4.0K", ""),
format_string(link_name_s, true, " 4.0K", "└──")
);
// Because this is a hard link the file and hard link look identicle. Therefore // Because this is a hard link the file and hard link look identicle. Therefore
// we cannot guarantee which version will appear first. // we cannot guarantee which version will appear first.
@@ -171,6 +195,40 @@ pub fn test_hard_sym_link() {
} }
} }
#[cfg(target_os = "macos")]
fn hard_link_output(dir_s: &str, file_path_s: &str, link_name_s: &str) -> (String, String) {
let r = format!(
"{}
{}",
format_string(dir_s, true, true, " 4.0K", ""),
format_string(file_path_s, true, true, " 4.0K", "└──")
);
let r2 = format!(
"{}
{}",
format_string(dir_s, true, true, " 4.0K", ""),
format_string(link_name_s, true, true, " 4.0K", "└──")
);
(r, r2)
}
#[cfg(target_os = "linux")]
fn hard_link_output(dir_s: &str, file_path_s: &str, link_name_s: &str) -> (String, String) {
let r = format!(
"{}
{}",
format_string(dir_s, true, true, " 8.0K", ""),
format_string(file_path_s, true, true, " 4.0K", "└──")
);
let r2 = format!(
"{}
{}",
format_string(dir_s, true, true, " 8.0K", ""),
format_string(link_name_s, true, true, " 4.0K", "└──")
);
(r, r2)
}
//Check we don't recurse down an infinite symlink tree //Check we don't recurse down an infinite symlink tree
#[test] #[test]
pub fn test_recursive_sym_link() { pub fn test_recursive_sym_link() {
@@ -199,8 +257,8 @@ fn recursive_sym_link_output(dir: &str, link_name: &str) -> String {
format!( format!(
"{} "{}
{}", {}",
format_string(dir, true, " 4.0K", ""), format_string(dir, true, true, " 4.0K", ""),
format_string(link_name, true, " 4.0K", "└──",), format_string(link_name, true, true, " 4.0K", "└──",),
) )
} }
#[cfg(target_os = "linux")] #[cfg(target_os = "linux")]
@@ -208,7 +266,9 @@ fn recursive_sym_link_output(dir: &str, link_name: &str) -> String {
format!( format!(
"{} "{}
{}", {}",
format_string(dir, true, " 0B", ""), format_string(dir, true, true, " 4.0K", ""),
format_string(link_name, true, " 0B", "└──",), format_string(link_name, true, true, " 0B", "└──",),
) )
} }
// TODO: add test for bad path
+61 -77
View File
@@ -1,27 +1,36 @@
use std::collections::HashMap;
use std::collections::HashSet; use std::collections::HashSet;
use std::cmp::Ordering;
use std::fs; use walkdir::WalkDir;
use std::path::Path; use std::path::Path;
use std::path::PathBuf;
use lib::Node;
mod platform; mod platform;
use self::platform::*; use self::platform::*;
pub fn get_dir_tree(filenames: &Vec<&str>, apparent_size: bool) -> (bool, Vec<Node>) { pub fn get_dir_tree(
let mut permissions = true; filenames: &Vec<&str>,
apparent_size: bool,
) -> (bool, HashMap<String, u64>, Vec<String>) {
let mut permissions = 0;
let mut inodes: HashSet<(u64, u64)> = HashSet::new(); let mut inodes: HashSet<(u64, u64)> = HashSet::new();
let mut results = vec![]; let mut data: HashMap<String, u64> = HashMap::new();
for &b in filenames { let mut top_level_names = Vec::new();
let filename = strip_end_slashes(b);
let (hp, data) = examine_dir(&Path::new(&filename), apparent_size, &mut inodes); for b in filenames {
permissions = permissions && hp; let top_level_name = strip_end_slashes(b);
match data { examine_dir(
Some(d) => results.push(d), &Path::new(&top_level_name).to_path_buf(),
None => permissions = false, apparent_size,
&mut inodes,
&mut data,
&mut permissions,
);
top_level_names.push(top_level_name);
} }
} (permissions == 0, data, top_level_names)
(permissions, results)
} }
fn strip_end_slashes(s: &str) -> String { fn strip_end_slashes(s: &str) -> String {
@@ -33,24 +42,19 @@ fn strip_end_slashes(s: &str) -> String {
} }
fn examine_dir( fn examine_dir(
sdir: &Path, top_dir: &PathBuf,
apparent_size: bool, apparent_size: bool,
inodes: &mut HashSet<(u64, u64)>, inodes: &mut HashSet<(u64, u64)>,
) -> (bool, Option<Node>) { data: &mut HashMap<String, u64>,
match fs::read_dir(sdir) { permissions: &mut u64,
Ok(file_iter) => { ) {
let mut result = vec![]; for entry in WalkDir::new(top_dir) {
let mut have_permission = true; match entry {
let mut total_size = 0; Ok(e) => {
let maybe_size_and_inode = get_metadata(&e, apparent_size);
for single_path in file_iter { match maybe_size_and_inode {
match single_path { Some((size, maybe_inode)) => {
Ok(d) => {
let file_type = d.file_type().ok();
let maybe_size_and_inode = get_metadata(&d, apparent_size);
match (file_type, maybe_size_and_inode) {
(Some(file_type), Some((size, maybe_inode))) => {
if !apparent_size { if !apparent_size {
if let Some(inode_dev_pair) = maybe_inode { if let Some(inode_dev_pair) = maybe_inode {
if inodes.contains(&inode_dev_pair) { if inodes.contains(&inode_dev_pair) {
@@ -59,61 +63,41 @@ fn examine_dir(
inodes.insert(inode_dev_pair); inodes.insert(inode_dev_pair);
} }
} }
total_size += size; let mut e_path = e.path().to_path_buf();
loop {
if d.path().is_dir() && !file_type.is_symlink() { let path_name = e_path.to_string_lossy().to_string();
let (hp, child) = examine_dir(&d.path(), apparent_size, inodes); let s = data.entry(path_name).or_insert(0);
have_permission = have_permission && hp; *s += size;
if e_path == *top_dir {
match child { break;
Some(c) => {
total_size += c.size();
result.push(c);
} }
None => (), e_path.pop();
} }
}
None => *permissions += 1,
}
}
_ => {}
}
}
}
pub fn compare_tuple(a :&(String, u64), b: &(String, u64)) -> Ordering {
let result = b.1.cmp(&a.1);
if result == Ordering::Equal {
a.0.cmp(&b.0)
} else { } else {
let path_name = d.path().to_string_lossy().to_string(); result
result.push(Node::new(path_name, size, vec![]))
}
}
(_, None) => have_permission = false,
(_, _) => (),
}
}
Err(_) => (),
}
}
let n = Node::new(sdir.to_string_lossy().to_string(), total_size, result);
(have_permission, Some(n))
}
Err(_) => (false, None),
} }
} }
// We start with a list of root directories - these must be the biggest folders pub fn sort<'a>(data: HashMap<String, u64>) -> Vec<(String, u64)> {
// We then repeadedly merge in the children of the biggest directory - Each iteration let mut new_l: Vec<(String, u64)> = data.iter().map(|(a, b)| (a.clone(), *b)).collect();
// the next biggest directory's children are merged in. new_l.sort_by(|a, b| compare_tuple(&a, &b));
pub fn find_big_ones<'a>(l: &'a Vec<Node>, max_to_show: usize) -> Vec<&Node> { new_l
let mut new_l: Vec<&Node> = l.iter().map(|a| a).collect(); }
new_l.sort();
for processed_pointer in 0..max_to_show { pub fn find_big_ones<'a>(new_l: Vec<(String, u64)>, max_to_show: usize) -> Vec<(String, u64)> {
if new_l.len() == processed_pointer { if max_to_show > 0 && new_l.len() > max_to_show {
break;
}
// Must be a list of pointers into new_l otherwise b_list will go out of scope
// when it is deallocated
let mut b_list: Vec<&Node> = new_l[processed_pointer]
.children()
.iter()
.map(|a| a)
.collect();
new_l.extend(b_list);
new_l.sort();
}
if new_l.len() > max_to_show {
new_l[0..max_to_show + 1].to_vec() new_l[0..max_to_show + 1].to_vec()
} else { } else {
new_l new_l
+5 -14
View File
@@ -1,4 +1,4 @@
use std; use walkdir::DirEntry;
fn get_block_size() -> u64 { fn get_block_size() -> u64 {
// All os specific implementations of MetatdataExt seem to define a block as 512 bytes // All os specific implementations of MetatdataExt seem to define a block as 512 bytes
@@ -7,10 +7,7 @@ fn get_block_size() -> u64 {
} }
#[cfg(target_os = "linux")] #[cfg(target_os = "linux")]
pub fn get_metadata( pub fn get_metadata(d: &DirEntry, use_apparent_size: bool) -> Option<(u64, Option<(u64, u64)>)> {
d: &std::fs::DirEntry,
use_apparent_size: bool,
) -> Option<(u64, Option<(u64, u64)>)> {
use std::os::linux::fs::MetadataExt; use std::os::linux::fs::MetadataExt;
match d.metadata().ok() { match d.metadata().ok() {
Some(md) => { Some(md) => {
@@ -26,10 +23,7 @@ pub fn get_metadata(
} }
#[cfg(target_os = "unix")] #[cfg(target_os = "unix")]
pub fn get_metadata( pub fn get_metadata(d: &DirEntry, use_apparent_size: bool) -> Option<(u64, Option<(u64, u64)>)> {
d: &std::fs::DirEntry,
use_apparent_size: bool,
) -> Option<(u64, Option<(u64, u64)>)> {
use std::os::unix::fs::MetadataExt; use std::os::unix::fs::MetadataExt;
match d.metadata().ok() { match d.metadata().ok() {
Some(md) => { Some(md) => {
@@ -45,10 +39,7 @@ pub fn get_metadata(
} }
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]
pub fn get_metadata( pub fn get_metadata(d: &DirEntry, use_apparent_size: bool) -> Option<(u64, Option<(u64, u64)>)> {
d: &std::fs::DirEntry,
use_apparent_size: bool,
) -> Option<(u64, Option<(u64, u64)>)> {
use std::os::macos::fs::MetadataExt; use std::os::macos::fs::MetadataExt;
match d.metadata().ok() { match d.metadata().ok() {
Some(md) => { Some(md) => {
@@ -64,7 +55,7 @@ pub fn get_metadata(
} }
#[cfg(not(any(target_os = "linux", target_os = "unix", target_os = "macos")))] #[cfg(not(any(target_os = "linux", target_os = "unix", target_os = "macos")))]
pub fn get_metadata(d: &std::fs::DirEntry, _apparent: bool) -> Option<(u64, Option<(u64, u64)>)> { pub fn get_metadata(d: &DirEntry, _apparent: bool) -> Option<(u64, Option<(u64, u64)>)> {
match d.metadata().ok() { match d.metadata().ok() {
Some(md) => Some((md.len(), None)), Some(md) => Some((md.len(), None)),
None => None, None => None,