mirror of
https://github.com/Llewellynvdm/exa.git
synced 2025-02-08 21:48:25 +00:00
133 lines
6.3 KiB
Rust
133 lines
6.3 KiB
Rust
//! Parsing the options for `DirAction`.
|
||
|
||
use crate::options::parser::MatchedFlags;
|
||
use crate::options::{flags, OptionsError, NumberSource};
|
||
|
||
use crate::fs::dir_action::{DirAction, RecurseOptions};
|
||
|
||
|
||
impl DirAction {
|
||
|
||
/// Determine which action to perform when trying to list a directory.
|
||
/// There are three possible actions, and they overlap somewhat: the
|
||
/// `--tree` flag is another form of recursion, so those two are allowed
|
||
/// to both be present, but the `--list-dirs` flag is used separately.
|
||
pub fn deduce(matches: &MatchedFlags<'_>, can_tree: bool) -> Result<Self, OptionsError> {
|
||
let recurse = matches.has(&flags::RECURSE)?;
|
||
let as_file = matches.has(&flags::LIST_DIRS)?;
|
||
let tree = matches.has(&flags::TREE)?;
|
||
|
||
if matches.is_strict() {
|
||
// Early check for --level when it wouldn’t do anything
|
||
if ! recurse && ! tree && matches.count(&flags::LEVEL) > 0 {
|
||
return Err(OptionsError::Useless2(&flags::LEVEL, &flags::RECURSE, &flags::TREE));
|
||
}
|
||
else if recurse && as_file {
|
||
return Err(OptionsError::Conflict(&flags::RECURSE, &flags::LIST_DIRS));
|
||
}
|
||
else if tree && as_file {
|
||
return Err(OptionsError::Conflict(&flags::TREE, &flags::LIST_DIRS));
|
||
}
|
||
}
|
||
|
||
if tree && can_tree {
|
||
// Tree is only appropriate in details mode, so this has to
|
||
// examine the View, which should have already been deduced by now
|
||
Ok(Self::Recurse(RecurseOptions::deduce(matches, true)?))
|
||
}
|
||
else if recurse {
|
||
Ok(Self::Recurse(RecurseOptions::deduce(matches, false)?))
|
||
}
|
||
else if as_file {
|
||
Ok(Self::AsFile)
|
||
}
|
||
else {
|
||
Ok(Self::List)
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
impl RecurseOptions {
|
||
|
||
/// Determine which files should be recursed into, based on the `--level`
|
||
/// flag’s value, and whether the `--tree` flag was passed, which was
|
||
/// determined earlier. The maximum level should be a number, and this
|
||
/// will fail with an `Err` if it isn’t.
|
||
pub fn deduce(matches: &MatchedFlags<'_>, tree: bool) -> Result<Self, OptionsError> {
|
||
if let Some(level) = matches.get(&flags::LEVEL)? {
|
||
let arg_str = level.to_string_lossy();
|
||
match arg_str.parse() {
|
||
Ok(l) => {
|
||
Ok(Self { tree, max_depth: Some(l) })
|
||
}
|
||
Err(e) => {
|
||
let source = NumberSource::Arg(&flags::LEVEL);
|
||
Err(OptionsError::FailedParse(arg_str.to_string(), source, e))
|
||
}
|
||
}
|
||
}
|
||
else {
|
||
Ok(Self { tree, max_depth: None })
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
#[cfg(test)]
|
||
mod test {
|
||
use super::*;
|
||
use crate::options::flags;
|
||
use crate::options::parser::Flag;
|
||
|
||
macro_rules! test {
|
||
($name:ident: $type:ident <- $inputs:expr; $stricts:expr => $result:expr) => {
|
||
#[test]
|
||
fn $name() {
|
||
use crate::options::parser::Arg;
|
||
use crate::options::test::parse_for_test;
|
||
use crate::options::test::Strictnesses::*;
|
||
|
||
static TEST_ARGS: &[&Arg] = &[&flags::RECURSE, &flags::LIST_DIRS, &flags::TREE, &flags::LEVEL ];
|
||
for result in parse_for_test($inputs.as_ref(), TEST_ARGS, $stricts, |mf| $type::deduce(mf, true)) {
|
||
assert_eq!(result, $result);
|
||
}
|
||
}
|
||
};
|
||
}
|
||
|
||
|
||
// Default behaviour
|
||
test!(empty: DirAction <- []; Both => Ok(DirAction::List));
|
||
|
||
// Listing files as directories
|
||
test!(dirs_short: DirAction <- ["-d"]; Both => Ok(DirAction::AsFile));
|
||
test!(dirs_long: DirAction <- ["--list-dirs"]; Both => Ok(DirAction::AsFile));
|
||
|
||
// Recursing
|
||
use self::DirAction::Recurse;
|
||
test!(rec_short: DirAction <- ["-R"]; Both => Ok(Recurse(RecurseOptions { tree: false, max_depth: None })));
|
||
test!(rec_long: DirAction <- ["--recurse"]; Both => Ok(Recurse(RecurseOptions { tree: false, max_depth: None })));
|
||
test!(rec_lim_short: DirAction <- ["-RL4"]; Both => Ok(Recurse(RecurseOptions { tree: false, max_depth: Some(4) })));
|
||
test!(rec_lim_short_2: DirAction <- ["-RL=5"]; Both => Ok(Recurse(RecurseOptions { tree: false, max_depth: Some(5) })));
|
||
test!(rec_lim_long: DirAction <- ["--recurse", "--level", "666"]; Both => Ok(Recurse(RecurseOptions { tree: false, max_depth: Some(666) })));
|
||
test!(rec_lim_long_2: DirAction <- ["--recurse", "--level=0118"]; Both => Ok(Recurse(RecurseOptions { tree: false, max_depth: Some(118) })));
|
||
test!(tree: DirAction <- ["--tree"]; Both => Ok(Recurse(RecurseOptions { tree: true, max_depth: None })));
|
||
test!(rec_tree: DirAction <- ["--recurse", "--tree"]; Both => Ok(Recurse(RecurseOptions { tree: true, max_depth: None })));
|
||
test!(rec_short_tree: DirAction <- ["-TR"]; Both => Ok(Recurse(RecurseOptions { tree: true, max_depth: None })));
|
||
|
||
// Overriding --list-dirs, --recurse, and --tree
|
||
test!(dirs_recurse: DirAction <- ["--list-dirs", "--recurse"]; Last => Ok(Recurse(RecurseOptions { tree: false, max_depth: None })));
|
||
test!(dirs_tree: DirAction <- ["--list-dirs", "--tree"]; Last => Ok(Recurse(RecurseOptions { tree: true, max_depth: None })));
|
||
test!(just_level: DirAction <- ["--level=4"]; Last => Ok(DirAction::List));
|
||
|
||
test!(dirs_recurse_2: DirAction <- ["--list-dirs", "--recurse"]; Complain => Err(OptionsError::Conflict(&flags::RECURSE, &flags::LIST_DIRS)));
|
||
test!(dirs_tree_2: DirAction <- ["--list-dirs", "--tree"]; Complain => Err(OptionsError::Conflict(&flags::TREE, &flags::LIST_DIRS)));
|
||
test!(just_level_2: DirAction <- ["--level=4"]; Complain => Err(OptionsError::Useless2(&flags::LEVEL, &flags::RECURSE, &flags::TREE)));
|
||
|
||
|
||
// Overriding levels
|
||
test!(overriding_1: DirAction <- ["-RL=6", "-L=7"]; Last => Ok(Recurse(RecurseOptions { tree: false, max_depth: Some(7) })));
|
||
test!(overriding_2: DirAction <- ["-RL=6", "-L=7"]; Complain => Err(OptionsError::Duplicate(Flag::Short(b'L'), Flag::Short(b'L'))));
|
||
}
|