exa/src/options.rs

228 lines
7.0 KiB
Rust
Raw Normal View History

2014-05-25 16:14:50 +00:00
extern crate getopts;
extern crate natord;
2014-05-25 16:14:50 +00:00
use file::File;
use column::{Column, SizeFormat};
2014-11-23 21:29:11 +00:00
use column::Column::*;
2015-01-12 20:08:42 +00:00
use output::View;
2014-11-23 21:29:11 +00:00
use term::dimensions;
use std::ascii::AsciiExt;
2015-01-12 18:44:39 +00:00
use std::slice::Iter;
pub enum SortField {
2014-07-22 20:27:36 +00:00
Unsorted, Name, Extension, Size, FileInode
}
2014-12-14 18:22:56 +00:00
impl Copy for SortField { }
impl SortField {
2015-01-12 21:14:27 +00:00
fn from_word(word: String) -> Result<SortField, Error> {
match word.as_slice() {
2015-01-12 21:14:27 +00:00
"name" => Ok(SortField::Name),
"size" => Ok(SortField::Size),
"ext" => Ok(SortField::Extension),
"none" => Ok(SortField::Unsorted),
"inode" => Ok(SortField::FileInode),
field => Err(no_sort_field(field))
}
}
}
2015-01-12 21:14:27 +00:00
fn no_sort_field(field: &str) -> Error {
Error::InvalidOptions(getopts::Fail::UnrecognizedOption(format!("--sort {}", field)))
}
pub struct Options {
pub list_dirs: bool,
2015-01-12 21:47:05 +00:00
pub path_strs: Vec<String>,
2015-01-12 18:44:39 +00:00
reverse: bool,
show_invisibles: bool,
sort_field: SortField,
pub view: View,
}
2015-01-12 21:47:05 +00:00
#[derive(Show)]
2015-01-12 21:14:27 +00:00
pub enum Error {
InvalidOptions(getopts::Fail),
Help(String),
}
impl Options {
2015-01-12 21:47:05 +00:00
pub fn getopts(args: &[String]) -> Result<Options, Error> {
2015-01-12 21:14:27 +00:00
let opts = &[
getopts::optflag("1", "oneline", "display one entry per line"),
getopts::optflag("a", "all", "show dot-files"),
getopts::optflag("b", "binary", "use binary prefixes in file sizes"),
2014-12-18 07:04:31 +00:00
getopts::optflag("B", "bytes", "list file sizes in bytes, without prefixes"),
getopts::optflag("d", "list-dirs", "list directories as regular files"),
getopts::optflag("g", "group", "show group as well as user"),
getopts::optflag("h", "header", "show a header row at the top"),
getopts::optflag("H", "links", "show number of hard links"),
getopts::optflag("l", "long", "display extended details and attributes"),
getopts::optflag("i", "inode", "show each file's inode number"),
getopts::optflag("r", "reverse", "reverse order of files"),
getopts::optopt ("s", "sort", "field to sort by", "WORD"),
getopts::optflag("S", "blocks", "show number of file system blocks"),
getopts::optflag("x", "across", "sort multi-column view entries across"),
2014-11-25 15:54:42 +00:00
getopts::optflag("?", "help", "show list of command-line options"),
2014-05-25 16:14:50 +00:00
];
2014-11-25 20:50:23 +00:00
2015-01-12 21:47:05 +00:00
let matches = match getopts::getopts(args, opts) {
Ok(m) => m,
2015-01-12 21:14:27 +00:00
Err(e) => return Err(Error::InvalidOptions(e)),
2014-11-25 15:54:42 +00:00
};
2014-11-25 20:50:23 +00:00
2014-11-25 15:54:42 +00:00
if matches.opt_present("help") {
2015-01-12 21:14:27 +00:00
return Err(Error::Help(getopts::usage("Usage:\n exa [options] [files...]", opts)));
2014-05-25 16:14:50 +00:00
}
2014-11-25 15:54:42 +00:00
2015-01-12 21:14:27 +00:00
let sort_field = match matches.opt_str("sort") {
Some(word) => try!(SortField::from_word(word)),
None => SortField::Name,
};
2014-11-25 15:54:42 +00:00
Ok(Options {
list_dirs: matches.opt_present("list-dirs"),
path_strs: if matches.free.is_empty() { vec![ ".".to_string() ] } else { matches.free.clone() },
reverse: matches.opt_present("reverse"),
show_invisibles: matches.opt_present("all"),
2015-01-12 21:14:27 +00:00
sort_field: sort_field,
view: Options::view(&matches),
})
2014-05-25 16:14:50 +00:00
}
2014-11-25 20:50:23 +00:00
2015-01-12 18:44:39 +00:00
pub fn path_strings(&self) -> Iter<String> {
self.path_strs.iter()
}
2014-11-23 22:36:03 +00:00
fn view(matches: &getopts::Matches) -> View {
if matches.opt_present("long") {
2015-01-12 20:08:42 +00:00
View::Details(Options::columns(matches), matches.opt_present("header"))
}
else if matches.opt_present("oneline") {
2014-11-23 21:29:11 +00:00
View::Lines
}
else {
2014-11-23 21:29:11 +00:00
match dimensions() {
None => View::Lines,
Some((width, _)) => View::Grid(matches.opt_present("across"), width),
}
}
}
2014-11-25 20:50:23 +00:00
2014-11-23 22:36:03 +00:00
fn columns(matches: &getopts::Matches) -> Vec<Column> {
2014-06-22 06:44:00 +00:00
let mut columns = vec![];
if matches.opt_present("inode") {
columns.push(Inode);
}
columns.push(Permissions);
if matches.opt_present("links") {
columns.push(HardLinks);
}
2014-11-25 20:50:23 +00:00
2015-01-12 21:47:05 +00:00
if matches.opt_present("binary") {
columns.push(FileSize(SizeFormat::BinaryBytes))
}
else if matches.opt_present("bytes") {
columns.push(FileSize(SizeFormat::JustBytes))
}
else {
columns.push(FileSize(SizeFormat::DecimalBytes))
}
2014-06-22 07:09:16 +00:00
if matches.opt_present("blocks") {
columns.push(Blocks);
}
columns.push(User);
2014-05-26 17:08:58 +00:00
if matches.opt_present("group") {
columns.push(Group);
}
columns.push(FileName);
columns
}
2014-05-26 19:24:51 +00:00
fn should_display(&self, f: &File) -> bool {
if self.show_invisibles {
true
2014-11-24 02:12:52 +00:00
}
else {
!f.name.as_slice().starts_with(".")
}
}
2014-05-25 18:42:31 +00:00
pub fn transform_files<'a>(&self, unordered_files: Vec<File<'a>>) -> Vec<File<'a>> {
let mut files: Vec<File<'a>> = unordered_files.into_iter()
.filter(|f| self.should_display(f))
.collect();
match self.sort_field {
2014-11-23 21:29:11 +00:00
SortField::Unsorted => {},
SortField::Name => files.sort_by(|a, b| natord::compare(a.name.as_slice(), b.name.as_slice())),
2014-11-23 21:29:11 +00:00
SortField::Size => files.sort_by(|a, b| a.stat.size.cmp(&b.stat.size)),
SortField::FileInode => files.sort_by(|a, b| a.stat.unstable.inode.cmp(&b.stat.unstable.inode)),
SortField::Extension => files.sort_by(|a, b| {
2015-01-01 02:37:10 +00:00
let exts = a.ext.clone().map(|e| e.to_ascii_lowercase()).cmp(&b.ext.clone().map(|e| e.to_ascii_lowercase()));
let names = a.name.to_ascii_lowercase().cmp(&b.name.to_ascii_lowercase());
exts.cmp(&names)
}),
}
if self.reverse {
files.reverse();
}
files
}
}
2015-01-12 21:47:05 +00:00
#[cfg(test)]
mod test {
use super::Options;
use super::Error;
use super::Error::*;
fn is_helpful(error: Result<Options, Error>) -> bool {
match error {
Err(Help(_)) => true,
_ => false,
}
}
#[test]
fn help() {
let opts = Options::getopts(&[ "--help".to_string() ]);
assert!(is_helpful(opts))
}
#[test]
fn help_with_file() {
let opts = Options::getopts(&[ "--help".to_string(), "me".to_string() ]);
assert!(is_helpful(opts))
}
#[test]
fn files() {
let opts = Options::getopts(&[ "this file".to_string(), "that file".to_string() ]);
assert_eq!(opts.unwrap().path_strs, vec![ "this file".to_string(), "that file".to_string() ])
}
#[test]
fn no_args() {
let opts = Options::getopts(&[]);
assert_eq!(opts.unwrap().path_strs, vec![ ".".to_string() ])
}
#[test]
fn view() {
let opts = Options::getopts(&[ "this file".to_string(), "that file".to_string() ]);
assert_eq!(opts.unwrap().path_strs, vec![ "this file".to_string(), "that file".to_string() ])
}
}