mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-03 16:48:45 +00:00
builtin/test: refactor the Token enum to be more granular (#10357)
* builtin/test: Split Token enum into 2-level hierarchy * builtin/test: Rearrange the Token enum hierarchy * builtin/test: Separate Token into Unary and Binary * builtin/test: import IsOkAnd polyfill * builtin/test: Rename enum variants one more time
This commit is contained in:
parent
360342bb9e
commit
08220c2189
1 changed files with 350 additions and 359 deletions
|
@ -4,6 +4,8 @@ use crate::common;
|
||||||
mod test_expressions {
|
mod test_expressions {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
|
#[allow(unused_imports)]
|
||||||
|
use crate::future::IsOkAnd;
|
||||||
use crate::nix::isatty;
|
use crate::nix::isatty;
|
||||||
use crate::wutil::{
|
use crate::wutil::{
|
||||||
file_id_for_path, fish_wcswidth, lwstat, waccess, wcstod::wcstod, wcstoi_opts, wstat,
|
file_id_for_path, fish_wcswidth, lwstat, waccess, wcstod::wcstod, wcstoi_opts, wstat,
|
||||||
|
@ -15,54 +17,128 @@ mod test_expressions {
|
||||||
|
|
||||||
#[derive(Copy, Clone, PartialEq, Eq)]
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
||||||
pub(super) enum Token {
|
pub(super) enum Token {
|
||||||
unknown, // arbitrary string
|
Unknown, // Arbitrary string
|
||||||
|
Unary(UnaryToken), // Takes one string/file
|
||||||
|
Binary(BinaryToken), // Takes two strings/files/numbers
|
||||||
|
UnaryBoolean(UnaryBooleanToken), // Unary truth function
|
||||||
|
BinaryBoolean(Combiner), // Binary truth function
|
||||||
|
ParenOpen, // (
|
||||||
|
ParenClose, // )
|
||||||
|
}
|
||||||
|
|
||||||
bang, // "!", inverts sense
|
impl From<BinaryToken> for Token {
|
||||||
|
fn from(value: BinaryToken) -> Self {
|
||||||
|
Self::Binary(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
filetype_b, // "-b", for block special files
|
impl From<UnaryToken> for Token {
|
||||||
filetype_c, // "-c", for character special files
|
fn from(value: UnaryToken) -> Self {
|
||||||
filetype_d, // "-d", for directories
|
Self::Unary(value)
|
||||||
filetype_e, // "-e", for files that exist
|
}
|
||||||
filetype_f, // "-f", for for regular files
|
}
|
||||||
filetype_G, // "-G", for check effective group id
|
|
||||||
filetype_g, // "-g", for set-group-id
|
|
||||||
filetype_h, // "-h", for symbolic links
|
|
||||||
filetype_k, // "-k", for sticky bit
|
|
||||||
filetype_L, // "-L", same as -h
|
|
||||||
filetype_O, // "-O", for check effective user id
|
|
||||||
filetype_p, // "-p", for FIFO
|
|
||||||
filetype_S, // "-S", socket
|
|
||||||
|
|
||||||
filesize_s, // "-s", size greater than zero
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
||||||
|
pub(super) enum UnaryBooleanToken {
|
||||||
|
Bang, // "!", inverts sense
|
||||||
|
}
|
||||||
|
|
||||||
filedesc_t, // "-t", whether the fd is associated with a terminal
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
||||||
|
pub(super) enum Combiner {
|
||||||
|
And, // "-a", true if left and right are both true
|
||||||
|
Or, // "-o", true if either left or right is true
|
||||||
|
}
|
||||||
|
|
||||||
fileperm_r, // "-r", read permission
|
macro_rules! define_token {
|
||||||
fileperm_u, // "-u", whether file is setuid
|
(
|
||||||
fileperm_w, // "-w", whether file write permission is allowed
|
enum $enum:ident;
|
||||||
fileperm_x, // "-x", whether file execute/search is allowed
|
$(
|
||||||
|
$variant:ident($sub_type:ident) {
|
||||||
|
$($sub_variant:ident)+
|
||||||
|
}
|
||||||
|
)*
|
||||||
|
) => {
|
||||||
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
||||||
|
pub(super) enum $enum {
|
||||||
|
$($variant($sub_type),)*
|
||||||
|
}
|
||||||
|
|
||||||
string_n, // "-n", non-empty string
|
$(
|
||||||
string_z, // "-z", true if length of string is 0
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
||||||
string_equal, // "=", true if strings are identical
|
pub(super) enum $sub_type { $($sub_variant,)+ }
|
||||||
string_not_equal, // "!=", true if strings are not identical
|
|
||||||
|
|
||||||
file_newer, // f1 -nt f2, true if f1 exists and is newer than f2, or there is no f2
|
impl From<$sub_type> for Token {
|
||||||
file_older, // f1 -ot f2, true if f2 exists and f1 does not, or f1 is older than f2
|
fn from(value: $sub_type) -> Token {
|
||||||
file_same, // f1 -ef f2, true if f1 and f2 exist and refer to same file
|
$enum::$variant(value).into()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)*
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
number_equal, // "-eq", true if numbers are equal
|
define_token! {
|
||||||
number_not_equal, // "-ne", true if numbers are not equal
|
enum UnaryToken;
|
||||||
number_greater, // "-gt", true if first number is larger than second
|
|
||||||
number_greater_equal, // "-ge", true if first number is at least second
|
|
||||||
number_lesser, // "-lt", true if first number is smaller than second
|
|
||||||
number_lesser_equal, // "-le", true if first number is at most second
|
|
||||||
|
|
||||||
combine_and, // "-a", true if left and right are both true
|
// based on stat()
|
||||||
combine_or, // "-o", true if either left or right is true
|
FileStat(StatPredicate) {
|
||||||
|
b // "-b", for block special files
|
||||||
|
c // "-c", for character special files
|
||||||
|
d // "-d", for directories
|
||||||
|
e // "-e", for files that exist
|
||||||
|
f // "-f", for for regular files
|
||||||
|
G // "-G", for check effective group id
|
||||||
|
g // "-g", for set-group-id
|
||||||
|
k // "-k", for sticky bit
|
||||||
|
O // "-O", for check effective user id
|
||||||
|
p // "-p", for FIFO
|
||||||
|
S // "-S", socket
|
||||||
|
s // "-s", size greater than zero
|
||||||
|
u // "-u", whether file is setuid
|
||||||
|
}
|
||||||
|
|
||||||
paren_open, // "(", open paren
|
// based on access()
|
||||||
paren_close, // ")", close paren
|
FilePerm(FilePermission) {
|
||||||
|
r // "-r", read permission
|
||||||
|
w // "-w", whether file write permission is allowed
|
||||||
|
x // "-x", whether file execute/search is allowed
|
||||||
|
}
|
||||||
|
|
||||||
|
// miscellaneous
|
||||||
|
FileType(FilePredicate) {
|
||||||
|
h // "-h", for symbolic links
|
||||||
|
L // "-L", same as -h
|
||||||
|
t // "-t", whether the fd is associated with a terminal
|
||||||
|
}
|
||||||
|
|
||||||
|
String(StringPredicate) {
|
||||||
|
n // "-n", non-empty string
|
||||||
|
z // "-z", true if length of string is 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
define_token! {
|
||||||
|
enum BinaryToken;
|
||||||
|
|
||||||
|
// based on inode + more distinguishing info (see FileId struct)
|
||||||
|
FileId(FileComparison) {
|
||||||
|
Newer // f1 -nt f2, true if f1 exists and is newer than f2, or there is no f2
|
||||||
|
Older // f1 -ot f2, true if f2 exists and f1 does not, or f1 is older than f2
|
||||||
|
Same // f1 -ef f2, true if f1 and f2 exist and refer to same file
|
||||||
|
}
|
||||||
|
|
||||||
|
String(StringComparison) {
|
||||||
|
Equal // "=", true if strings are identical
|
||||||
|
NotEqual // "!=", true if strings are not identical
|
||||||
|
}
|
||||||
|
|
||||||
|
Number(NumberComparison) {
|
||||||
|
Equal // "-eq", true if numbers are equal
|
||||||
|
NotEqual // "-ne", true if numbers are not equal
|
||||||
|
Greater // "-gt", true if first number is larger than second
|
||||||
|
GreaterEqual // "-ge", true if first number is at least second
|
||||||
|
Lesser // "-lt", true if first number is smaller than second
|
||||||
|
LesserEqual // "-le", true if first number is at most second
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Our number type. We support both doubles and long longs. We have to support these separately
|
/// Our number type. We support both doubles and long longs. We have to support these separately
|
||||||
|
@ -103,71 +179,50 @@ mod test_expressions {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const UNARY_PRIMARY: u32 = 1 << 0;
|
fn token_for_string(str: &wstr) -> Token {
|
||||||
const BINARY_PRIMARY: u32 = 1 << 1;
|
TOKEN_INFOS.get(str).copied().unwrap_or(Token::Unknown)
|
||||||
|
|
||||||
struct TokenInfo {
|
|
||||||
tok: Token,
|
|
||||||
flags: u32,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TokenInfo {
|
static TOKEN_INFOS: Lazy<HashMap<&'static wstr, Token>> = Lazy::new(|| {
|
||||||
fn new(tok: Token, flags: u32) -> Self {
|
|
||||||
Self { tok, flags }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn token_for_string(str: &wstr) -> &'static TokenInfo {
|
|
||||||
if let Some(res) = TOKEN_INFOS.get(str) {
|
|
||||||
res
|
|
||||||
} else {
|
|
||||||
TOKEN_INFOS
|
|
||||||
.get(L!(""))
|
|
||||||
.expect("Should have token for empty string")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static TOKEN_INFOS: Lazy<HashMap<&'static wstr, TokenInfo>> = Lazy::new(|| {
|
|
||||||
#[rustfmt::skip]
|
|
||||||
let pairs = [
|
let pairs = [
|
||||||
(L!(""), TokenInfo::new(Token::unknown, 0)),
|
(L!(""), Token::Unknown),
|
||||||
(L!("!"), TokenInfo::new(Token::bang, 0)),
|
(L!("!"), Token::UnaryBoolean(UnaryBooleanToken::Bang)),
|
||||||
(L!("-b"), TokenInfo::new(Token::filetype_b, UNARY_PRIMARY)),
|
(L!("-b"), StatPredicate::b.into()),
|
||||||
(L!("-c"), TokenInfo::new(Token::filetype_c, UNARY_PRIMARY)),
|
(L!("-c"), StatPredicate::c.into()),
|
||||||
(L!("-d"), TokenInfo::new(Token::filetype_d, UNARY_PRIMARY)),
|
(L!("-d"), StatPredicate::d.into()),
|
||||||
(L!("-e"), TokenInfo::new(Token::filetype_e, UNARY_PRIMARY)),
|
(L!("-e"), StatPredicate::e.into()),
|
||||||
(L!("-f"), TokenInfo::new(Token::filetype_f, UNARY_PRIMARY)),
|
(L!("-f"), StatPredicate::f.into()),
|
||||||
(L!("-G"), TokenInfo::new(Token::filetype_G, UNARY_PRIMARY)),
|
(L!("-G"), StatPredicate::G.into()),
|
||||||
(L!("-g"), TokenInfo::new(Token::filetype_g, UNARY_PRIMARY)),
|
(L!("-g"), StatPredicate::g.into()),
|
||||||
(L!("-h"), TokenInfo::new(Token::filetype_h, UNARY_PRIMARY)),
|
(L!("-h"), FilePredicate::h.into()),
|
||||||
(L!("-k"), TokenInfo::new(Token::filetype_k, UNARY_PRIMARY)),
|
(L!("-k"), StatPredicate::k.into()),
|
||||||
(L!("-L"), TokenInfo::new(Token::filetype_L, UNARY_PRIMARY)),
|
(L!("-L"), FilePredicate::L.into()),
|
||||||
(L!("-O"), TokenInfo::new(Token::filetype_O, UNARY_PRIMARY)),
|
(L!("-O"), StatPredicate::O.into()),
|
||||||
(L!("-p"), TokenInfo::new(Token::filetype_p, UNARY_PRIMARY)),
|
(L!("-p"), StatPredicate::p.into()),
|
||||||
(L!("-S"), TokenInfo::new(Token::filetype_S, UNARY_PRIMARY)),
|
(L!("-S"), StatPredicate::S.into()),
|
||||||
(L!("-s"), TokenInfo::new(Token::filesize_s, UNARY_PRIMARY)),
|
(L!("-s"), StatPredicate::s.into()),
|
||||||
(L!("-t"), TokenInfo::new(Token::filedesc_t, UNARY_PRIMARY)),
|
(L!("-t"), FilePredicate::t.into()),
|
||||||
(L!("-r"), TokenInfo::new(Token::fileperm_r, UNARY_PRIMARY)),
|
(L!("-r"), FilePermission::r.into()),
|
||||||
(L!("-u"), TokenInfo::new(Token::fileperm_u, UNARY_PRIMARY)),
|
(L!("-u"), StatPredicate::u.into()),
|
||||||
(L!("-w"), TokenInfo::new(Token::fileperm_w, UNARY_PRIMARY)),
|
(L!("-w"), FilePermission::w.into()),
|
||||||
(L!("-x"), TokenInfo::new(Token::fileperm_x, UNARY_PRIMARY)),
|
(L!("-x"), FilePermission::x.into()),
|
||||||
(L!("-n"), TokenInfo::new(Token::string_n, UNARY_PRIMARY)),
|
(L!("-n"), StringPredicate::n.into()),
|
||||||
(L!("-z"), TokenInfo::new(Token::string_z, UNARY_PRIMARY)),
|
(L!("-z"), StringPredicate::z.into()),
|
||||||
(L!("="), TokenInfo::new(Token::string_equal, BINARY_PRIMARY)),
|
(L!("="), StringComparison::Equal.into()),
|
||||||
(L!("!="), TokenInfo::new(Token::string_not_equal, BINARY_PRIMARY)),
|
(L!("!="), StringComparison::NotEqual.into()),
|
||||||
(L!("-nt"), TokenInfo::new(Token::file_newer, BINARY_PRIMARY)),
|
(L!("-nt"), FileComparison::Newer.into()),
|
||||||
(L!("-ot"), TokenInfo::new(Token::file_older, BINARY_PRIMARY)),
|
(L!("-ot"), FileComparison::Older.into()),
|
||||||
(L!("-ef"), TokenInfo::new(Token::file_same, BINARY_PRIMARY)),
|
(L!("-ef"), FileComparison::Same.into()),
|
||||||
(L!("-eq"), TokenInfo::new(Token::number_equal, BINARY_PRIMARY)),
|
(L!("-eq"), NumberComparison::Equal.into()),
|
||||||
(L!("-ne"), TokenInfo::new(Token::number_not_equal, BINARY_PRIMARY)),
|
(L!("-ne"), NumberComparison::NotEqual.into()),
|
||||||
(L!("-gt"), TokenInfo::new(Token::number_greater, BINARY_PRIMARY)),
|
(L!("-gt"), NumberComparison::Greater.into()),
|
||||||
(L!("-ge"), TokenInfo::new(Token::number_greater_equal, BINARY_PRIMARY)),
|
(L!("-ge"), NumberComparison::GreaterEqual.into()),
|
||||||
(L!("-lt"), TokenInfo::new(Token::number_lesser, BINARY_PRIMARY)),
|
(L!("-lt"), NumberComparison::Lesser.into()),
|
||||||
(L!("-le"), TokenInfo::new(Token::number_lesser_equal, BINARY_PRIMARY)),
|
(L!("-le"), NumberComparison::LesserEqual.into()),
|
||||||
(L!("-a"), TokenInfo::new(Token::combine_and, 0)),
|
(L!("-a"), Token::BinaryBoolean(Combiner::And)),
|
||||||
(L!("-o"), TokenInfo::new(Token::combine_or, 0)),
|
(L!("-o"), Token::BinaryBoolean(Combiner::Or)),
|
||||||
(L!("("), TokenInfo::new(Token::paren_open, 0)),
|
(L!("("), Token::ParenOpen),
|
||||||
(L!(")"), TokenInfo::new(Token::paren_close, 0))
|
(L!(")"), Token::ParenClose),
|
||||||
];
|
];
|
||||||
pairs.into_iter().collect()
|
pairs.into_iter().collect()
|
||||||
});
|
});
|
||||||
|
@ -225,10 +280,16 @@ mod test_expressions {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Single argument like -n foo or "just a string".
|
/// Something that is not a token of any other type.
|
||||||
|
struct JustAString {
|
||||||
|
arg: WString,
|
||||||
|
range: Range,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Single argument like -n foo.
|
||||||
struct UnaryPrimary {
|
struct UnaryPrimary {
|
||||||
arg: WString,
|
arg: WString,
|
||||||
token: Token,
|
token: UnaryToken,
|
||||||
range: Range,
|
range: Range,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -236,14 +297,14 @@ mod test_expressions {
|
||||||
struct BinaryPrimary {
|
struct BinaryPrimary {
|
||||||
arg_left: WString,
|
arg_left: WString,
|
||||||
arg_right: WString,
|
arg_right: WString,
|
||||||
token: Token,
|
token: BinaryToken,
|
||||||
range: Range,
|
range: Range,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Unary operator like bang.
|
/// Unary operator like bang.
|
||||||
struct UnaryOperator {
|
struct UnaryOperator {
|
||||||
subject: Box<dyn Expression>,
|
subject: Box<dyn Expression>,
|
||||||
token: Token,
|
token: UnaryBooleanToken,
|
||||||
range: Range,
|
range: Range,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -251,8 +312,7 @@ mod test_expressions {
|
||||||
/// we don't have to worry about precedence in the parser.
|
/// we don't have to worry about precedence in the parser.
|
||||||
struct CombiningExpression {
|
struct CombiningExpression {
|
||||||
subjects: Vec<Box<dyn Expression>>,
|
subjects: Vec<Box<dyn Expression>>,
|
||||||
combiners: Vec<Token>,
|
combiners: Vec<Combiner>,
|
||||||
token: Token,
|
|
||||||
range: Range,
|
range: Range,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -262,6 +322,16 @@ mod test_expressions {
|
||||||
range: Range,
|
range: Range,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Expression for JustAString {
|
||||||
|
fn evaluate(&self, _streams: &mut IoStreams, _errors: &mut Vec<WString>) -> bool {
|
||||||
|
!self.arg.is_empty()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn range(&self) -> Range {
|
||||||
|
self.range.clone()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Expression for UnaryPrimary {
|
impl Expression for UnaryPrimary {
|
||||||
fn evaluate(&self, streams: &mut IoStreams, errors: &mut Vec<WString>) -> bool {
|
fn evaluate(&self, streams: &mut IoStreams, errors: &mut Vec<WString>) -> bool {
|
||||||
unary_primary_evaluate(self.token, &self.arg, streams, errors)
|
unary_primary_evaluate(self.token, &self.arg, streams, errors)
|
||||||
|
@ -284,11 +354,8 @@ mod test_expressions {
|
||||||
|
|
||||||
impl Expression for UnaryOperator {
|
impl Expression for UnaryOperator {
|
||||||
fn evaluate(&self, streams: &mut IoStreams, errors: &mut Vec<WString>) -> bool {
|
fn evaluate(&self, streams: &mut IoStreams, errors: &mut Vec<WString>) -> bool {
|
||||||
if self.token == Token::bang {
|
match self.token {
|
||||||
!self.subject.evaluate(streams, errors)
|
UnaryBooleanToken::Bang => !self.subject.evaluate(streams, errors),
|
||||||
} else {
|
|
||||||
errors.push(L!("Unknown token type in unary_operator_evaluate").to_owned());
|
|
||||||
false
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -300,7 +367,6 @@ mod test_expressions {
|
||||||
impl Expression for CombiningExpression {
|
impl Expression for CombiningExpression {
|
||||||
fn evaluate(&self, streams: &mut IoStreams, errors: &mut Vec<WString>) -> bool {
|
fn evaluate(&self, streams: &mut IoStreams, errors: &mut Vec<WString>) -> bool {
|
||||||
let _res = self.subjects[0].evaluate(streams, errors);
|
let _res = self.subjects[0].evaluate(streams, errors);
|
||||||
if self.token == Token::combine_and || self.token == Token::combine_or {
|
|
||||||
assert!(!self.subjects.is_empty());
|
assert!(!self.subjects.is_empty());
|
||||||
assert!(self.combiners.len() + 1 == self.subjects.len());
|
assert!(self.combiners.len() + 1 == self.subjects.len());
|
||||||
|
|
||||||
|
@ -328,7 +394,7 @@ mod test_expressions {
|
||||||
|
|
||||||
// If the combiner at this index (which corresponding to how we combine with the
|
// If the combiner at this index (which corresponding to how we combine with the
|
||||||
// next subject) is not AND, then exit the loop.
|
// next subject) is not AND, then exit the loop.
|
||||||
if idx + 1 < max && self.combiners[idx] != Token::combine_and {
|
if idx + 1 < max && self.combiners[idx] != Combiner::And {
|
||||||
idx += 1;
|
idx += 1;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -341,9 +407,6 @@ mod test_expressions {
|
||||||
}
|
}
|
||||||
return or_result;
|
return or_result;
|
||||||
}
|
}
|
||||||
errors.push(L!("Unknown token type in CombiningExpression.evaluate").to_owned());
|
|
||||||
false
|
|
||||||
}
|
|
||||||
|
|
||||||
fn range(&self) -> Range {
|
fn range(&self) -> Range {
|
||||||
self.range.clone()
|
self.range.clone()
|
||||||
|
@ -374,20 +437,16 @@ mod test_expressions {
|
||||||
if start >= end {
|
if start >= end {
|
||||||
return self.error(start, sprintf!("Missing argument at index %u", start + 1));
|
return self.error(start, sprintf!("Missing argument at index %u", start + 1));
|
||||||
}
|
}
|
||||||
let tok = token_for_string(self.arg(start)).tok;
|
if let Token::UnaryBoolean(token) = token_for_string(self.arg(start)) {
|
||||||
if tok == Token::bang {
|
let subject = self.parse_unary_expression(start + 1, end)?;
|
||||||
let subject = self.parse_unary_expression(start + 1, end);
|
|
||||||
if let Some(subject) = subject {
|
|
||||||
let range = start..subject.range().end;
|
let range = start..subject.range().end;
|
||||||
return UnaryOperator {
|
return UnaryOperator {
|
||||||
subject,
|
subject,
|
||||||
token: tok,
|
token,
|
||||||
range,
|
range,
|
||||||
}
|
}
|
||||||
.into_some_box();
|
.into_some_box();
|
||||||
}
|
}
|
||||||
return None;
|
|
||||||
}
|
|
||||||
self.parse_primary(start, end)
|
self.parse_primary(start, end)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -407,8 +466,7 @@ mod test_expressions {
|
||||||
while idx < end {
|
while idx < end {
|
||||||
if !first {
|
if !first {
|
||||||
// This is not the first expression, so we expect a combiner.
|
// This is not the first expression, so we expect a combiner.
|
||||||
let combiner = token_for_string(self.arg(idx)).tok;
|
let Token::BinaryBoolean(combiner) = token_for_string(self.arg(idx)) else {
|
||||||
if combiner != Token::combine_and && combiner != Token::combine_or {
|
|
||||||
/* Not a combiner, we're done */
|
/* Not a combiner, we're done */
|
||||||
self.errors.insert(
|
self.errors.insert(
|
||||||
0,
|
0,
|
||||||
|
@ -419,7 +477,7 @@ mod test_expressions {
|
||||||
);
|
);
|
||||||
self.error_idx = idx;
|
self.error_idx = idx;
|
||||||
break;
|
break;
|
||||||
}
|
};
|
||||||
combiners.push(combiner);
|
combiners.push(combiner);
|
||||||
idx += 1;
|
idx += 1;
|
||||||
}
|
}
|
||||||
|
@ -448,7 +506,6 @@ mod test_expressions {
|
||||||
CombiningExpression {
|
CombiningExpression {
|
||||||
subjects,
|
subjects,
|
||||||
combiners,
|
combiners,
|
||||||
token: Token::combine_and,
|
|
||||||
range: start..idx,
|
range: start..idx,
|
||||||
}
|
}
|
||||||
.into_some_box()
|
.into_some_box()
|
||||||
|
@ -467,40 +524,36 @@ mod test_expressions {
|
||||||
}
|
}
|
||||||
|
|
||||||
// All our unary primaries are prefix, so the operator is at start.
|
// All our unary primaries are prefix, so the operator is at start.
|
||||||
let info: &TokenInfo = token_for_string(self.arg(start));
|
let Token::Unary(token) = token_for_string(self.arg(start)) else {
|
||||||
if info.flags & UNARY_PRIMARY == 0 {
|
|
||||||
return None;
|
return None;
|
||||||
}
|
};
|
||||||
UnaryPrimary {
|
UnaryPrimary {
|
||||||
arg: self.arg(start + 1).to_owned(),
|
arg: self.arg(start + 1).to_owned(),
|
||||||
token: info.tok,
|
token,
|
||||||
range: start..start + 2,
|
range: start..start + 2,
|
||||||
}
|
}
|
||||||
.into_some_box()
|
.into_some_box()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_just_a_string(&mut self, start: usize, end: usize) -> Option<Box<dyn Expression>> {
|
fn parse_just_a_string(&mut self, start: usize, end: usize) -> Option<Box<dyn Expression>> {
|
||||||
// Handle a string as a unary primary that is not a token of any other type. e.g. 'test foo -a
|
// Handle a string as a unary primary that is not a token of any other type.
|
||||||
// bar' should evaluate to true. We handle this with a unary primary of test_string_n.
|
// e.g. 'test foo -a bar' should evaluate to true.
|
||||||
|
|
||||||
// We need one argument.
|
// We need one argument.
|
||||||
if start >= end {
|
if start >= end {
|
||||||
return self.error(start, sprintf!("Missing argument at index %u", start + 1));
|
return self.error(start, sprintf!("Missing argument at index %u", start + 1));
|
||||||
}
|
}
|
||||||
|
|
||||||
let info = token_for_string(self.arg(start));
|
let tok = token_for_string(self.arg(start));
|
||||||
if info.tok != Token::unknown {
|
if tok != Token::Unknown {
|
||||||
return self.error(
|
return self.error(
|
||||||
start,
|
start,
|
||||||
sprintf!("Unexpected argument type at index %u", start + 1),
|
sprintf!("Unexpected argument type at index %u", start + 1),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// This is hackish; a nicer way to implement this would be with a "just a string" expression
|
return JustAString {
|
||||||
// type.
|
|
||||||
return UnaryPrimary {
|
|
||||||
arg: self.arg(start).to_owned(),
|
arg: self.arg(start).to_owned(),
|
||||||
token: Token::string_n,
|
|
||||||
range: start..start + 1,
|
range: start..start + 1,
|
||||||
}
|
}
|
||||||
.into_some_box();
|
.into_some_box();
|
||||||
|
@ -519,14 +572,13 @@ mod test_expressions {
|
||||||
}
|
}
|
||||||
|
|
||||||
// All our binary primaries are infix, so the operator is at start + 1.
|
// All our binary primaries are infix, so the operator is at start + 1.
|
||||||
let info = token_for_string(self.arg(start + 1));
|
let Token::Binary(token) = token_for_string(self.arg(start + 1)) else {
|
||||||
if info.flags & BINARY_PRIMARY == 0 {
|
|
||||||
return None;
|
return None;
|
||||||
}
|
};
|
||||||
BinaryPrimary {
|
BinaryPrimary {
|
||||||
arg_left: self.arg(start).to_owned(),
|
arg_left: self.arg(start).to_owned(),
|
||||||
arg_right: self.arg(start + 2).to_owned(),
|
arg_right: self.arg(start + 2).to_owned(),
|
||||||
token: info.tok,
|
token,
|
||||||
range: start..start + 3,
|
range: start..start + 3,
|
||||||
}
|
}
|
||||||
.into_some_box()
|
.into_some_box()
|
||||||
|
@ -539,8 +591,7 @@ mod test_expressions {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Must start with an open expression.
|
// Must start with an open expression.
|
||||||
let open_paren = token_for_string(self.arg(start));
|
if token_for_string(self.arg(start)) != Token::ParenOpen {
|
||||||
if open_paren.tok != Token::paren_open {
|
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -556,8 +607,7 @@ mod test_expressions {
|
||||||
sprintf!("Missing close paren at index %u", close_index + 1),
|
sprintf!("Missing close paren at index %u", close_index + 1),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
let close_paren = token_for_string(self.arg(close_index));
|
if token_for_string(self.arg(close_index)) != Token::ParenClose {
|
||||||
if close_paren.tok != Token::paren_close {
|
|
||||||
return self.error(
|
return self.error(
|
||||||
close_index,
|
close_index,
|
||||||
sprintf!("Expected close paren at index %u", close_index + 1),
|
sprintf!("Expected close paren at index %u", close_index + 1),
|
||||||
|
@ -599,31 +649,24 @@ mod test_expressions {
|
||||||
end: usize,
|
end: usize,
|
||||||
) -> Option<Box<dyn Expression>> {
|
) -> Option<Box<dyn Expression>> {
|
||||||
assert!(end - start == 3);
|
assert!(end - start == 3);
|
||||||
let mut result = None;
|
|
||||||
let center_token = token_for_string(self.arg(start + 1));
|
let center_token = token_for_string(self.arg(start + 1));
|
||||||
if center_token.flags & BINARY_PRIMARY != 0 {
|
|
||||||
result = self.parse_binary_primary(start, end);
|
if matches!(center_token, Token::Binary(_)) {
|
||||||
} else if center_token.tok == Token::combine_and
|
self.parse_binary_primary(start, end)
|
||||||
|| center_token.tok == Token::combine_or
|
} else if let Token::BinaryBoolean(combiner) = center_token {
|
||||||
{
|
let left = self.parse_unary_expression(start, start + 1)?;
|
||||||
let left = self.parse_unary_expression(start, start + 1);
|
let right = self.parse_unary_expression(start + 2, start + 3)?;
|
||||||
let right = self.parse_unary_expression(start + 2, start + 3);
|
|
||||||
if left.is_some() && right.is_some() {
|
|
||||||
// Transfer ownership to the vector of subjects.
|
// Transfer ownership to the vector of subjects.
|
||||||
let combiners = vec![center_token.tok];
|
CombiningExpression {
|
||||||
let subjects = vec![left.unwrap(), right.unwrap()];
|
subjects: vec![left, right],
|
||||||
result = CombiningExpression {
|
combiners: vec![combiner],
|
||||||
subjects,
|
|
||||||
combiners,
|
|
||||||
token: center_token.tok,
|
|
||||||
range: start..end,
|
range: start..end,
|
||||||
}
|
}
|
||||||
.into_some_box()
|
.into_some_box()
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
result = self.parse_unary_expression(start, end);
|
self.parse_unary_expression(start, end)
|
||||||
}
|
}
|
||||||
result
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_4_arg_expression(
|
fn parse_4_arg_expression(
|
||||||
|
@ -632,24 +675,22 @@ mod test_expressions {
|
||||||
end: usize,
|
end: usize,
|
||||||
) -> Option<Box<dyn Expression>> {
|
) -> Option<Box<dyn Expression>> {
|
||||||
assert!(end - start == 4);
|
assert!(end - start == 4);
|
||||||
let mut result = None;
|
|
||||||
let first_token = token_for_string(self.arg(start)).tok;
|
let first_token = token_for_string(self.arg(start));
|
||||||
if first_token == Token::bang {
|
|
||||||
let subject = self.parse_3_arg_expression(start + 1, end);
|
if let Token::UnaryBoolean(token) = first_token {
|
||||||
if let Some(subject) = subject {
|
let subject = self.parse_3_arg_expression(start + 1, end)?;
|
||||||
result = UnaryOperator {
|
UnaryOperator {
|
||||||
subject,
|
subject,
|
||||||
token: first_token,
|
token,
|
||||||
range: start..end,
|
range: start..end,
|
||||||
}
|
}
|
||||||
.into_some_box();
|
.into_some_box()
|
||||||
}
|
} else if first_token == Token::ParenOpen {
|
||||||
} else if first_token == Token::paren_open {
|
self.parse_parenthetical(start, end)
|
||||||
result = self.parse_parenthetical(start, end);
|
|
||||||
} else {
|
} else {
|
||||||
result = self.parse_combining_expression(start, end);
|
self.parse_combining_expression(start, end)
|
||||||
}
|
}
|
||||||
result
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_expression(&mut self, start: usize, end: usize) -> Option<Box<dyn Expression>> {
|
fn parse_expression(&mut self, start: usize, end: usize) -> Option<Box<dyn Expression>> {
|
||||||
|
@ -826,168 +867,118 @@ mod test_expressions {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn binary_primary_evaluate(
|
fn binary_primary_evaluate(
|
||||||
token: Token,
|
token: BinaryToken,
|
||||||
left: &wstr,
|
left: &wstr,
|
||||||
right: &wstr,
|
right: &wstr,
|
||||||
errors: &mut Vec<WString>,
|
errors: &mut Vec<WString>,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
|
match token {
|
||||||
|
BinaryToken::String(StringComparison::Equal) => left == right,
|
||||||
|
BinaryToken::String(StringComparison::NotEqual) => left != right,
|
||||||
|
BinaryToken::FileId(comparison) => {
|
||||||
|
let left = file_id_for_path(left);
|
||||||
|
let right = file_id_for_path(right);
|
||||||
|
match comparison {
|
||||||
|
FileComparison::Newer => right.older_than(&left),
|
||||||
|
FileComparison::Older => left.older_than(&right),
|
||||||
|
FileComparison::Same => left == right,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BinaryToken::Number(comparison) => {
|
||||||
let mut ln = Number::default();
|
let mut ln = Number::default();
|
||||||
let mut rn = Number::default();
|
let mut rn = Number::default();
|
||||||
match token {
|
if !parse_number(left, &mut ln, errors) || !parse_number(right, &mut rn, errors) {
|
||||||
Token::string_equal => left == right,
|
return false;
|
||||||
Token::string_not_equal => left != right,
|
|
||||||
Token::file_newer => file_id_for_path(right).older_than(&file_id_for_path(left)),
|
|
||||||
Token::file_older => file_id_for_path(left).older_than(&file_id_for_path(right)),
|
|
||||||
Token::file_same => file_id_for_path(left) == file_id_for_path(right),
|
|
||||||
Token::number_equal => {
|
|
||||||
parse_number(left, &mut ln, errors)
|
|
||||||
&& parse_number(right, &mut rn, errors)
|
|
||||||
&& ln == rn
|
|
||||||
}
|
}
|
||||||
Token::number_not_equal => {
|
match comparison {
|
||||||
parse_number(left, &mut ln, errors)
|
NumberComparison::Equal => ln == rn,
|
||||||
&& parse_number(right, &mut rn, errors)
|
NumberComparison::NotEqual => ln != rn,
|
||||||
&& ln != rn
|
NumberComparison::Greater => ln > rn,
|
||||||
|
NumberComparison::GreaterEqual => ln >= rn,
|
||||||
|
NumberComparison::Lesser => ln < rn,
|
||||||
|
NumberComparison::LesserEqual => ln <= rn,
|
||||||
}
|
}
|
||||||
Token::number_greater => {
|
|
||||||
parse_number(left, &mut ln, errors)
|
|
||||||
&& parse_number(right, &mut rn, errors)
|
|
||||||
&& ln > rn
|
|
||||||
}
|
|
||||||
Token::number_greater_equal => {
|
|
||||||
parse_number(left, &mut ln, errors)
|
|
||||||
&& parse_number(right, &mut rn, errors)
|
|
||||||
&& ln >= rn
|
|
||||||
}
|
|
||||||
Token::number_lesser => {
|
|
||||||
parse_number(left, &mut ln, errors)
|
|
||||||
&& parse_number(right, &mut rn, errors)
|
|
||||||
&& ln < rn
|
|
||||||
}
|
|
||||||
Token::number_lesser_equal => {
|
|
||||||
parse_number(left, &mut ln, errors)
|
|
||||||
&& parse_number(right, &mut rn, errors)
|
|
||||||
&& ln <= rn
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
errors.push(L!("Unknown token type in binary_primary_evaluate").to_owned());
|
|
||||||
false
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn unary_primary_evaluate(
|
fn unary_primary_evaluate(
|
||||||
token: Token,
|
token: UnaryToken,
|
||||||
arg: &wstr,
|
arg: &wstr,
|
||||||
streams: &mut IoStreams,
|
streams: &mut IoStreams,
|
||||||
errors: &mut Vec<WString>,
|
errors: &mut Vec<WString>,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
const S_ISGID: u32 = 0o2000;
|
|
||||||
const S_ISVTX: u32 = 0o1000;
|
|
||||||
|
|
||||||
// Helper to call wstat and then apply a function to the result.
|
|
||||||
fn stat_and<F>(arg: &wstr, f: F) -> bool
|
|
||||||
where
|
|
||||||
F: FnOnce(std::fs::Metadata) -> bool,
|
|
||||||
{
|
|
||||||
wstat(arg).map_or(false, f)
|
|
||||||
}
|
|
||||||
|
|
||||||
match token {
|
match token {
|
||||||
Token::filetype_b => {
|
#[allow(clippy::unnecessary_cast)] // mode_t is u32 on many platforms, but not all
|
||||||
|
UnaryToken::FileStat(stat_token) => {
|
||||||
|
let Ok(md) = wstat(arg) else {
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
const S_ISUID: u32 = libc::S_ISUID as u32;
|
||||||
|
const S_ISGID: u32 = libc::S_ISGID as u32;
|
||||||
|
const S_ISVTX: u32 = libc::S_ISVTX as u32;
|
||||||
|
|
||||||
|
match stat_token {
|
||||||
// "-b", for block special files
|
// "-b", for block special files
|
||||||
stat_and(arg, |buf| buf.file_type().is_block_device())
|
StatPredicate::b => md.file_type().is_block_device(),
|
||||||
}
|
|
||||||
Token::filetype_c => {
|
|
||||||
// "-c", for character special files
|
// "-c", for character special files
|
||||||
stat_and(arg, |buf: std::fs::Metadata| {
|
StatPredicate::c => md.file_type().is_char_device(),
|
||||||
buf.file_type().is_char_device()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
Token::filetype_d => {
|
|
||||||
// "-d", for directories
|
// "-d", for directories
|
||||||
stat_and(arg, |buf: std::fs::Metadata| buf.file_type().is_dir())
|
StatPredicate::d => md.file_type().is_dir(),
|
||||||
}
|
|
||||||
Token::filetype_e => {
|
|
||||||
// "-e", for files that exist
|
// "-e", for files that exist
|
||||||
stat_and(arg, |_| true)
|
StatPredicate::e => true,
|
||||||
}
|
|
||||||
Token::filetype_f => {
|
|
||||||
// "-f", for regular files
|
// "-f", for regular files
|
||||||
stat_and(arg, |buf| buf.file_type().is_file())
|
StatPredicate::f => md.file_type().is_file(),
|
||||||
}
|
|
||||||
Token::filetype_G => {
|
|
||||||
// "-G", for check effective group id
|
// "-G", for check effective group id
|
||||||
stat_and(arg, |buf| crate::nix::getegid() == buf.gid())
|
StatPredicate::G => md.gid() == crate::nix::getegid(),
|
||||||
}
|
|
||||||
Token::filetype_g => {
|
|
||||||
// "-g", for set-group-id
|
// "-g", for set-group-id
|
||||||
stat_and(arg, |buf| buf.permissions().mode() & S_ISGID != 0)
|
StatPredicate::g => md.permissions().mode() & S_ISGID != 0,
|
||||||
|
// "-k", for sticky bit
|
||||||
|
StatPredicate::k => md.permissions().mode() & S_ISVTX != 0,
|
||||||
|
// "-O", for check effective user id
|
||||||
|
StatPredicate::O => md.uid() == crate::nix::geteuid(),
|
||||||
|
// "-p", for FIFO
|
||||||
|
StatPredicate::p => md.file_type().is_fifo(),
|
||||||
|
// "-S", socket
|
||||||
|
StatPredicate::S => md.file_type().is_socket(),
|
||||||
|
// "-s", size greater than zero
|
||||||
|
StatPredicate::s => md.len() > 0,
|
||||||
|
// "-u", whether file is setuid
|
||||||
|
StatPredicate::u => md.permissions().mode() & S_ISUID != 0,
|
||||||
}
|
}
|
||||||
Token::filetype_h | Token::filetype_L => {
|
}
|
||||||
|
UnaryToken::FileType(file_type) => {
|
||||||
|
match file_type {
|
||||||
// "-h", for symbolic links
|
// "-h", for symbolic links
|
||||||
// "-L", same as -h
|
// "-L", same as -h
|
||||||
lwstat(arg).map_or(false, |buf| buf.file_type().is_symlink())
|
FilePredicate::h | FilePredicate::L => {
|
||||||
|
lwstat(arg).is_ok_and(|md| md.file_type().is_symlink())
|
||||||
}
|
}
|
||||||
Token::filetype_k => {
|
|
||||||
// "-k", for sticky bit
|
|
||||||
stat_and(arg, |buf| buf.permissions().mode() & S_ISVTX != 0)
|
|
||||||
}
|
|
||||||
Token::filetype_O => {
|
|
||||||
// "-O", for check effective user id
|
|
||||||
stat_and(arg, |buf: std::fs::Metadata| {
|
|
||||||
crate::nix::geteuid() == buf.uid()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
Token::filetype_p => {
|
|
||||||
// "-p", for FIFO
|
|
||||||
stat_and(arg, |buf: std::fs::Metadata| buf.file_type().is_fifo())
|
|
||||||
}
|
|
||||||
Token::filetype_S => {
|
|
||||||
// "-S", socket
|
|
||||||
stat_and(arg, |buf| buf.file_type().is_socket())
|
|
||||||
}
|
|
||||||
Token::filesize_s => {
|
|
||||||
// "-s", size greater than zero
|
|
||||||
stat_and(arg, |buf| buf.len() > 0)
|
|
||||||
}
|
|
||||||
Token::filedesc_t => {
|
|
||||||
// "-t", whether the fd is associated with a terminal
|
// "-t", whether the fd is associated with a terminal
|
||||||
|
FilePredicate::t => {
|
||||||
let mut num = Number::default();
|
let mut num = Number::default();
|
||||||
parse_number(arg, &mut num, errors) && num.isatty(streams)
|
parse_number(arg, &mut num, errors) && num.isatty(streams)
|
||||||
}
|
}
|
||||||
Token::fileperm_r => {
|
}
|
||||||
|
}
|
||||||
|
UnaryToken::FilePerm(permission) => {
|
||||||
|
let mode = match permission {
|
||||||
// "-r", read permission
|
// "-r", read permission
|
||||||
waccess(arg, libc::R_OK) == 0
|
FilePermission::r => libc::R_OK,
|
||||||
}
|
|
||||||
Token::fileperm_u => {
|
|
||||||
// "-u", whether file is setuid
|
|
||||||
#[allow(clippy::unnecessary_cast)]
|
|
||||||
stat_and(arg, |buf| {
|
|
||||||
buf.permissions().mode() & (libc::S_ISUID as u32) != 0
|
|
||||||
})
|
|
||||||
}
|
|
||||||
Token::fileperm_w => {
|
|
||||||
// "-w", whether file write permission is allowed
|
// "-w", whether file write permission is allowed
|
||||||
waccess(arg, libc::W_OK) == 0
|
FilePermission::w => libc::W_OK,
|
||||||
}
|
|
||||||
Token::fileperm_x => {
|
|
||||||
// "-x", whether file execute/search is allowed
|
// "-x", whether file execute/search is allowed
|
||||||
waccess(arg, libc::X_OK) == 0
|
FilePermission::x => libc::X_OK,
|
||||||
|
};
|
||||||
|
waccess(arg, mode) == 0
|
||||||
}
|
}
|
||||||
Token::string_n => {
|
UnaryToken::String(predicate) => match predicate {
|
||||||
// "-n", non-empty string
|
// "-n", non-empty string
|
||||||
!arg.is_empty()
|
StringPredicate::n => !arg.is_empty(),
|
||||||
}
|
|
||||||
Token::string_z => {
|
|
||||||
// "-z", true if length of string is 0
|
// "-z", true if length of string is 0
|
||||||
arg.is_empty()
|
StringPredicate::z => arg.is_empty(),
|
||||||
}
|
},
|
||||||
_ => {
|
|
||||||
// Unknown token.
|
|
||||||
errors.push(L!("Unknown token type in unary_primary_evaluate").to_owned());
|
|
||||||
false
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue