mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-08 11:08:53 +00:00
50a7798041
builtin_test stashes some variables in statics, to support the `test -t` expression. However this will cause conflicts with concurrent execution, where we may want to run two `test` expressions at once. Do the grunt work of threading the data into all places it needs to go.
913 lines
34 KiB
C++
913 lines
34 KiB
C++
// Functions used for implementing the test builtin.
|
|
//
|
|
// Implemented from scratch (yes, really) by way of IEEE 1003.1 as reference.
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
#include "builtin.h"
|
|
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
|
|
#include <cerrno>
|
|
#include <cmath>
|
|
#include <cstdarg>
|
|
#include <cstring>
|
|
#include <cwchar>
|
|
#include <cwctype>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <type_traits>
|
|
#include <utility>
|
|
|
|
#include "common.h"
|
|
#include "io.h"
|
|
#include "parser.h"
|
|
#include "wutil.h" // IWYU pragma: keep
|
|
|
|
using std::move;
|
|
using std::unique_ptr;
|
|
|
|
namespace {
|
|
namespace test_expressions {
|
|
|
|
enum token_t {
|
|
test_unknown, // arbitrary string
|
|
|
|
test_bang, // "!", inverts sense
|
|
|
|
test_filetype_b, // "-b", for block special files
|
|
test_filetype_c, // "-c", for character special files
|
|
test_filetype_d, // "-d", for directories
|
|
test_filetype_e, // "-e", for files that exist
|
|
test_filetype_f, // "-f", for for regular files
|
|
test_filetype_G, // "-G", for check effective group id
|
|
test_filetype_g, // "-g", for set-group-id
|
|
test_filetype_h, // "-h", for symbolic links
|
|
test_filetype_k, // "-k", for sticky bit
|
|
test_filetype_L, // "-L", same as -h
|
|
test_filetype_O, // "-O", for check effective user id
|
|
test_filetype_p, // "-p", for FIFO
|
|
test_filetype_S, // "-S", socket
|
|
|
|
test_filesize_s, // "-s", size greater than zero
|
|
|
|
test_filedesc_t, // "-t", whether the fd is associated with a terminal
|
|
|
|
test_fileperm_r, // "-r", read permission
|
|
test_fileperm_u, // "-u", whether file is setuid
|
|
test_fileperm_w, // "-w", whether file write permission is allowed
|
|
test_fileperm_x, // "-x", whether file execute/search is allowed
|
|
|
|
test_string_n, // "-n", non-empty string
|
|
test_string_z, // "-z", true if length of string is 0
|
|
test_string_equal, // "=", true if strings are identical
|
|
test_string_not_equal, // "!=", true if strings are not identical
|
|
|
|
test_number_equal, // "-eq", true if numbers are equal
|
|
test_number_not_equal, // "-ne", true if numbers are not equal
|
|
test_number_greater, // "-gt", true if first number is larger than second
|
|
test_number_greater_equal, // "-ge", true if first number is at least second
|
|
test_number_lesser, // "-lt", true if first number is smaller than second
|
|
test_number_lesser_equal, // "-le", true if first number is at most second
|
|
|
|
test_combine_and, // "-a", true if left and right are both true
|
|
test_combine_or, // "-o", true if either left or right is true
|
|
|
|
test_paren_open, // "(", open paren
|
|
test_paren_close, // ")", close paren
|
|
};
|
|
|
|
/// Our number type. We support both doubles and long longs. We have to support these separately
|
|
/// because some integers are not representable as doubles; these may come up in practice (e.g.
|
|
/// inodes).
|
|
class number_t {
|
|
// A number has an integral base and a floating point delta.
|
|
// Conceptually the number is base + delta.
|
|
// We enforce the property that 0 <= delta < 1.
|
|
long long base;
|
|
double delta;
|
|
|
|
public:
|
|
number_t(long long base, double delta) : base(base), delta(delta) {
|
|
assert(0.0 <= delta && delta < 1.0 && "Invalid delta");
|
|
}
|
|
number_t() : number_t(0, 0.0) {}
|
|
|
|
// Compare two numbers. Returns an integer -1, 0, 1 corresponding to whether we are less than,
|
|
// equal to, or greater than the rhs.
|
|
int compare(number_t rhs) const {
|
|
if (this->base != rhs.base) return (this->base > rhs.base) - (this->base < rhs.base);
|
|
return (this->delta > rhs.delta) - (this->delta < rhs.delta);
|
|
}
|
|
|
|
// Return true if the number is a tty().
|
|
bool isatty(const io_streams_t *streams) const {
|
|
if (delta != 0.0 || base > INT_MAX || base < INT_MIN) return false;
|
|
int bint = static_cast<int>(base);
|
|
if (bint == 0) return ::isatty(streams->stdin_fd);
|
|
if (bint == 1) return !streams->out_is_redirected && ::isatty(STDOUT_FILENO);
|
|
if (bint == 2) return !streams->err_is_redirected && ::isatty(STDERR_FILENO);
|
|
return ::isatty(bint);
|
|
}
|
|
};
|
|
|
|
static bool binary_primary_evaluate(test_expressions::token_t token, const wcstring &left,
|
|
const wcstring &right, wcstring_list_t &errors);
|
|
static bool unary_primary_evaluate(test_expressions::token_t token, const wcstring &arg,
|
|
io_streams_t *streams, wcstring_list_t &errors);
|
|
|
|
enum { UNARY_PRIMARY = 1 << 0, BINARY_PRIMARY = 1 << 1 };
|
|
|
|
struct token_info_t {
|
|
token_t tok;
|
|
unsigned int flags;
|
|
};
|
|
|
|
const token_info_t *token_for_string(const wcstring &str) {
|
|
static const std::map<wcstring, const token_info_t> token_infos = {
|
|
{L"", {test_unknown, 0}},
|
|
{L"!", {test_bang, 0}},
|
|
{L"-b", {test_filetype_b, UNARY_PRIMARY}},
|
|
{L"-c", {test_filetype_c, UNARY_PRIMARY}},
|
|
{L"-d", {test_filetype_d, UNARY_PRIMARY}},
|
|
{L"-e", {test_filetype_e, UNARY_PRIMARY}},
|
|
{L"-f", {test_filetype_f, UNARY_PRIMARY}},
|
|
{L"-G", {test_filetype_G, UNARY_PRIMARY}},
|
|
{L"-g", {test_filetype_g, UNARY_PRIMARY}},
|
|
{L"-h", {test_filetype_h, UNARY_PRIMARY}},
|
|
{L"-k", {test_filetype_k, UNARY_PRIMARY}},
|
|
{L"-L", {test_filetype_L, UNARY_PRIMARY}},
|
|
{L"-O", {test_filetype_O, UNARY_PRIMARY}},
|
|
{L"-p", {test_filetype_p, UNARY_PRIMARY}},
|
|
{L"-S", {test_filetype_S, UNARY_PRIMARY}},
|
|
{L"-s", {test_filesize_s, UNARY_PRIMARY}},
|
|
{L"-t", {test_filedesc_t, UNARY_PRIMARY}},
|
|
{L"-r", {test_fileperm_r, UNARY_PRIMARY}},
|
|
{L"-u", {test_fileperm_u, UNARY_PRIMARY}},
|
|
{L"-w", {test_fileperm_w, UNARY_PRIMARY}},
|
|
{L"-x", {test_fileperm_x, UNARY_PRIMARY}},
|
|
{L"-n", {test_string_n, UNARY_PRIMARY}},
|
|
{L"-z", {test_string_z, UNARY_PRIMARY}},
|
|
{L"=", {test_string_equal, BINARY_PRIMARY}},
|
|
{L"!=", {test_string_not_equal, BINARY_PRIMARY}},
|
|
{L"-eq", {test_number_equal, BINARY_PRIMARY}},
|
|
{L"-ne", {test_number_not_equal, BINARY_PRIMARY}},
|
|
{L"-gt", {test_number_greater, BINARY_PRIMARY}},
|
|
{L"-ge", {test_number_greater_equal, BINARY_PRIMARY}},
|
|
{L"-lt", {test_number_lesser, BINARY_PRIMARY}},
|
|
{L"-le", {test_number_lesser_equal, BINARY_PRIMARY}},
|
|
{L"-a", {test_combine_and, 0}},
|
|
{L"-o", {test_combine_or, 0}},
|
|
{L"(", {test_paren_open, 0}},
|
|
{L")", {test_paren_close, 0}}};
|
|
|
|
auto t = token_infos.find(str);
|
|
if (t != token_infos.end()) return &t->second;
|
|
return &token_infos.find(L"")->second;
|
|
}
|
|
|
|
// Grammar.
|
|
//
|
|
// <expr> = <combining_expr>
|
|
//
|
|
// <combining_expr> = <unary_expr> and/or <combining_expr> |
|
|
// <unary_expr>
|
|
//
|
|
// <unary_expr> = bang <unary_expr> |
|
|
// <primary>
|
|
//
|
|
// <primary> = <unary_primary> arg |
|
|
// arg <binary_primary> arg |
|
|
// '(' <expr> ')'
|
|
|
|
class expression;
|
|
class test_parser {
|
|
private:
|
|
wcstring_list_t strings;
|
|
wcstring_list_t errors;
|
|
int error_idx;
|
|
|
|
unique_ptr<expression> error(unsigned int idx, const wchar_t *fmt, ...);
|
|
void add_error(unsigned int idx, const wchar_t *fmt, ...);
|
|
|
|
const wcstring &arg(unsigned int idx) { return strings.at(idx); }
|
|
|
|
public:
|
|
explicit test_parser(wcstring_list_t val) : strings(std::move(val)) {}
|
|
|
|
unique_ptr<expression> parse_expression(unsigned int start, unsigned int end);
|
|
unique_ptr<expression> parse_3_arg_expression(unsigned int start, unsigned int end);
|
|
unique_ptr<expression> parse_4_arg_expression(unsigned int start, unsigned int end);
|
|
unique_ptr<expression> parse_combining_expression(unsigned int start, unsigned int end);
|
|
unique_ptr<expression> parse_unary_expression(unsigned int start, unsigned int end);
|
|
|
|
unique_ptr<expression> parse_primary(unsigned int start, unsigned int end);
|
|
unique_ptr<expression> parse_parenthentical(unsigned int start, unsigned int end);
|
|
unique_ptr<expression> parse_unary_primary(unsigned int start, unsigned int end);
|
|
unique_ptr<expression> parse_binary_primary(unsigned int start, unsigned int end);
|
|
unique_ptr<expression> parse_just_a_string(unsigned int start, unsigned int end);
|
|
|
|
static unique_ptr<expression> parse_args(const wcstring_list_t &args, wcstring &err,
|
|
wchar_t *program_name);
|
|
};
|
|
|
|
struct range_t {
|
|
unsigned int start;
|
|
unsigned int end;
|
|
|
|
range_t(unsigned s, unsigned e) : start(s), end(e) {}
|
|
};
|
|
|
|
/// Base class for expressions.
|
|
class expression {
|
|
protected:
|
|
expression(token_t what, range_t where) : token(what), range(where) {}
|
|
|
|
public:
|
|
const token_t token;
|
|
range_t range;
|
|
|
|
virtual ~expression() = default;
|
|
|
|
/// Evaluate returns true if the expression is true (i.e. STATUS_CMD_OK).
|
|
virtual bool evaluate(io_streams_t *streams, wcstring_list_t &errors) = 0;
|
|
};
|
|
|
|
/// Single argument like -n foo or "just a string".
|
|
class unary_primary final : public expression {
|
|
public:
|
|
wcstring arg;
|
|
unary_primary(token_t tok, range_t where, wcstring what)
|
|
: expression(tok, where), arg(std::move(what)) {}
|
|
bool evaluate(io_streams_t *streams, wcstring_list_t &errors) override;
|
|
};
|
|
|
|
/// Two argument primary like foo != bar.
|
|
class binary_primary final : public expression {
|
|
public:
|
|
wcstring arg_left;
|
|
wcstring arg_right;
|
|
|
|
binary_primary(token_t tok, range_t where, wcstring left, wcstring right)
|
|
: expression(tok, where), arg_left(std::move(left)), arg_right(std::move(right)) {}
|
|
bool evaluate(io_streams_t *streams, wcstring_list_t &errors) override;
|
|
};
|
|
|
|
/// Unary operator like bang.
|
|
class unary_operator final : public expression {
|
|
public:
|
|
unique_ptr<expression> subject;
|
|
unary_operator(token_t tok, range_t where, unique_ptr<expression> exp)
|
|
: expression(tok, where), subject(move(exp)) {}
|
|
bool evaluate(io_streams_t *streams, wcstring_list_t &errors) override;
|
|
};
|
|
|
|
/// Combining expression. Contains a list of AND or OR expressions. It takes more than two so that
|
|
/// we don't have to worry about precedence in the parser.
|
|
class combining_expression final : public expression {
|
|
public:
|
|
const std::vector<unique_ptr<expression>> subjects;
|
|
const std::vector<token_t> combiners;
|
|
|
|
combining_expression(token_t tok, range_t where, std::vector<unique_ptr<expression>> exprs,
|
|
std::vector<token_t> combs)
|
|
: expression(tok, where), subjects(std::move(exprs)), combiners(std::move(combs)) {
|
|
// We should have one more subject than combiner.
|
|
assert(subjects.size() == combiners.size() + 1);
|
|
}
|
|
|
|
~combining_expression() override = default;
|
|
|
|
bool evaluate(io_streams_t *streams, wcstring_list_t &errors) override;
|
|
};
|
|
|
|
/// Parenthetical expression.
|
|
class parenthetical_expression final : public expression {
|
|
public:
|
|
unique_ptr<expression> contents;
|
|
parenthetical_expression(token_t tok, range_t where, unique_ptr<expression> expr)
|
|
: expression(tok, where), contents(move(expr)) {}
|
|
|
|
bool evaluate(io_streams_t *streams, wcstring_list_t &errors) override;
|
|
};
|
|
|
|
void test_parser::add_error(unsigned int idx, const wchar_t *fmt, ...) {
|
|
assert(fmt != nullptr);
|
|
va_list va;
|
|
va_start(va, fmt);
|
|
this->errors.push_back(vformat_string(fmt, va));
|
|
va_end(va);
|
|
if (this->errors.size() == 1) {
|
|
this->error_idx = idx;
|
|
}
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::error(unsigned int idx, const wchar_t *fmt, ...) {
|
|
assert(fmt != nullptr);
|
|
va_list va;
|
|
va_start(va, fmt);
|
|
this->errors.push_back(vformat_string(fmt, va));
|
|
va_end(va);
|
|
if (this->errors.size() == 1) {
|
|
this->error_idx = idx;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::parse_unary_expression(unsigned int start, unsigned int end) {
|
|
if (start >= end) {
|
|
return error(start, L"Missing argument at index %u", start + 1);
|
|
}
|
|
token_t tok = token_for_string(arg(start))->tok;
|
|
if (tok == test_bang) {
|
|
unique_ptr<expression> subject(parse_unary_expression(start + 1, end));
|
|
if (subject) {
|
|
return make_unique<unary_operator>(tok, range_t(start, subject->range.end),
|
|
move(subject));
|
|
}
|
|
return nullptr;
|
|
}
|
|
return parse_primary(start, end);
|
|
}
|
|
|
|
/// Parse a combining expression (AND, OR).
|
|
unique_ptr<expression> test_parser::parse_combining_expression(unsigned int start,
|
|
unsigned int end) {
|
|
if (start >= end) return nullptr;
|
|
|
|
std::vector<unique_ptr<expression>> subjects;
|
|
std::vector<token_t> combiners;
|
|
unsigned int idx = start;
|
|
bool first = true;
|
|
|
|
while (idx < end) {
|
|
if (!first) {
|
|
// This is not the first expression, so we expect a combiner.
|
|
token_t combiner = token_for_string(arg(idx))->tok;
|
|
if (combiner != test_combine_and && combiner != test_combine_or) {
|
|
/* Not a combiner, we're done */
|
|
this->errors.insert(
|
|
this->errors.begin(),
|
|
format_string(L"Expected a combining operator like '-a' at index %u", idx + 1));
|
|
error_idx = idx;
|
|
break;
|
|
}
|
|
combiners.push_back(combiner);
|
|
idx++;
|
|
}
|
|
|
|
// Parse another expression.
|
|
unique_ptr<expression> expr = parse_unary_expression(idx, end);
|
|
if (!expr) {
|
|
add_error(idx, L"Missing argument at index %u", idx + 1);
|
|
if (!first) {
|
|
// Clean up the dangling combiner, since it never got its right hand expression.
|
|
combiners.pop_back();
|
|
}
|
|
break;
|
|
}
|
|
|
|
// Go to the end of this expression.
|
|
idx = expr->range.end;
|
|
subjects.push_back(move(expr));
|
|
first = false;
|
|
}
|
|
|
|
if (subjects.empty()) {
|
|
return nullptr; // no subjects
|
|
}
|
|
// Our new expression takes ownership of all expressions we created. The token we pass is
|
|
// irrelevant.
|
|
return make_unique<combining_expression>(test_combine_and, range_t(start, idx), move(subjects),
|
|
move(combiners));
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::parse_unary_primary(unsigned int start, unsigned int end) {
|
|
// We need two arguments.
|
|
if (start >= end) {
|
|
return error(start, L"Missing argument at index %u", start + 1);
|
|
}
|
|
if (start + 1 >= end) {
|
|
return error(start + 1, L"Missing argument at index %u", start + 2);
|
|
}
|
|
|
|
// All our unary primaries are prefix, so the operator is at start.
|
|
const token_info_t *info = token_for_string(arg(start));
|
|
if (!(info->flags & UNARY_PRIMARY)) return nullptr;
|
|
|
|
return make_unique<unary_primary>(info->tok, range_t(start, start + 2), arg(start + 1));
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::parse_just_a_string(unsigned int start, unsigned int end) {
|
|
// Handle a string as a unary primary that is not a token of any other type. e.g. 'test foo -a
|
|
// bar' should evaluate to true We handle this with a unary primary of test_string_n.
|
|
|
|
// We need one argument.
|
|
if (start >= end) {
|
|
return error(start, L"Missing argument at index %u", start + 1);
|
|
}
|
|
|
|
const token_info_t *info = token_for_string(arg(start));
|
|
if (info->tok != test_unknown) {
|
|
return error(start, L"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
|
|
// type.
|
|
return make_unique<unary_primary>(test_string_n, range_t(start, start + 1), arg(start));
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::parse_binary_primary(unsigned int start, unsigned int end) {
|
|
// We need three arguments.
|
|
for (unsigned int idx = start; idx < start + 3; idx++) {
|
|
if (idx >= end) {
|
|
return error(idx, L"Missing argument at index %u", idx + 1);
|
|
}
|
|
}
|
|
|
|
// All our binary primaries are infix, so the operator is at start + 1.
|
|
const token_info_t *info = token_for_string(arg(start + 1));
|
|
if (!(info->flags & BINARY_PRIMARY)) return nullptr;
|
|
|
|
return make_unique<binary_primary>(info->tok, range_t(start, start + 3), arg(start),
|
|
arg(start + 2));
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::parse_parenthentical(unsigned int start, unsigned int end) {
|
|
// We need at least three arguments: open paren, argument, close paren.
|
|
if (start + 3 >= end) return nullptr;
|
|
|
|
// Must start with an open expression.
|
|
const token_info_t *open_paren = token_for_string(arg(start));
|
|
if (open_paren->tok != test_paren_open) return nullptr;
|
|
|
|
// Parse a subexpression.
|
|
unique_ptr<expression> subexpr = parse_expression(start + 1, end);
|
|
if (!subexpr) return nullptr;
|
|
|
|
// Parse a close paren.
|
|
unsigned close_index = subexpr->range.end;
|
|
assert(close_index <= end);
|
|
if (close_index == end) {
|
|
return error(close_index, L"Missing close paren at index %u", close_index + 1);
|
|
}
|
|
const token_info_t *close_paren = token_for_string(arg(close_index));
|
|
if (close_paren->tok != test_paren_close) {
|
|
return error(close_index, L"Expected close paren at index %u", close_index + 1);
|
|
}
|
|
|
|
// Success.
|
|
return make_unique<parenthetical_expression>(test_paren_open, range_t(start, close_index + 1),
|
|
move(subexpr));
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::parse_primary(unsigned int start, unsigned int end) {
|
|
if (start >= end) {
|
|
return error(start, L"Missing argument at index %u", start + 1);
|
|
}
|
|
|
|
unique_ptr<expression> expr = nullptr;
|
|
if (!expr) expr = parse_parenthentical(start, end);
|
|
if (!expr) expr = parse_unary_primary(start, end);
|
|
if (!expr) expr = parse_binary_primary(start, end);
|
|
if (!expr) expr = parse_just_a_string(start, end);
|
|
return expr;
|
|
}
|
|
|
|
// See IEEE 1003.1 breakdown of the behavior for different parameter counts.
|
|
unique_ptr<expression> test_parser::parse_3_arg_expression(unsigned int start, unsigned int end) {
|
|
assert(end - start == 3);
|
|
unique_ptr<expression> result = nullptr;
|
|
|
|
const token_info_t *center_token = token_for_string(arg(start + 1));
|
|
if (center_token->flags & BINARY_PRIMARY) {
|
|
result = parse_binary_primary(start, end);
|
|
} else if (center_token->tok == test_combine_and || center_token->tok == test_combine_or) {
|
|
unique_ptr<expression> left(parse_unary_expression(start, start + 1));
|
|
unique_ptr<expression> right(parse_unary_expression(start + 2, start + 3));
|
|
if (left.get() && right.get()) {
|
|
// Transfer ownership to the vector of subjects.
|
|
std::vector<token_t> combiners = {center_token->tok};
|
|
std::vector<unique_ptr<expression>> subjects;
|
|
subjects.push_back(move(left));
|
|
subjects.push_back(move(right));
|
|
result = make_unique<combining_expression>(center_token->tok, range_t(start, end),
|
|
move(subjects), move(combiners));
|
|
}
|
|
} else {
|
|
result = parse_unary_expression(start, end);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::parse_4_arg_expression(unsigned int start, unsigned int end) {
|
|
assert(end - start == 4);
|
|
unique_ptr<expression> result = nullptr;
|
|
|
|
token_t first_token = token_for_string(arg(start))->tok;
|
|
if (first_token == test_bang) {
|
|
unique_ptr<expression> subject(parse_3_arg_expression(start + 1, end));
|
|
if (subject) {
|
|
result = make_unique<unary_operator>(first_token, range_t(start, subject->range.end),
|
|
move(subject));
|
|
}
|
|
} else if (first_token == test_paren_open) {
|
|
result = parse_parenthentical(start, end);
|
|
} else {
|
|
result = parse_combining_expression(start, end);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::parse_expression(unsigned int start, unsigned int end) {
|
|
if (start >= end) {
|
|
return error(start, L"Missing argument at index %u", start + 1);
|
|
}
|
|
|
|
unsigned int argc = end - start;
|
|
switch (argc) {
|
|
case 0: {
|
|
DIE("argc should not be zero"); // should have been caught by the above test
|
|
}
|
|
case 1: {
|
|
return error(start + 1, L"Missing argument at index %u", start + 2);
|
|
}
|
|
case 2: {
|
|
return parse_unary_expression(start, end);
|
|
}
|
|
case 3: {
|
|
return parse_3_arg_expression(start, end);
|
|
}
|
|
case 4: {
|
|
return parse_4_arg_expression(start, end);
|
|
}
|
|
default: {
|
|
return parse_combining_expression(start, end);
|
|
}
|
|
}
|
|
}
|
|
|
|
unique_ptr<expression> test_parser::parse_args(const wcstring_list_t &args, wcstring &err,
|
|
wchar_t *program_name) {
|
|
// Empty list and one-arg list should be handled by caller.
|
|
assert(args.size() > 1);
|
|
|
|
test_parser parser(args);
|
|
unique_ptr<expression> result =
|
|
parser.parse_expression(0, static_cast<unsigned int>(args.size()));
|
|
|
|
// Handle errors.
|
|
// For now we only show the first error.
|
|
if (!parser.errors.empty()) {
|
|
int narg = 0;
|
|
int len_to_err = 0;
|
|
wcstring commandline;
|
|
for (const wcstring &arg : args) {
|
|
if (narg > 0) {
|
|
commandline.append(L" ");
|
|
}
|
|
commandline.append(arg);
|
|
narg++;
|
|
if (narg == parser.error_idx) {
|
|
len_to_err = fish_wcswidth(commandline.c_str(), commandline.length());
|
|
}
|
|
}
|
|
err.append(program_name);
|
|
err.append(L": ");
|
|
err.append(parser.errors.at(0));
|
|
err.push_back(L'\n');
|
|
err.append(commandline);
|
|
err.push_back(L'\n');
|
|
err.append(format_string(L"%*ls%ls\n", len_to_err + 1, L" ", L"^"));
|
|
}
|
|
|
|
if (result) {
|
|
// It's also an error if there are any unused arguments. This is not detected by
|
|
// parse_expression().
|
|
assert(result->range.end <= args.size());
|
|
if (result->range.end < args.size()) {
|
|
if (err.empty()) {
|
|
append_format(err, L"%ls: unexpected argument at index %lu: '%ls'\n", program_name,
|
|
static_cast<unsigned long>(result->range.end) + 1,
|
|
args.at(result->range.end).c_str());
|
|
}
|
|
result.reset(nullptr);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
bool unary_primary::evaluate(io_streams_t *streams, wcstring_list_t &errors) {
|
|
return unary_primary_evaluate(token, arg, streams, errors);
|
|
}
|
|
|
|
bool binary_primary::evaluate(io_streams_t *, wcstring_list_t &errors) {
|
|
return binary_primary_evaluate(token, arg_left, arg_right, errors);
|
|
}
|
|
|
|
bool unary_operator::evaluate(io_streams_t *streams, wcstring_list_t &errors) {
|
|
if (token == test_bang) {
|
|
assert(subject.get());
|
|
return !subject->evaluate(streams, errors);
|
|
}
|
|
|
|
errors.push_back(format_string(L"Unknown token type in %s", __func__));
|
|
return false;
|
|
}
|
|
|
|
bool combining_expression::evaluate(io_streams_t *streams, wcstring_list_t &errors) {
|
|
if (token == test_combine_and || token == test_combine_or) {
|
|
assert(!subjects.empty()); //!OCLINT(multiple unary operator)
|
|
assert(combiners.size() + 1 == subjects.size());
|
|
|
|
// One-element case.
|
|
if (subjects.size() == 1) return subjects.front()->evaluate(streams, errors);
|
|
|
|
// Evaluate our lists, remembering that AND has higher precedence than OR. We can
|
|
// visualize this as a sequence of OR expressions of AND expressions.
|
|
size_t idx = 0, max = subjects.size();
|
|
bool or_result = false;
|
|
while (idx < max) {
|
|
if (or_result) { // short circuit
|
|
break;
|
|
}
|
|
|
|
// Evaluate a stream of AND starting at given subject index. It may only have one
|
|
// element.
|
|
bool and_result = true;
|
|
for (; idx < max; idx++) {
|
|
// Evaluate it, short-circuiting.
|
|
and_result = and_result && subjects.at(idx)->evaluate(streams, errors);
|
|
|
|
// If the combiner at this index (which corresponding to how we combine with the
|
|
// next subject) is not AND, then exit the loop.
|
|
if (idx + 1 < max && combiners.at(idx) != test_combine_and) {
|
|
idx++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// OR it in.
|
|
or_result = or_result || and_result;
|
|
}
|
|
return or_result;
|
|
}
|
|
|
|
errors.push_back(format_string(L"Unknown token type in %s", __func__));
|
|
return false;
|
|
}
|
|
|
|
bool parenthetical_expression::evaluate(io_streams_t *streams, wcstring_list_t &errors) {
|
|
return contents->evaluate(streams, errors);
|
|
}
|
|
|
|
// Parse a double from arg. Return true on success, false on failure.
|
|
static bool parse_double(const wchar_t *arg, double *out_res) {
|
|
// Consume leading spaces.
|
|
while (arg && *arg != L'\0' && iswspace(*arg)) arg++;
|
|
if (!arg) return false;
|
|
errno = 0;
|
|
wchar_t *end = nullptr;
|
|
*out_res = fish_wcstod(arg, &end);
|
|
// Consume trailing spaces.
|
|
while (end && *end != L'\0' && iswspace(*end)) end++;
|
|
return errno == 0 && end > arg && *end == L'\0';
|
|
}
|
|
|
|
// IEEE 1003.1 says nothing about what it means for two strings to be "algebraically equal". For
|
|
// example, should we interpret 0x10 as 0, 10, or 16? Here we use only base 10 and use wcstoll,
|
|
// which allows for leading + and -, and whitespace. This is consistent, albeit a bit more lenient
|
|
// since we allow trailing whitespace, with other implementations such as bash.
|
|
static bool parse_number(const wcstring &arg, number_t *number, wcstring_list_t &errors) {
|
|
const wchar_t *argcs = arg.c_str();
|
|
double floating = 0;
|
|
bool got_float = parse_double(argcs, &floating);
|
|
errno = 0;
|
|
long long integral = fish_wcstoll(argcs);
|
|
bool got_int = (errno == 0);
|
|
if (got_int) {
|
|
// Here the value is just an integer; ignore the floating point parse because it may be
|
|
// invalid (e.g. not a representable integer).
|
|
*number = number_t{integral, 0.0};
|
|
|
|
return true;
|
|
} else if (got_float && errno != ERANGE && std::isfinite(floating)) {
|
|
// Here we parsed an (in range) floating point value that could not be parsed as an integer.
|
|
// Break the floating point value into base and delta. Ensure that base is <= the floating
|
|
// point value.
|
|
//
|
|
// Note that a non-finite number like infinity or NaN doesn't work for us, so we checked
|
|
// above.
|
|
double intpart = std::floor(floating);
|
|
double delta = floating - intpart;
|
|
*number = number_t{static_cast<long long>(intpart), delta};
|
|
|
|
return true;
|
|
} else {
|
|
// We could not parse a float or an int.
|
|
// Check for special fish_wcsto* value or show standard EINVAL/ERANGE error.
|
|
if (errno == -1) {
|
|
errors.push_back(
|
|
format_string(_(L"Integer %lld in '%ls' followed by non-digit"), integral, argcs));
|
|
} else if (std::isnan(floating)) {
|
|
// NaN is an error as far as we're concerned.
|
|
errors.push_back(_(L"Not a number"));
|
|
} else if (std::isinf(floating)) {
|
|
errors.push_back(_(L"Number is infinite"));
|
|
} else {
|
|
errors.push_back(format_string(L"%s: '%ls'", std::strerror(errno), argcs));
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
static bool binary_primary_evaluate(test_expressions::token_t token, const wcstring &left,
|
|
const wcstring &right, wcstring_list_t &errors) {
|
|
using namespace test_expressions;
|
|
number_t ln, rn;
|
|
switch (token) {
|
|
case test_string_equal: {
|
|
return left == right;
|
|
}
|
|
case test_string_not_equal: {
|
|
return left != right;
|
|
}
|
|
case test_number_equal: {
|
|
return parse_number(left, &ln, errors) && parse_number(right, &rn, errors) &&
|
|
ln.compare(rn) == 0;
|
|
}
|
|
case test_number_not_equal: {
|
|
return parse_number(left, &ln, errors) && parse_number(right, &rn, errors) &&
|
|
ln.compare(rn) != 0;
|
|
}
|
|
case test_number_greater: {
|
|
return parse_number(left, &ln, errors) && parse_number(right, &rn, errors) &&
|
|
ln.compare(rn) > 0;
|
|
}
|
|
case test_number_greater_equal: {
|
|
return parse_number(left, &ln, errors) && parse_number(right, &rn, errors) &&
|
|
ln.compare(rn) >= 0;
|
|
}
|
|
case test_number_lesser: {
|
|
return parse_number(left, &ln, errors) && parse_number(right, &rn, errors) &&
|
|
ln.compare(rn) < 0;
|
|
}
|
|
case test_number_lesser_equal: {
|
|
return parse_number(left, &ln, errors) && parse_number(right, &rn, errors) &&
|
|
ln.compare(rn) <= 0;
|
|
}
|
|
default: {
|
|
errors.push_back(format_string(L"Unknown token type in %s", __func__));
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool unary_primary_evaluate(test_expressions::token_t token, const wcstring &arg,
|
|
io_streams_t *streams, wcstring_list_t &errors) {
|
|
using namespace test_expressions;
|
|
struct stat buf;
|
|
switch (token) {
|
|
case test_filetype_b: { // "-b", for block special files
|
|
return !wstat(arg, &buf) && S_ISBLK(buf.st_mode);
|
|
}
|
|
case test_filetype_c: { // "-c", for character special files
|
|
return !wstat(arg, &buf) && S_ISCHR(buf.st_mode);
|
|
}
|
|
case test_filetype_d: { // "-d", for directories
|
|
return !wstat(arg, &buf) && S_ISDIR(buf.st_mode);
|
|
}
|
|
case test_filetype_e: { // "-e", for files that exist
|
|
return !wstat(arg, &buf);
|
|
}
|
|
case test_filetype_f: { // "-f", for for regular files
|
|
return !wstat(arg, &buf) && S_ISREG(buf.st_mode);
|
|
}
|
|
case test_filetype_G: { // "-G", for check effective group id
|
|
return !wstat(arg, &buf) && getegid() == buf.st_gid;
|
|
}
|
|
case test_filetype_g: { // "-g", for set-group-id
|
|
return !wstat(arg, &buf) && (S_ISGID & buf.st_mode);
|
|
}
|
|
case test_filetype_h: // "-h", for symbolic links
|
|
case test_filetype_L: { // "-L", same as -h
|
|
return !lwstat(arg, &buf) && S_ISLNK(buf.st_mode);
|
|
}
|
|
case test_filetype_k: { // "-k", for sticky bit
|
|
#ifdef S_ISVTX
|
|
return !lwstat(arg, &buf) && buf.st_mode & S_ISVTX;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
case test_filetype_O: { // "-O", for check effective user id
|
|
return !wstat(arg, &buf) && geteuid() == buf.st_uid;
|
|
}
|
|
case test_filetype_p: { // "-p", for FIFO
|
|
return !wstat(arg, &buf) && S_ISFIFO(buf.st_mode);
|
|
}
|
|
case test_filetype_S: { // "-S", socket
|
|
return !wstat(arg, &buf) && S_ISSOCK(buf.st_mode);
|
|
}
|
|
case test_filesize_s: { // "-s", size greater than zero
|
|
return !wstat(arg, &buf) && buf.st_size > 0;
|
|
}
|
|
case test_filedesc_t: { // "-t", whether the fd is associated with a terminal
|
|
number_t num;
|
|
return parse_number(arg, &num, errors) && num.isatty(streams);
|
|
}
|
|
case test_fileperm_r: { // "-r", read permission
|
|
return !waccess(arg, R_OK);
|
|
}
|
|
case test_fileperm_u: { // "-u", whether file is setuid
|
|
return !wstat(arg, &buf) && (S_ISUID & buf.st_mode);
|
|
}
|
|
case test_fileperm_w: { // "-w", whether file write permission is allowed
|
|
return !waccess(arg, W_OK);
|
|
}
|
|
case test_fileperm_x: { // "-x", whether file execute/search is allowed
|
|
return !waccess(arg, X_OK);
|
|
}
|
|
case test_string_n: { // "-n", non-empty string
|
|
return !arg.empty();
|
|
}
|
|
case test_string_z: { // "-z", true if length of string is 0
|
|
return arg.empty();
|
|
}
|
|
default: {
|
|
errors.push_back(format_string(L"Unknown token type in %s", __func__));
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}; // namespace test_expressions
|
|
}; // anonymous namespace
|
|
|
|
/// Evaluate a conditional expression given the arguments. If fromtest is set, the caller is the
|
|
/// test or [ builtin; with the pointer giving the name of the command. for POSIX conformance this
|
|
/// supports a more limited range of functionality.
|
|
///
|
|
/// Return status is the final shell status, i.e. 0 for true, 1 for false and 2 for error.
|
|
maybe_t<int> builtin_test(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
|
UNUSED(parser);
|
|
using namespace test_expressions;
|
|
|
|
// The first argument should be the name of the command ('test').
|
|
if (!argv[0]) return STATUS_INVALID_ARGS;
|
|
|
|
// Whether we are invoked with bracket '[' or not.
|
|
wchar_t *program_name = argv[0];
|
|
const bool is_bracket = !std::wcscmp(program_name, L"[");
|
|
|
|
size_t argc = 0;
|
|
while (argv[argc + 1]) argc++;
|
|
|
|
// If we're bracket, the last argument ought to be ]; we ignore it. Note that argc is the number
|
|
// of arguments after the command name; thus argv[argc] is the last argument.
|
|
if (is_bracket) {
|
|
if (!std::wcscmp(argv[argc], L"]")) {
|
|
// Ignore the closing bracket from now on.
|
|
argc--;
|
|
} else {
|
|
streams.err.append(L"[: the last argument must be ']'\n");
|
|
builtin_print_error_trailer(parser, streams.err, program_name);
|
|
return STATUS_INVALID_ARGS;
|
|
}
|
|
}
|
|
|
|
// Collect the arguments into a list.
|
|
const wcstring_list_t args(argv + 1, argv + 1 + argc);
|
|
|
|
if (argc == 0) {
|
|
return STATUS_INVALID_ARGS; // Per 1003.1, exit false.
|
|
} else if (argc == 1) {
|
|
// Per 1003.1, exit true if the arg is non-empty.
|
|
return args.at(0).empty() ? STATUS_CMD_ERROR : STATUS_CMD_OK;
|
|
}
|
|
|
|
// Try parsing
|
|
wcstring err;
|
|
unique_ptr<expression> expr = test_parser::parse_args(args, err, program_name);
|
|
if (!expr) {
|
|
streams.err.append(err);
|
|
streams.err.append(parser.current_line());
|
|
return STATUS_CMD_ERROR;
|
|
}
|
|
|
|
wcstring_list_t eval_errors;
|
|
bool result = expr->evaluate(&streams, eval_errors);
|
|
if (!eval_errors.empty()) {
|
|
if (!should_suppress_stderr_for_tests()) {
|
|
for (const auto &eval_error : eval_errors) {
|
|
streams.err.append_format(L"%ls\n", eval_error.c_str());
|
|
}
|
|
// Add a backtrace but not the "see help" message
|
|
// because this isn't about passing the wrong options.
|
|
streams.err.append(parser.current_line());
|
|
}
|
|
return STATUS_INVALID_ARGS;
|
|
}
|
|
return result ? STATUS_CMD_OK : STATUS_CMD_ERROR;
|
|
}
|