mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-27 20:25:12 +00:00
Teach builtin_test about the special precedence rules in IEEE 1003.1
https://github.com/fish-shell/fish-shell/issues/601
This commit is contained in:
parent
cbcc439e03
commit
1d75c232ff
2 changed files with 139 additions and 42 deletions
177
builtin_test.cpp
177
builtin_test.cpp
|
@ -177,6 +177,8 @@ public:
|
||||||
{ }
|
{ }
|
||||||
|
|
||||||
expression *parse_expression(unsigned int start, unsigned int end);
|
expression *parse_expression(unsigned int start, unsigned int end);
|
||||||
|
expression *parse_3_arg_expression(unsigned int start, unsigned int end);
|
||||||
|
expression *parse_4_arg_expression(unsigned int start, unsigned int end);
|
||||||
expression *parse_combining_expression(unsigned int start, unsigned int end);
|
expression *parse_combining_expression(unsigned int start, unsigned int end);
|
||||||
expression *parse_unary_expression(unsigned int start, unsigned int end);
|
expression *parse_unary_expression(unsigned int start, unsigned int end);
|
||||||
|
|
||||||
|
@ -539,14 +541,102 @@ expression *test_parser::parse_primary(unsigned int start, unsigned int end)
|
||||||
return expr;
|
return expr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// See IEEE 1003.1 breakdown of the behavior for different parameter counts
|
||||||
|
expression *test_parser::parse_3_arg_expression(unsigned int start, unsigned int end)
|
||||||
|
{
|
||||||
|
assert(end - start == 3);
|
||||||
|
expression *result = NULL;
|
||||||
|
|
||||||
|
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)
|
||||||
|
{
|
||||||
|
expr_ref_t left(parse_unary_expression(start, start + 1));
|
||||||
|
expr_ref_t 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(1, center_token->tok);
|
||||||
|
std::vector<expression *> subjects;
|
||||||
|
subjects.push_back(left.release());
|
||||||
|
subjects.push_back(right.release());
|
||||||
|
result = new combining_expression(center_token->tok, range_t(start, end), subjects, combiners);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
result = parse_unary_expression(start, end);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
expression *test_parser::parse_4_arg_expression(unsigned int start, unsigned int end)
|
||||||
|
{
|
||||||
|
assert(end - start == 4);
|
||||||
|
expression *result = NULL;
|
||||||
|
|
||||||
|
token_t first_token = token_for_string(arg(start))->tok;
|
||||||
|
if (first_token == test_bang)
|
||||||
|
{
|
||||||
|
expr_ref_t subject(parse_3_arg_expression(start + 1, end));
|
||||||
|
if (subject.get())
|
||||||
|
{
|
||||||
|
result = new unary_operator(first_token, range_t(start, subject->range.end), subject);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (first_token == test_paren_open)
|
||||||
|
{
|
||||||
|
result = parse_parenthentical(start, end);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
result = parse_combining_expression(start, end);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
expression *test_parser::parse_expression(unsigned int start, unsigned int end)
|
expression *test_parser::parse_expression(unsigned int start, unsigned int end)
|
||||||
{
|
{
|
||||||
if (start >= end)
|
if (start >= end)
|
||||||
{
|
{
|
||||||
return error(L"Missing argument at index %u", start);
|
return error(L"Missing argument at index %u", start);
|
||||||
}
|
}
|
||||||
|
|
||||||
return parse_combining_expression(start, end);
|
unsigned int argc = end - start;
|
||||||
|
switch (argc)
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
assert(0); //should have been caught by the above test
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
{
|
||||||
|
return error(L"Missing argument at index %u", start + 1);
|
||||||
|
}
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
expression *test_parser::parse_args(const wcstring_list_t &args, wcstring &err)
|
expression *test_parser::parse_args(const wcstring_list_t &args, wcstring &err)
|
||||||
|
@ -798,18 +888,18 @@ static bool unary_primary_evaluate(test_expressions::token_t token, const wcstri
|
||||||
int builtin_test(parser_t &parser, wchar_t **argv)
|
int builtin_test(parser_t &parser, wchar_t **argv)
|
||||||
{
|
{
|
||||||
using namespace test_expressions;
|
using namespace test_expressions;
|
||||||
|
|
||||||
/* The first argument should be the name of the command ('test') */
|
/* The first argument should be the name of the command ('test') */
|
||||||
if (! argv[0])
|
if (! argv[0])
|
||||||
return BUILTIN_TEST_FAIL;
|
return BUILTIN_TEST_FAIL;
|
||||||
|
|
||||||
/* Whether we are invoked with bracket '[' or not */
|
/* Whether we are invoked with bracket '[' or not */
|
||||||
const bool is_bracket = ! wcscmp(argv[0], L"[");
|
const bool is_bracket = ! wcscmp(argv[0], L"[");
|
||||||
|
|
||||||
size_t argc = 0;
|
size_t argc = 0;
|
||||||
while (argv[argc + 1])
|
while (argv[argc + 1])
|
||||||
argc++;
|
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 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 (is_bracket)
|
||||||
{
|
{
|
||||||
|
@ -823,54 +913,57 @@ int builtin_test(parser_t &parser, wchar_t **argv)
|
||||||
builtin_show_error(L"[: the last argument must be ']'\n");
|
builtin_show_error(L"[: the last argument must be ']'\n");
|
||||||
return BUILTIN_TEST_FAIL;
|
return BUILTIN_TEST_FAIL;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Collect the arguments into a list */
|
/* Collect the arguments into a list */
|
||||||
const wcstring_list_t args(argv + 1, argv + 1 + argc);
|
const wcstring_list_t args(argv + 1, argv + 1 + argc);
|
||||||
|
|
||||||
if (argc == 0)
|
switch (argc)
|
||||||
{
|
{
|
||||||
// Per 1003.1, exit false
|
case 0:
|
||||||
return BUILTIN_TEST_FAIL;
|
|
||||||
}
|
|
||||||
else if (argc == 1)
|
|
||||||
{
|
|
||||||
// Per 1003.1, exit true if the arg is non-empty
|
|
||||||
return args.at(0).empty() ? BUILTIN_TEST_FAIL : BUILTIN_TEST_SUCCESS;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Try parsing. If expr is not nil, we are responsible for deleting it.
|
|
||||||
wcstring err;
|
|
||||||
expression *expr = test_parser::parse_args(args, err);
|
|
||||||
if (! expr)
|
|
||||||
{
|
{
|
||||||
#if 0
|
// Per 1003.1, exit false
|
||||||
printf("Oops! test was given args:\n");
|
|
||||||
for (size_t i=0; i < argc; i++)
|
|
||||||
{
|
|
||||||
printf("\t%ls\n", args.at(i).c_str());
|
|
||||||
}
|
|
||||||
printf("and returned parse error: %ls\n", err.c_str());
|
|
||||||
#endif
|
|
||||||
builtin_show_error(err);
|
|
||||||
return BUILTIN_TEST_FAIL;
|
return BUILTIN_TEST_FAIL;
|
||||||
}
|
}
|
||||||
else
|
case 1:
|
||||||
{
|
{
|
||||||
wcstring_list_t eval_errors;
|
// Per 1003.1, exit true if the arg is non-empty
|
||||||
bool result = expr->evaluate(eval_errors);
|
return args.at(0).empty() ? BUILTIN_TEST_FAIL : BUILTIN_TEST_SUCCESS;
|
||||||
if (! eval_errors.empty())
|
}
|
||||||
|
default:
|
||||||
|
{
|
||||||
|
// Try parsing. If expr is not nil, we are responsible for deleting it.
|
||||||
|
wcstring err;
|
||||||
|
expression *expr = test_parser::parse_args(args, err);
|
||||||
|
if (! expr)
|
||||||
{
|
{
|
||||||
printf("test returned eval errors:\n");
|
#if 0
|
||||||
for (size_t i=0; i < eval_errors.size(); i++)
|
printf("Oops! test was given args:\n");
|
||||||
|
for (size_t i=0; i < argc; i++)
|
||||||
{
|
{
|
||||||
printf("\t%ls\n", eval_errors.at(i).c_str());
|
printf("\t%ls\n", args.at(i).c_str());
|
||||||
}
|
}
|
||||||
|
printf("and returned parse error: %ls\n", err.c_str());
|
||||||
|
#endif
|
||||||
|
builtin_show_error(err);
|
||||||
|
return BUILTIN_TEST_FAIL;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
wcstring_list_t eval_errors;
|
||||||
|
bool result = expr->evaluate(eval_errors);
|
||||||
|
if (! eval_errors.empty())
|
||||||
|
{
|
||||||
|
printf("test returned eval errors:\n");
|
||||||
|
for (size_t i=0; i < eval_errors.size(); i++)
|
||||||
|
{
|
||||||
|
printf("\t%ls\n", eval_errors.at(i).c_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delete expr;
|
||||||
|
return result ? BUILTIN_TEST_SUCCESS : BUILTIN_TEST_FAIL;
|
||||||
}
|
}
|
||||||
delete expr;
|
|
||||||
return result ? BUILTIN_TEST_SUCCESS : BUILTIN_TEST_FAIL;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 1;
|
return 1;
|
||||||
|
|
|
@ -936,6 +936,10 @@ static void test_test()
|
||||||
|
|
||||||
/* This crashed */
|
/* This crashed */
|
||||||
assert(run_test_test(1, L"1 = 1 -a = 1"));
|
assert(run_test_test(1, L"1 = 1 -a = 1"));
|
||||||
|
|
||||||
|
/* Make sure we can treat -S as a parameter instead of an operator. https://github.com/fish-shell/fish-shell/issues/601 */
|
||||||
|
assert(run_test_test(0, L"-S = -S"));
|
||||||
|
assert(run_test_test(1, L"! ! ! A"));
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Testing colors */
|
/** Testing colors */
|
||||||
|
|
Loading…
Reference in a new issue