2017-07-22 17:06:05 +00:00
|
|
|
|
//! Parsing command-line strings into exa options.
|
|
|
|
|
//!
|
2017-07-24 07:34:50 +00:00
|
|
|
|
//! This module imports exa’s configuration types, such as `View` (the details
|
|
|
|
|
//! of displaying multiple files) and `DirAction` (what to do when encountering
|
|
|
|
|
//! a directory), and implements `deduce` methods on them so they can be
|
|
|
|
|
//! configured using command-line options.
|
|
|
|
|
//!
|
2017-07-22 17:06:05 +00:00
|
|
|
|
//!
|
|
|
|
|
//! ## Useless and overridden options
|
|
|
|
|
//!
|
|
|
|
|
//! Let’s say exa was invoked with just one argument: `exa --inode`. The
|
|
|
|
|
//! `--inode` option is used in the details view, where it adds the inode
|
|
|
|
|
//! column to the output. But because the details view is *only* activated with
|
|
|
|
|
//! the `--long` argument, adding `--inode` without it would not have any
|
|
|
|
|
//! effect.
|
|
|
|
|
//!
|
|
|
|
|
//! For a long time, exa’s philosophy was that the user should be warned
|
|
|
|
|
//! whenever they could be mistaken like this. If you tell exa to display the
|
|
|
|
|
//! inode, and it *doesn’t* display the inode, isn’t that more annoying than
|
|
|
|
|
//! having it throw an error back at you?
|
|
|
|
|
//!
|
|
|
|
|
//! However, this doesn’t take into account *configuration*. Say a user wants
|
|
|
|
|
//! to configure exa so that it lists inodes in the details view, but otherwise
|
|
|
|
|
//! functions normally. A common way to do this for command-line programs is to
|
|
|
|
|
//! define a shell alias that specifies the details they want to use every
|
|
|
|
|
//! time. For the inode column, the alias would be:
|
|
|
|
|
//!
|
|
|
|
|
//! `alias exa="exa --inode"`
|
|
|
|
|
//!
|
|
|
|
|
//! Using this alias means that although the inode column will be shown in the
|
|
|
|
|
//! details view, you’re now *only* allowed to use the details view, as any
|
|
|
|
|
//! other view type will result in an error. Oops!
|
|
|
|
|
//!
|
|
|
|
|
//! Another example is when an option is specified twice, such as `exa
|
|
|
|
|
//! --sort=Name --sort=size`. Did the user change their mind about sorting, and
|
|
|
|
|
//! accidentally specify the option twice?
|
|
|
|
|
//!
|
|
|
|
|
//! Again, exa rejected this case, throwing an error back to the user instead
|
|
|
|
|
//! of trying to guess how they want their output sorted. And again, this
|
|
|
|
|
//! doesn’t take into account aliases being used to set defaults. A user who
|
|
|
|
|
//! wants their files to be sorted case-insensitively may configure their shell
|
|
|
|
|
//! with the following:
|
|
|
|
|
//!
|
|
|
|
|
//! `alias exa="exa --sort=Name"`
|
|
|
|
|
//!
|
|
|
|
|
//! Just like the earlier example, the user now can’t use any other sort order,
|
|
|
|
|
//! because exa refuses to guess which one they meant. It’s *more* annoying to
|
|
|
|
|
//! have to go back and edit the command than if there were no error.
|
|
|
|
|
//!
|
|
|
|
|
//! Fortunately, there’s a heuristic for telling which options came from an
|
|
|
|
|
//! alias and which came from the actual command-line: aliased options are
|
|
|
|
|
//! nearer the beginning of the options array, and command-line options are
|
|
|
|
|
//! nearer the end. This means that after the options have been parsed, exa
|
|
|
|
|
//! needs to traverse them *backwards* to find the last-most-specified one.
|
|
|
|
|
//!
|
|
|
|
|
//! For example, invoking exa with `exa --sort=size` when that alias is present
|
|
|
|
|
//! would result in a full command-line of:
|
|
|
|
|
//!
|
|
|
|
|
//! `exa --sort=Name --sort=size`
|
|
|
|
|
//!
|
|
|
|
|
//! `--sort=size` should override `--sort=Name` because it’s closer to the end
|
|
|
|
|
//! of the arguments array. In fact, because there’s no way to tell where the
|
|
|
|
|
//! arguments came from -- it’s just a heuristic -- this will still work even
|
|
|
|
|
//! if no aliases are being used!
|
|
|
|
|
//!
|
|
|
|
|
//! Finally, this isn’t just useful when options could override each other.
|
|
|
|
|
//! Creating an alias `exal=”exa --long --inode --header”` then invoking `exal
|
|
|
|
|
//! --grid --long` shouldn’t complain about `--long` being given twice when
|
|
|
|
|
//! it’s clear what the user wants.
|
|
|
|
|
|
|
|
|
|
|
2017-07-26 16:48:18 +00:00
|
|
|
|
use std::ffi::{OsStr, OsString};
|
2016-04-17 19:38:37 +00:00
|
|
|
|
|
|
|
|
|
use fs::feature::xattr;
|
2017-07-24 07:34:50 +00:00
|
|
|
|
use fs::dir_action::DirAction;
|
|
|
|
|
use fs::filter::FileFilter;
|
|
|
|
|
use output::{View, Mode};
|
2017-06-25 23:53:48 +00:00
|
|
|
|
use output::details;
|
2016-04-17 19:38:37 +00:00
|
|
|
|
|
|
|
|
|
mod dir_action;
|
|
|
|
|
mod filter;
|
2017-07-24 07:34:50 +00:00
|
|
|
|
mod view;
|
2016-04-17 19:38:37 +00:00
|
|
|
|
|
|
|
|
|
mod help;
|
2017-06-23 21:50:29 +00:00
|
|
|
|
use self::help::HelpString;
|
2016-04-17 19:38:37 +00:00
|
|
|
|
|
|
|
|
|
mod misfire;
|
|
|
|
|
pub use self::misfire::Misfire;
|
|
|
|
|
|
2017-07-12 11:03:07 +00:00
|
|
|
|
mod parser;
|
2017-07-26 16:48:18 +00:00
|
|
|
|
mod flags;
|
|
|
|
|
use self::parser::Matches;
|
2017-07-12 11:03:07 +00:00
|
|
|
|
|
2016-04-17 19:38:37 +00:00
|
|
|
|
|
|
|
|
|
/// These **options** represent a parsed, error-checked versions of the
|
|
|
|
|
/// user’s command-line options.
|
2017-07-05 19:16:04 +00:00
|
|
|
|
#[derive(Debug)]
|
2016-04-17 19:38:37 +00:00
|
|
|
|
pub struct Options {
|
|
|
|
|
|
|
|
|
|
/// The action to perform when encountering a directory rather than a
|
|
|
|
|
/// regular file.
|
|
|
|
|
pub dir_action: DirAction,
|
|
|
|
|
|
|
|
|
|
/// How to sort and filter files before outputting them.
|
|
|
|
|
pub filter: FileFilter,
|
|
|
|
|
|
|
|
|
|
/// The type of output to use (lines, grid, or details).
|
|
|
|
|
pub view: View,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Options {
|
|
|
|
|
|
2017-05-18 23:08:13 +00:00
|
|
|
|
// Even though the arguments go in as OsStrings, they come out
|
|
|
|
|
// as Strings. Invalid UTF-8 won’t be parsed, but it won’t make
|
|
|
|
|
// exa core dump either.
|
|
|
|
|
//
|
|
|
|
|
// https://github.com/rust-lang-nursery/getopts/pull/29
|
|
|
|
|
|
2016-04-17 19:38:37 +00:00
|
|
|
|
/// Call getopts on the given slice of command-line strings.
|
|
|
|
|
#[allow(unused_results)]
|
2017-07-26 16:48:18 +00:00
|
|
|
|
pub fn getopts<'args, I>(args: I) -> Result<(Options, Vec<&'args OsStr>), Misfire>
|
|
|
|
|
where I: IntoIterator<Item=&'args OsString> {
|
2016-04-17 19:38:37 +00:00
|
|
|
|
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let matches = match parser::parse(&flags::ALL_ARGS, args) {
|
2016-04-17 19:38:37 +00:00
|
|
|
|
Ok(m) => m,
|
|
|
|
|
Err(e) => return Err(Misfire::InvalidOptions(e)),
|
|
|
|
|
};
|
|
|
|
|
|
2017-07-26 16:48:18 +00:00
|
|
|
|
if matches.has(&flags::HELP) {
|
2017-06-23 21:50:29 +00:00
|
|
|
|
let help = HelpString {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
only_long: matches.has(&flags::LONG),
|
2017-06-23 21:50:29 +00:00
|
|
|
|
git: cfg!(feature="git"),
|
|
|
|
|
xattrs: xattr::ENABLED,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return Err(Misfire::Help(help));
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
2017-07-26 16:48:18 +00:00
|
|
|
|
else if matches.has(&flags::VERSION) {
|
2016-04-17 19:38:37 +00:00
|
|
|
|
return Err(Misfire::Version);
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-26 16:35:50 +00:00
|
|
|
|
let options = Options::deduce(&matches)?;
|
2017-07-26 16:48:18 +00:00
|
|
|
|
Ok((options, matches.frees))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Whether the View specified in this set of options includes a Git
|
|
|
|
|
/// status column. It’s only worth trying to discover a repository if the
|
|
|
|
|
/// results will end up being displayed.
|
|
|
|
|
pub fn should_scan_for_git(&self) -> bool {
|
2017-06-24 21:39:15 +00:00
|
|
|
|
match self.view.mode {
|
2017-07-05 20:01:01 +00:00
|
|
|
|
Mode::Details(details::Options { table: Some(ref table), .. }) |
|
|
|
|
|
Mode::GridDetails(_, details::Options { table: Some(ref table), .. }) => table.should_scan_for_git(),
|
2016-04-17 19:38:37 +00:00
|
|
|
|
_ => false,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Determines the complete set of options based on the given command-line
|
|
|
|
|
/// arguments, after they’ve been parsed.
|
2017-07-26 16:48:18 +00:00
|
|
|
|
fn deduce(matches: &Matches) -> Result<Options, Misfire> {
|
2017-03-31 16:09:32 +00:00
|
|
|
|
let dir_action = DirAction::deduce(matches)?;
|
|
|
|
|
let filter = FileFilter::deduce(matches)?;
|
2017-06-26 07:28:32 +00:00
|
|
|
|
let view = View::deduce(matches)?;
|
2016-04-17 19:38:37 +00:00
|
|
|
|
|
2017-05-18 21:43:32 +00:00
|
|
|
|
Ok(Options { dir_action, view, filter })
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2017-07-26 16:48:18 +00:00
|
|
|
|
|
2016-04-17 19:38:37 +00:00
|
|
|
|
#[cfg(test)]
|
|
|
|
|
mod test {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
use super::{Options, Misfire, flags};
|
|
|
|
|
use std::ffi::OsString;
|
2017-07-24 07:34:50 +00:00
|
|
|
|
use fs::filter::{SortField, SortCase};
|
2016-04-17 19:38:37 +00:00
|
|
|
|
use fs::feature::xattr;
|
|
|
|
|
|
|
|
|
|
fn is_helpful<T>(misfire: Result<T, Misfire>) -> bool {
|
|
|
|
|
match misfire {
|
|
|
|
|
Err(Misfire::Help(_)) => true,
|
|
|
|
|
_ => false,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-26 16:48:18 +00:00
|
|
|
|
/// Creates an `OSStr` (used in tests)
|
|
|
|
|
#[cfg(test)]
|
|
|
|
|
fn os(input: &'static str) -> OsString {
|
|
|
|
|
let mut os = OsString::new();
|
|
|
|
|
os.push(input);
|
|
|
|
|
os
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-04-17 19:38:37 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn help() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--help") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
2016-04-17 19:38:37 +00:00
|
|
|
|
assert!(is_helpful(opts))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn help_with_file() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--help"), os("me") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
2016-04-17 19:38:37 +00:00
|
|
|
|
assert!(is_helpful(opts))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn files() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
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") ])
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn no_args() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
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
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn file_sizes() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--long"), os("--binary"), os("--bytes") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Conflict(&flags::BINARY, &flags::BYTES))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn just_binary() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--binary") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::BINARY, false, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn just_bytes() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--bytes") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::BYTES, false, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn long_across() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--long"), os("--across") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::ACROSS, true, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn oneline_across() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--oneline"), os("--across") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::ACROSS, true, &flags::ONE_LINE))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn just_header() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--header") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::HEADER, false, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn just_group() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--group") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::GROUP, false, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn just_inode() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--inode") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::INODE, false, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn just_links() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--links") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::LINKS, false, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn just_blocks() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--blocks") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::BLOCKS, false, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_sort_size() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--sort=size") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
2016-04-17 19:38:37 +00:00
|
|
|
|
assert_eq!(opts.unwrap().0.filter.sort_field, SortField::Size);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_sort_name() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--sort=name") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
2016-04-17 19:38:37 +00:00
|
|
|
|
assert_eq!(opts.unwrap().0.filter.sort_field, SortField::Name(SortCase::Sensitive));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_sort_name_lowercase() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--sort=Name") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
2016-04-17 19:38:37 +00:00
|
|
|
|
assert_eq!(opts.unwrap().0.filter.sort_field, SortField::Name(SortCase::Insensitive));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[cfg(feature="git")]
|
|
|
|
|
fn just_git() {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--git") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::GIT, false, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn extended_without_long() {
|
|
|
|
|
if xattr::ENABLED {
|
2017-07-26 16:48:18 +00:00
|
|
|
|
let args = [ os("--extended") ];
|
|
|
|
|
let opts = Options::getopts(&args);
|
|
|
|
|
assert_eq!(opts.unwrap_err(), Misfire::Useless(&flags::EXTENDED, false, &flags::LONG))
|
2016-04-17 19:38:37 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|