Print xattrs in tree view like we do errors

This changes the way extended attributes (xattrs) are printed. Before, they were artificially printed out on their own line both in lines mode *and* details mode, which looked a bit weird. Now, they are additional 'child nodes' of that item that get printed alongside errors.

All this allows all the 'extra info' that is going to be present for very few entries to be consolidated and listed in the same way, without resorting to extra printlns.

As a great side-effect, it allows taking out some of the more redundant code in the Table impl -- it is now *always* going to be in create-child-nodes mode, as *any* file now can, not only when we have the --tree flag in use.

Also, it now actually displays errors when failing to read the extended attributes, such as if the user doesn't have permission to read them.

The extended attribute flag has been temporarily disabled while I work out the best way to do it!
This commit is contained in:
Ben S 2015-08-25 21:08:25 +01:00
parent 17c493b370
commit 69b22a0d66
4 changed files with 51 additions and 62 deletions

View File

@ -39,23 +39,29 @@ pub struct Attribute {
pub fn list_attrs(lister: lister::Lister, path: &Path) -> io::Result<Vec<Attribute>> { pub fn list_attrs(lister: lister::Lister, path: &Path) -> io::Result<Vec<Attribute>> {
let c_path = match path.as_os_str().to_cstring() { let c_path = match path.as_os_str().to_cstring() {
Some(cstring) => cstring, Some(cstring) => cstring,
None => return Err(io::Error::new(io::ErrorKind::Other, "could not read extended attributes")), None => return Err(io::Error::new(io::ErrorKind::Other, "Error: path somehow contained a NUL?")),
}; };
let mut names = Vec::new();
let bufsize = lister.listxattr_first(&c_path); let bufsize = lister.listxattr_first(&c_path);
if bufsize > 0 { if bufsize < 0 {
return Err(io::Error::last_os_error());
}
else if bufsize > 0 {
let mut buf = vec![0u8; bufsize as usize]; let mut buf = vec![0u8; bufsize as usize];
let err = lister.listxattr_second(&c_path, &mut buf, bufsize); let err = lister.listxattr_second(&c_path, &mut buf, bufsize);
if err < 0 {
return Err(io::Error::last_os_error());
}
if err > 0 { if err > 0 {
// End indicies of the attribute names // End indicies of the attribute names
// the buffer contains 0-terminates c-strings // the buffer contains 0-terminates c-strings
let idx = buf.iter().enumerate().filter_map(|(i, v)| let idx = buf.iter().enumerate().filter_map(|(i, v)|
if *v == 0 { Some(i) } else { None } if *v == 0 { Some(i) } else { None }
); );
let mut names = Vec::new();
let mut start = 0; let mut start = 0;
for end in idx { for end in idx {
@ -72,15 +78,10 @@ pub fn list_attrs(lister: lister::Lister, path: &Path) -> io::Result<Vec<Attribu
start = c_end; start = c_end;
} }
}
}
Ok(names) Ok(names)
}
else {
Err(io::Error::new(io::ErrorKind::Other, "could not read extended attributes"))
}
}
else {
Err(io::Error::new(io::ErrorKind::Other, "could not read extended attributes"))
}
} }
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]

View File

