|
|
@@ -69,9 +69,7 @@
|
|
|
//! it’s clear what the user wants.
|
|
|
|
|
|
|
|
|
-use std::ffi::OsStr;
|
|
|
-
|
|
|
-use getopts;
|
|
|
+use std::ffi::{OsStr, OsString};
|
|
|
|
|
|
use fs::feature::xattr;
|
|
|
use fs::dir_action::DirAction;
|
|
|
@@ -90,6 +88,8 @@ mod misfire;
|
|
|
pub use self::misfire::Misfire;
|
|
|
|
|
|
mod parser;
|
|
|
+mod flags;
|
|
|
+use self::parser::Matches;
|
|
|
|
|
|
|
|
|
/// These **options** represent a parsed, error-checked versions of the
|
|
|
@@ -118,77 +118,29 @@ impl Options {
|
|
|
|
|
|
/// Call getopts on the given slice of command-line strings.
|
|
|
#[allow(unused_results)]
|
|
|
- pub fn getopts<C>(args: C) -> Result<(Options, Vec<String>), Misfire>
|
|
|
- where C: IntoIterator, C::Item: AsRef<OsStr> {
|
|
|
- let mut opts = getopts::Options::new();
|
|
|
-
|
|
|
- opts.optflag("v", "version", "show version of exa");
|
|
|
- opts.optflag("?", "help", "show list of command-line options");
|
|
|
-
|
|
|
- // Display options
|
|
|
- opts.optflag("1", "oneline", "display one entry per line");
|
|
|
- opts.optflag("l", "long", "display extended file metadata in a table");
|
|
|
- opts.optflag("G", "grid", "display entries as a grid (default)");
|
|
|
- opts.optflag("x", "across", "sort the grid across, rather than downwards");
|
|
|
- opts.optflag("R", "recurse", "recurse into directories");
|
|
|
- opts.optflag("T", "tree", "recurse into directories as a tree");
|
|
|
- opts.optflag("F", "classify", "display type indicator by file names (one of */=@|)");
|
|
|
- opts.optopt ("", "color", "when to use terminal colours", "WHEN");
|
|
|
- opts.optopt ("", "colour", "when to use terminal colours", "WHEN");
|
|
|
- opts.optflag("", "color-scale", "highlight levels of file sizes distinctly");
|
|
|
- opts.optflag("", "colour-scale", "highlight levels of file sizes distinctly");
|
|
|
-
|
|
|
- // Filtering and sorting options
|
|
|
- opts.optflag("", "group-directories-first", "sort directories before other files");
|
|
|
- opts.optflagmulti("a", "all", "show hidden and 'dot' files");
|
|
|
- opts.optflag("d", "list-dirs", "list directories like regular files");
|
|
|
- opts.optopt ("L", "level", "limit the depth of recursion", "DEPTH");
|
|
|
- opts.optflag("r", "reverse", "reverse the sert order");
|
|
|
- opts.optopt ("s", "sort", "which field to sort by", "WORD");
|
|
|
- opts.optopt ("I", "ignore-glob", "ignore files that match these glob patterns", "GLOB1|GLOB2...");
|
|
|
-
|
|
|
- // Long view options
|
|
|
- opts.optflag("b", "binary", "list file sizes with binary prefixes");
|
|
|
- opts.optflag("B", "bytes", "list file sizes in bytes, without prefixes");
|
|
|
- opts.optflag("g", "group", "list each file's group");
|
|
|
- opts.optflag("h", "header", "add a header row to each column");
|
|
|
- opts.optflag("H", "links", "list each file's number of hard links");
|
|
|
- opts.optflag("i", "inode", "list each file's inode number");
|
|
|
- opts.optflag("m", "modified", "use the modified timestamp field");
|
|
|
- opts.optflag("S", "blocks", "list each file's number of file system blocks");
|
|
|
- opts.optopt ("t", "time", "which timestamp field to show", "WORD");
|
|
|
- opts.optflag("u", "accessed", "use the accessed timestamp field");
|
|
|
- opts.optflag("U", "created", "use the created timestamp field");
|
|
|
- opts.optopt ("", "time-style", "how to format timestamp fields", "STYLE");
|
|
|
-
|
|
|
- if cfg!(feature="git") {
|
|
|
- opts.optflag("", "git", "list each file's git status");
|
|
|
- }
|
|
|
-
|
|
|
- if xattr::ENABLED {
|
|
|
- opts.optflag("@", "extended", "list each file's extended attribute keys and sizes");
|
|
|
- }
|
|
|
+ pub fn getopts<'args, I>(args: I) -> Result<(Options, Vec<&'args OsStr>), Misfire>
|
|
|
+ where I: IntoIterator<Item=&'args OsString> {
|
|
|
|
|
|
- let matches = match opts.parse(args) {
|
|
|
+ let matches = match parser::parse(&flags::ALL_ARGS, args) {
|
|
|
Ok(m) => m,
|
|
|
Err(e) => return Err(Misfire::InvalidOptions(e)),
|
|
|
};
|
|
|
|
|
|
- if matches.opt_present("help") {
|
|
|
+ if matches.has(&flags::HELP) {
|
|
|
let help = HelpString {
|
|
|
- only_long: matches.opt_present("long"),
|
|
|
+ only_long: matches.has(&flags::LONG),
|
|
|
git: cfg!(feature="git"),
|
|
|
xattrs: xattr::ENABLED,
|
|
|
};
|
|
|
|
|
|
return Err(Misfire::Help(help));
|
|
|
}
|
|
|
- else if matches.opt_present("version") {
|
|
|
+ else if matches.has(&flags::VERSION) {
|
|
|
return Err(Misfire::Version);
|
|
|
}
|
|
|
|
|
|
let options = Options::deduce(&matches)?;
|
|
|
- Ok((options, matches.free))
|
|
|
+ Ok((options, matches.frees))
|
|
|
}
|
|
|
|
|
|
/// Whether the View specified in this set of options includes a Git
|
|
|
@@ -204,7 +156,7 @@ impl Options {
|
|
|
|
|
|
/// Determines the complete set of options based on the given command-line
|
|
|
/// arguments, after they’ve been parsed.
|
|
|
- fn deduce(matches: &getopts::Matches) -> Result<Options, Misfire> {
|
|
|
+ fn deduce(matches: &Matches) -> Result<Options, Misfire> {
|
|
|
let dir_action = DirAction::deduce(matches)?;
|
|
|
let filter = FileFilter::deduce(matches)?;
|
|
|
let view = View::deduce(matches)?;
|
|
|
@@ -214,9 +166,11 @@ impl Options {
|
|
|
}
|
|
|
|
|
|
|
|
|
+
|
|
|
#[cfg(test)]
|
|
|
mod test {
|
|
|
- use super::{Options, Misfire};
|
|
|
+ use super::{Options, Misfire, flags};
|
|
|
+ use std::ffi::OsString;
|
|
|
use fs::DotFilter;
|
|
|
use fs::filter::{SortField, SortCase};
|
|
|
use fs::feature::xattr;
|
|
|
@@ -228,152 +182,183 @@ mod test {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+ /// Creates an `OSStr` (used in tests)
|
|
|
+ #[cfg(test)]
|
|
|
+ fn os(input: &'static str) -> OsString {
|
|
|
+ let mut os = OsString::new();
|
|
|
+ os.push(input);
|
|
|
+ os
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
#[test]
|
|
|
fn help() {
|
|
|
- let opts = Options::getopts(&[ "--help".to_string() ]);
|
|
|
+ let args = [ os("--help") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
assert!(is_helpful(opts))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn help_with_file() {
|
|
|
- let opts = Options::getopts(&[ "--help".to_string(), "me".to_string() ]);
|
|
|
+ let args = [ os("--help"), os("me") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
assert!(is_helpful(opts))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn files() {
|
|
|
- let args = Options::getopts(&[ "this file".to_string(), "that file".to_string() ]).unwrap().1;
|
|
|
- assert_eq!(args, vec![ "this file".to_string(), "that file".to_string() ])
|
|
|
+ let args = [ os("this file"), os("that file") ];
|
|
|
+ let outs = Options::getopts(&args).unwrap().1;
|
|
|
+ assert_eq!(outs, vec![ &os("this file"), &os("that file") ])
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn no_args() {
|
|
|
- let nothing: Vec<String> = Vec::new();
|
|
|
- let args = Options::getopts(¬hing).unwrap().1;
|
|
|
- assert!(args.is_empty()); // Listing the `.` directory is done in main.rs
|
|
|
+ let nothing: Vec<OsString> = Vec::new();
|
|
|
+ let outs = Options::getopts(¬hing).unwrap().1;
|
|
|
+ assert!(outs.is_empty()); // Listing the `.` directory is done in main.rs
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn file_sizes() {
|
|
|
- let opts = Options::getopts(&[ "--long", "--binary", "--bytes" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Conflict("binary", "bytes"))
|
|
|
+ let args = [ os("--long"), os("--binary"), os("--bytes") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Conflict(&flags::BINARY, &flags::BYTES))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn just_binary() {
|
|
|
- let opts = Options::getopts(&[ "--binary" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("binary", false, "long"))
|
|
|
+ let args = [ os("--binary") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::BINARY, false, &flags::LONG))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn just_bytes() {
|
|
|
- let opts = Options::getopts(&[ "--bytes" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("bytes", false, "long"))
|
|
|
+ let args = [ os("--bytes") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::BYTES, false, &flags::LONG))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn long_across() {
|
|
|
- let opts = Options::getopts(&[ "--long", "--across" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("across", true, "long"))
|
|
|
+ let args = [ os("--long"), os("--across") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::ACROSS, true, &flags::LONG))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn oneline_across() {
|
|
|
- let opts = Options::getopts(&[ "--oneline", "--across" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("across", true, "oneline"))
|
|
|
+ let args = [ os("--oneline"), os("--across") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::ACROSS, true, &flags::ONE_LINE))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn just_header() {
|
|
|
- let opts = Options::getopts(&[ "--header" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("header", false, "long"))
|
|
|
+ let args = [ os("--header") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::HEADER, false, &flags::LONG))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn just_group() {
|
|
|
- let opts = Options::getopts(&[ "--group" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("group", false, "long"))
|
|
|
+ let args = [ os("--group") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::GROUP, false, &flags::LONG))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn just_inode() {
|
|
|
- let opts = Options::getopts(&[ "--inode" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("inode", false, "long"))
|
|
|
+ let args = [ os("--inode") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::INODE, false, &flags::LONG))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn just_links() {
|
|
|
- let opts = Options::getopts(&[ "--links" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("links", false, "long"))
|
|
|
+ let args = [ os("--links") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::LINKS, false, &flags::LONG))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn just_blocks() {
|
|
|
- let opts = Options::getopts(&[ "--blocks" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("blocks", false, "long"))
|
|
|
+ let args = [ os("--blocks") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::BLOCKS, false, &flags::LONG))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn test_sort_size() {
|
|
|
- let opts = Options::getopts(&[ "--sort=size" ]);
|
|
|
+ let args = [ os("--sort=size") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
assert_eq!(opts.unwrap().0.filter.sort_field, SortField::Size);
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn test_sort_name() {
|
|
|
- let opts = Options::getopts(&[ "--sort=name" ]);
|
|
|
+ let args = [ os("--sort=name") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
assert_eq!(opts.unwrap().0.filter.sort_field, SortField::Name(SortCase::Sensitive));
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn test_sort_name_lowercase() {
|
|
|
- let opts = Options::getopts(&[ "--sort=Name" ]);
|
|
|
+ let args = [ os("--sort=Name") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
assert_eq!(opts.unwrap().0.filter.sort_field, SortField::Name(SortCase::Insensitive));
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
#[cfg(feature="git")]
|
|
|
fn just_git() {
|
|
|
- let opts = Options::getopts(&[ "--git" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("git", false, "long"))
|
|
|
+ let args = [ os("--git") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::GIT, false, &flags::LONG))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn extended_without_long() {
|
|
|
if xattr::ENABLED {
|
|
|
- let opts = Options::getopts(&[ "--extended" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("extended", false, "long"))
|
|
|
+ let args = [ os("--extended") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::EXTENDED, false, &flags::LONG))
|
|
|
}
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn level_without_recurse_or_tree() {
|
|
|
- let opts = Options::getopts(&[ "--level", "69105" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless2("level", "recurse", "tree"))
|
|
|
+ let args = [ os("--level"), os("69105") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::Useless2(&flags::LEVEL, &flags::RECURSE, &flags::TREE))
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn all_all_with_tree() {
|
|
|
- let opts = Options::getopts(&[ "--all", "--all", "--tree" ]);
|
|
|
- assert_eq!(opts.unwrap_err(), Misfire::Useless("all --all", true, "tree"))
|
|
|
+ let args = [ os("--all"), os("--all"), os("--tree") ];
|
|
|
+ let opts = Options::getopts(&args);
|
|
|
+ assert_eq!(opts.unwrap_err(), Misfire::TreeAllAll)
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn nowt() {
|
|
|
- let nothing: Vec<String> = Vec::new();
|
|
|
+ let nothing: Vec<OsString> = Vec::new();
|
|
|
let dots = Options::getopts(¬hing).unwrap().0.filter.dot_filter;
|
|
|
assert_eq!(dots, DotFilter::JustFiles);
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn all() {
|
|
|
- let dots = Options::getopts(&[ "--all".to_string() ]).unwrap().0.filter.dot_filter;
|
|
|
+ let args = [ os("--all") ];
|
|
|
+ let dots = Options::getopts(&args).unwrap().0.filter.dot_filter;
|
|
|
assert_eq!(dots, DotFilter::Dotfiles);
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
|
fn allall() {
|
|
|
- let dots = Options::getopts(&[ "-a".to_string(), "-a".to_string() ]).unwrap().0.filter.dot_filter;
|
|
|
+ let args = [ os("-a"), os("-a") ];
|
|
|
+ let dots = Options::getopts(&args).unwrap().0.filter.dot_filter;
|
|
|
assert_eq!(dots, DotFilter::DotfilesAndDots);
|
|
|
}
|
|
|
}
|