mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-14 05:53:59 +00:00
Report errors for arguments to 'end'
For example, `begin ; end arg` will now report an error. Fixes #986
This commit is contained in:
parent
9d48c68f24
commit
a39c57c1b6
3 changed files with 40 additions and 4 deletions
|
@ -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).
|
||||||
|
|
|
@ -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");
|
||||||
}
|
}
|
||||||
|
|
|
@ -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};
|
||||||
|
|
Loading…
Reference in a new issue