@ -12,7 +12,6 @@ use unicode_width::UnicodeWidthStr;
use dir::Dir; use dir::Dir;
use options::TimeType; use options::TimeType;
use feature::xattr::{Attribute, FileAttributes};
use self::fields as f; use self::fields as f;
@ -45,10 +44,6 @@ pub struct File<'dir> {
/// cache it. /// cache it.
pub metadata: fs::Metadata, pub metadata: fs::Metadata,
/// List of this file's extended attributes. These are only loaded if the
/// `xattr` feature is in use.
pub xattrs: Vec<Attribute>,
/// A reference to the directory that contains this file, if present. /// A reference to the directory that contains this file, if present.
/// ///
/// Filenames that get passed in on the command-line directly will have no /// Filenames that get passed in on the command-line directly will have no
@ -93,7 +88,6 @@ impl<'dir> File<'dir> {
dir: parent, dir: parent,
metadata: metadata, metadata: metadata,
ext: ext(&filename), ext: ext(&filename),
xattrs: path.symlink_attributes().unwrap_or(Vec::new()),
name: filename.to_string(), name: filename.to_string(),
this: this, this: this,
} }
@ -199,7 +193,6 @@ impl<'dir> File<'dir> {
dir: self.dir, dir: self.dir,
metadata: metadata, metadata: metadata,
ext: ext(&filename), ext: ext(&filename),
xattrs: target_path.attributes().unwrap_or(Vec::new()),
name: filename.to_string(), name: filename.to_string(),
this: None, this: None,
}) })
@ -316,7 +309,7 @@ impl<'dir> File<'dir> {
other_read: has_bit(unix::fs::OTHER_READ), other_read: has_bit(unix::fs::OTHER_READ),
other_write: has_bit(unix::fs::OTHER_WRITE), other_write: has_bit(unix::fs::OTHER_WRITE),
other_execute: has_bit(unix::fs::OTHER_EXECUTE), other_execute: has_bit(unix::fs::OTHER_EXECUTE),
attribute: !self.xattrs.is_empty() attribute: false, // !self.xattrs.is_empty()
} }
} }

View File

