add --du to ls command (#917)

This commit is contained in:
Darren Schroeder 2022-02-03 13:58:32 -06:00 committed by GitHub
parent 2f0bbf5adb
commit e1c28cf06b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -1,6 +1,6 @@
use crate::DirBuilder;
use crate::DirInfo;
use chrono::{DateTime, Utc}; use chrono::{DateTime, Utc};
use pathdiff::diff_paths;
use nu_engine::env::current_dir; use nu_engine::env::current_dir;
use nu_engine::CallExt; use nu_engine::CallExt;
use nu_protocol::ast::Call; use nu_protocol::ast::Call;
@ -9,10 +9,11 @@ use nu_protocol::{
Category, DataSource, IntoInterruptiblePipelineData, PipelineData, PipelineMetadata, Category, DataSource, IntoInterruptiblePipelineData, PipelineData, PipelineMetadata,
ShellError, Signature, Span, Spanned, SyntaxShape, Value, ShellError, Signature, Span, Spanned, SyntaxShape, Value,
}; };
use pathdiff::diff_paths;
#[cfg(unix)] #[cfg(unix)]
use std::os::unix::fs::PermissionsExt; use std::os::unix::fs::PermissionsExt;
use std::path::PathBuf; use std::path::PathBuf;
use std::sync::Arc;
#[derive(Clone)] #[derive(Clone)]
pub struct Ls; pub struct Ls;
@ -45,11 +46,11 @@ impl Command for Ls {
Some('s'), Some('s'),
) )
.switch("full-paths", "display paths as absolute paths", Some('f')) .switch("full-paths", "display paths as absolute paths", Some('f'))
// .switch( .switch(
// "du", "du",
// "Display the apparent directory size in place of the directory metadata size", "Display the apparent directory size in place of the directory metadata size",
// Some('d'), Some('d'),
// ) )
.category(Category::FileSystem) .category(Category::FileSystem)
} }
@ -64,6 +65,8 @@ impl Command for Ls {
let long = call.has_flag("long"); let long = call.has_flag("long");
let short_names = call.has_flag("short-names"); let short_names = call.has_flag("short-names");
let full_paths = call.has_flag("full-paths"); let full_paths = call.has_flag("full-paths");
let du = call.has_flag("du");
let ctrl_c = engine_state.ctrlc.clone();
let call_span = call.head; let call_span = call.head;
let cwd = current_dir(engine_state, stack)?; let cwd = current_dir(engine_state, stack)?;
@ -132,8 +135,15 @@ impl Command for Ls {
match display_name { match display_name {
Ok(name) => { Ok(name) => {
let entry = let entry = dir_entry_dict(
dir_entry_dict(&path, &name, metadata.as_ref(), call_span, long); &path,
&name,
metadata.as_ref(),
call_span,
long,
du,
ctrl_c.clone(),
);
match entry { match entry {
Ok(value) => Some(value), Ok(value) => Some(value),
Err(err) => Some(Value::Error { error: err }), Err(err) => Some(Value::Error { error: err }),
@ -190,6 +200,7 @@ fn path_contains_hidden_folder(path: &Path, folders: &[PathBuf]) -> bool {
#[cfg(unix)] #[cfg(unix)]
use std::os::unix::fs::FileTypeExt; use std::os::unix::fs::FileTypeExt;
use std::path::Path; use std::path::Path;
use std::sync::atomic::AtomicBool;
pub fn get_file_type(md: &std::fs::Metadata) -> &str { pub fn get_file_type(md: &std::fs::Metadata) -> &str {
let ft = md.file_type(); let ft = md.file_type();
@ -224,6 +235,8 @@ pub(crate) fn dir_entry_dict(
metadata: Option<&std::fs::Metadata>, metadata: Option<&std::fs::Metadata>,
span: Span, span: Span,
long: bool, long: bool,
du: bool,
ctrl_c: Option<Arc<AtomicBool>>,
) -> Result<Value, ShellError> { ) -> Result<Value, ShellError> {
let mut cols = vec![]; let mut cols = vec![];
let mut vals = vec![]; let mut vals = vec![];
@ -324,12 +337,22 @@ pub(crate) fn dir_entry_dict(
cols.push("size".to_string()); cols.push("size".to_string());
if let Some(md) = metadata { if let Some(md) = metadata {
if md.is_dir() { if md.is_dir() {
if du {
let params = DirBuilder::new(Span { start: 0, end: 2 }, None, false, None, false);
let dir_size = DirInfo::new(filename, &params, None, ctrl_c).get_size();
vals.push(Value::Filesize {
val: dir_size as i64,
span,
});
} else {
let dir_size: u64 = md.len(); let dir_size: u64 = md.len();
vals.push(Value::Filesize { vals.push(Value::Filesize {
val: dir_size as i64, val: dir_size as i64,
span, span,
}); });
};
} else if md.is_file() { } else if md.is_file() {
vals.push(Value::Filesize { vals.push(Value::Filesize {
val: md.len() as i64, val: md.len() as i64,