Report errors for arguments to 'end'

For example, `begin ; end arg` will now report an error.

Fixes #986
This commit is contained in:
ridiculousfish 2018-01-22 13:31:39 -08:00
parent 9d48c68f24
commit a39c57c1b6
3 changed files with 40 additions and 4 deletions

View file

@ -31,6 +31,7 @@ This section is for changes merged to the `major` branch that are not also merge
- `cd` tab completions no longer descend into the deepest unambiguous path (#4649) - `cd` tab completions no longer descend into the deepest unambiguous path (#4649)
- Setting `$PATH` no longer warns on non-existent directories, allowing for a single $PATH to be shared across machines (e.g. via dotfiles). - Setting `$PATH` no longer warns on non-existent directories, allowing for a single $PATH to be shared across machines (e.g. via dotfiles).
- `funced` now has a `-s` and `--save` option to automatically save the edited function after successfully editing (#4668). - `funced` now has a `-s` and `--save` option to automatically save the edited function after successfully editing (#4668).
- Arguments to `end` are now errors, instead of being silently ignored.
## Other significant changes ## Other significant changes
- Command substitution output is now limited to 10 MB by default (#3822). - Command substitution output is now limited to 10 MB by default (#3822).

View file

@ -675,6 +675,26 @@ static void test_parser() {
err(L"'exec' command in pipeline not reported as error"); err(L"'exec' command in pipeline not reported as error");
} }
if (!parse_util_detect_errors(L"begin ; end arg")) {
err(L"argument to 'end' not reported as error");
}
if (!parse_util_detect_errors(L"switch foo ; end arg")) {
err(L"argument to 'end' not reported as error");
}
if (!parse_util_detect_errors(L"if true; else if false ; end arg")) {
err(L"argument to 'end' not reported as error");
}
if (!parse_util_detect_errors(L"if true; else ; end arg")) {
err(L"argument to 'end' not reported as error");
}
if (parse_util_detect_errors(L"begin ; end 2> /dev/null")) {
err(L"redirection after 'end' wrongly reported as error");
}
if (detect_argument_errors(L"foo")) { if (detect_argument_errors(L"foo")) {
err(L"simple argument reported as error"); err(L"simple argument reported as error");
} }

View file

@ -35,6 +35,9 @@
#define BACKGROUND_IN_CONDITIONAL_ERROR_MSG \ #define BACKGROUND_IN_CONDITIONAL_ERROR_MSG \
_(L"Backgrounded commands can not be used as conditionals") _(L"Backgrounded commands can not be used as conditionals")
/// Error message for arguments to 'end'
#define END_ARG_ERR_MSG _(L"'end' does not take arguments. Did you forget a ';'?")
int parse_util_lineno(const wchar_t *str, size_t offset) { int parse_util_lineno(const wchar_t *str, size_t offset) {
if (!str) return 0; if (!str) return 0;
@ -1082,6 +1085,7 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src,
parse_error_list_t *out_errors, parse_error_list_t *out_errors,
bool allow_incomplete, bool allow_incomplete,
parsed_source_ref_t *out_pstree) { parsed_source_ref_t *out_pstree) {
namespace g = grammar;
parse_node_tree_t node_tree; parse_node_tree_t node_tree;
parse_error_list_t parse_errors; parse_error_list_t parse_errors;
@ -1141,16 +1145,16 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src,
has_unclosed_block = true; has_unclosed_block = true;
} else if (node.type == symbol_boolean_statement) { } else if (node.type == symbol_boolean_statement) {
// 'or' and 'and' can be in a pipeline, as long as they're first. // 'or' and 'and' can be in a pipeline, as long as they're first.
tnode_t<grammar::boolean_statement> gbs{&node_tree, &node}; tnode_t<g::boolean_statement> gbs{&node_tree, &node};
parse_bool_statement_type_t type = bool_statement_type(gbs); parse_bool_statement_type_t type = bool_statement_type(gbs);
if ((type == parse_bool_and || type == parse_bool_or) && if ((type == parse_bool_and || type == parse_bool_or) &&
statement_is_in_pipeline(gbs.try_get_parent<grammar::statement>(), statement_is_in_pipeline(gbs.try_get_parent<g::statement>(),
false /* don't count first */)) { false /* don't count first */)) {
errored = append_syntax_error(&parse_errors, node.source_start, EXEC_ERR_MSG, errored = append_syntax_error(&parse_errors, node.source_start, EXEC_ERR_MSG,
(type == parse_bool_and) ? L"and" : L"or"); (type == parse_bool_and) ? L"and" : L"or");
} }
} else if (node.type == symbol_argument) { } else if (node.type == symbol_argument) {
tnode_t<grammar::argument> arg{&node_tree, &node}; tnode_t<g::argument> arg{&node_tree, &node};
const wcstring arg_src = node.get_source(buff_src); const wcstring arg_src = node.get_source(buff_src);
res |= parse_util_detect_errors_in_argument(arg, arg_src, &parse_errors); res |= parse_util_detect_errors_in_argument(arg, arg_src, &parse_errors);
} else if (node.type == symbol_job) { } else if (node.type == symbol_job) {
@ -1161,10 +1165,21 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src,
// if foo & ; end // if foo & ; end
// while foo & ; end // while foo & ; end
// If it's not a background job, nothing to do. // If it's not a background job, nothing to do.
auto job = tnode_t<grammar::job>{&node_tree, &node}; auto job = tnode_t<g::job>{&node_tree, &node};
if (job_node_is_background(job)) { if (job_node_is_background(job)) {
errored |= detect_errors_in_backgrounded_job(job, &parse_errors); errored |= detect_errors_in_backgrounded_job(job, &parse_errors);
} }
} else if (node.type == symbol_arguments_or_redirections_list) {
// verify no arguments to the end command of if, switch, begin (#986).
auto list = tnode_t<g::arguments_or_redirections_list>{&node_tree, &node};
if (list.try_get_parent<g::if_statement>() ||
list.try_get_parent<g::switch_statement>() ||
list.try_get_parent<g::block_statement>()) {
if (auto arg = list.next_in_list<g::argument>()) {
errored = append_syntax_error(&parse_errors, arg.source_range()->start,
END_ARG_ERR_MSG);
}
}
} else if (node.type == symbol_plain_statement) { } else if (node.type == symbol_plain_statement) {
using namespace grammar; using namespace grammar;
tnode_t<plain_statement> pst{&node_tree, &node}; tnode_t<plain_statement> pst{&node_tree, &node};