@ -1,13 +1,12 @@
use std::error::Error; use std::error::Error;
use std::io; use std::io;
use std::iter::repeat;
use std::path::Path; use std::path::Path;
use std::string::ToString; use std::string::ToString;
use colours::Colours; use colours::Colours;
use column::{Alignment, Column, Cell}; use column::{Alignment, Column, Cell};
use dir::Dir; use dir::Dir;
use feature::xattr::Attribute; use feature::xattr::{Attribute, FileAttributes};
use file::fields as f; use file::fields as f;
use file::File; use file::File;
use options::{Columns, FileFilter, RecurseOptions, SizeFormat}; use options::{Columns, FileFilter, RecurseOptions, SizeFormat};
@ -78,7 +77,7 @@ impl Details {
// Then add files to the table and print it out. // Then add files to the table and print it out.
self.add_files_to_table(&mut table, files, 0); self.add_files_to_table(&mut table, files, 0);
for cell in table.print_table(self.xattr, self.recurse.is_some()) { for cell in table.print_table() {
println!("{}", cell.text); println!("{}", cell.text);
} }
} }
@ -89,6 +88,17 @@ impl Details {
for (index, file) in src.iter().enumerate() { for (index, file) in src.iter().enumerate() {
table.add_file(file, depth, index == src.len() - 1, true); table.add_file(file, depth, index == src.len() - 1, true);
if self.xattr {
match file.path.attributes() {
Ok(xattrs) => {
for xattr in xattrs {
table.add_xattr(xattr, depth + 1, false);
}
},
Err(e) => table.add_error(&e, depth + 1, true, None),
}
}
// There are two types of recursion that exa supports: a tree // There are two types of recursion that exa supports: a tree
// view, which is dealt with here, and multiple listings, which is // view, which is dealt with here, and multiple listings, which is
// dealt with in the main module. So only actually recurse if we // dealt with in the main module. So only actually recurse if we
@ -149,16 +159,9 @@ struct Row {
/// on top have depth 0. /// on top have depth 0.
depth: usize, depth: usize,
/// Vector of this file's extended attributes, if that feature is active.
attrs: Vec<Attribute>,
/// Whether this is the last entry in the directory. This flag is used /// Whether this is the last entry in the directory. This flag is used
/// when calculating the tree view. /// when calculating the tree view.
last: bool, last: bool,
/// Whether this file is a directory and has any children. Also used when
/// calculating the tree view.
children: bool,
} }
impl Row { impl Row {
@ -232,8 +235,6 @@ impl<U> Table<U> where U: Users {
cells: Some(self.columns.iter().map(|c| Cell::paint(self.colours.header, c.header())).collect()), cells: Some(self.columns.iter().map(|c| Cell::paint(self.colours.header, c.header())).collect()),
name: Cell::paint(self.colours.header, "Name"), name: Cell::paint(self.colours.header, "Name"),
last: false, last: false,
attrs: Vec::new(),
children: false,
}; };
self.rows.push(row); self.rows.push(row);
@ -250,8 +251,17 @@ impl<U> Table<U> where U: Users {
cells: None, cells: None,
name: Cell::paint(self.colours.broken_arrow, &error_message), name: Cell::paint(self.colours.broken_arrow, &error_message),
last: last, last: last,
attrs: Vec::new(), };
children: false,
self.rows.push(row);
}
pub fn add_xattr(&mut self, xattr: Attribute, depth: usize, last: bool) {
let row = Row {
depth: depth,
cells: None,
name: Cell::paint(self.colours.perms.attribute, &format!("{}\t{}", xattr.name, xattr.size)),
last: last,
}; };
self.rows.push(row); self.rows.push(row);
@ -269,8 +279,6 @@ impl<U> Table<U> where U: Users {
cells: Some(cells), cells: Some(cells),
name: Cell { text: filename(file, &self.colours, links), length: file.file_name_width() }, name: Cell { text: filename(file, &self.colours, links), length: file.file_name_width() },
last: last, last: last,
attrs: file.xattrs.clone(),
children: file.this.is_some(),
}; };
self.rows.push(row); self.rows.push(row);
@ -445,7 +453,7 @@ impl<U> Table<U> where U: Users {
} }
/// Render the table as a vector of Cells, to be displayed on standard output. /// Render the table as a vector of Cells, to be displayed on standard output.
pub fn print_table(&self, xattr: bool, show_children: bool) -> Vec<Cell> { pub fn print_table(&self) -> Vec<Cell> {
let mut stack = Vec::new(); let mut stack = Vec::new();
let mut cells = Vec::new(); let mut cells = Vec::new();
@ -482,7 +490,6 @@ impl<U> Table<U> where U: Users {
// necessary to maintain information about the previously-printed // necessary to maintain information about the previously-printed
// lines, as the output will change based on whether the // lines, as the output will change based on whether the
// *previous* entry was the last in its directory. // *previous* entry was the last in its directory.
if show_children {
stack.resize(row.depth + 1, TreePart::Edge); stack.resize(row.depth + 1, TreePart::Edge);
stack[row.depth] = if row.last { TreePart::Corner } else { TreePart::Edge }; stack[row.depth] = if row.last { TreePart::Corner } else { TreePart::Edge };
@ -491,9 +498,7 @@ impl<U> Table<U> where U: Users {
filename_length += 4; filename_length += 4;
} }
if row.children {
stack[row.depth] = if row.last { TreePart::Blank } else { TreePart::Line }; stack[row.depth] = if row.last { TreePart::Blank } else { TreePart::Line };
}
// If any tree characters have been printed, then add an extra // If any tree characters have been printed, then add an extra
// space, which makes the output look much better. // space, which makes the output look much better.
@ -501,21 +506,11 @@ impl<U> Table<U> where U: Users {
filename.push(' '); filename.push(' ');
filename_length += 1; filename_length += 1;
} }
}
// Print the name without worrying about padding. // Print the name without worrying about padding.
filename.push_str(&*row.name.text); filename.push_str(&*row.name.text);
filename_length += row.name.length; filename_length += row.name.length;
if xattr {
let width = row.attrs.iter().map(|a| a.name.len()).max().unwrap_or(0);
for attr in row.attrs.iter() {
let name = &*attr.name;
let spaces: String = repeat(" ").take(width - name.len()).collect();
filename.push_str(&*format!("\n{}{} {}", name, spaces, attr.size))
}
}
cell.append(&Cell { text: filename, length: filename_length }); cell.append(&Cell { text: filename, length: filename_length });
cells.push(cell); cells.push(cell);
} }

View File

@ -73,7 +73,7 @@ impl GridDetails {
tables[index].add_file_with_cells(row, file, 0, false, false); tables[index].add_file_with_cells(row, file, 0, false, false);
} }
let columns: Vec<_> = tables.iter().map(|t| t.print_table(false, false)).collect(); let columns: Vec<_> = tables.iter().map(|t| t.print_table()).collect();
let direction = if self.grid.across { grid::Direction::LeftToRight } let direction = if self.grid.across { grid::Direction::LeftToRight }
else { grid::Direction::TopToBottom }; else { grid::Direction::TopToBottom };