From a5dd96558f906e95e7fe6de30c2358333e254dc4 Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Thu, 1 Mar 2018 11:45:41 -0800 Subject: [PATCH 01/10] Remove special && and || error detection Soon these will no longer be errors. --- src/fish_tests.cpp | 7 +------ src/parse_constants.h | 11 +---------- src/parse_tree.cpp | 33 ++------------------------------- 3 files changed, 4 insertions(+), 47 deletions(-) diff --git a/src/fish_tests.cpp b/src/fish_tests.cpp index 29a01ba5f..3f0cb0b8f 100644 --- a/src/fish_tests.cpp +++ b/src/fish_tests.cpp @@ -3624,9 +3624,6 @@ static void test_new_parser_errors() { {L"case", parse_error_unbalancing_case}, {L"if true ; case ; end", parse_error_unbalancing_case}, - - {L"foo || bar", parse_error_double_pipe}, - {L"foo && bar", parse_error_double_background}, }; for (size_t i = 0; i < sizeof tests / sizeof *tests; i++) { @@ -3742,9 +3739,7 @@ static void test_error_messages() { {L"echo \"foo\"$\"bar\"", ERROR_NO_VAR_NAME}, {L"echo foo $ bar", ERROR_NO_VAR_NAME}, {L"echo foo$(foo)bar", ERROR_BAD_VAR_SUBCOMMAND1}, - {L"echo \"foo$(foo)bar\"", ERROR_BAD_VAR_SUBCOMMAND1}, - {L"echo foo || echo bar", ERROR_BAD_OR}, - {L"echo foo && echo bar", ERROR_BAD_AND}}; + {L"echo \"foo$(foo)bar\"", ERROR_BAD_VAR_SUBCOMMAND1}}; parse_error_list_t errors; for (size_t i = 0; i < sizeof error_tests / sizeof *error_tests; i++) { diff --git a/src/parse_constants.h b/src/parse_constants.h index b8e43fe0c..044f549e0 100644 --- a/src/parse_constants.h +++ b/src/parse_constants.h @@ -183,10 +183,7 @@ enum parse_error_code_t { parse_error_unbalancing_end, // end outside of block parse_error_unbalancing_else, // else outside of if - parse_error_unbalancing_case, // case outside of switch - - parse_error_double_pipe, // foo || bar, has special error message - parse_error_double_background // foo && bar, has special error message + parse_error_unbalancing_case // case outside of switch }; enum { PARSER_TEST_ERROR = 1, PARSER_TEST_INCOMPLETE = 2 }; @@ -289,12 +286,6 @@ void parse_error_offset_source_start(parse_error_list_t *errors, size_t amt); /// Error issued on $. #define ERROR_NO_VAR_NAME _(L"Expected a variable name after this $.") -/// Error on ||. -#define ERROR_BAD_OR _(L"Unsupported use of '||'. In fish, please use 'COMMAND; or COMMAND'.") - -/// Error on &&. -#define ERROR_BAD_AND _(L"Unsupported use of '&&'. In fish, please use 'COMMAND; and COMMAND'.") - /// Error on foo=bar. #define ERROR_BAD_EQUALS_IN_COMMAND5 \ _(L"Unsupported use of '='. To run '%ls' with a modified environment, please use 'env " \ diff --git a/src/parse_tree.cpp b/src/parse_tree.cpp index a284b346d..48094446b 100644 --- a/src/parse_tree.cpp +++ b/src/parse_tree.cpp @@ -674,37 +674,8 @@ void parse_ll_t::parse_error_failed_production(struct parse_stack_element_t &sta parse_token_t token) { fatal_errored = true; if (this->should_generate_error_messages) { - bool done = false; - - // Check for ||. - if (token.type == parse_token_type_pipe && token.source_start > 0) { - // Here we wanted a statement and instead got a pipe. See if this is a double pipe: foo - // || bar. If so, we have a special error message. - const parse_node_t *prev_pipe = nodes.find_node_matching_source_location( - parse_token_type_pipe, token.source_start - 1, NULL); - if (prev_pipe != NULL) { - // The pipe of the previous job abuts our current token. So we have ||. - this->parse_error(token, parse_error_double_pipe, ERROR_BAD_OR); - done = true; - } - } - - // Check for &&. - if (!done && token.type == parse_token_type_background && token.source_start > 0) { - // Check to see if there was a previous token_background. - const parse_node_t *prev_background = nodes.find_node_matching_source_location( - parse_token_type_background, token.source_start - 1, NULL); - if (prev_background != NULL) { - // We have &&. - this->parse_error(token, parse_error_double_background, ERROR_BAD_AND); - done = true; - } - } - - if (!done) { - const wcstring expected = stack_elem.user_presentable_description(); - this->parse_error_unexpected_token(expected.c_str(), token); - } + const wcstring expected = stack_elem.user_presentable_description(); + this->parse_error_unexpected_token(expected.c_str(), token); } } From 8ded04135257e8fd22dda7b46a793cbb9972503c Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Thu, 1 Mar 2018 12:56:15 -0800 Subject: [PATCH 02/10] Add && and || support to tokenizer --- src/fish_tests.cpp | 10 +++++--- src/history.cpp | 4 ++++ src/parse_tree.cpp | 59 +++++++++++++++++++--------------------------- src/tokenizer.cpp | 26 ++++++++++++++------ src/tokenizer.h | 2 ++ 5 files changed, 56 insertions(+), 45 deletions(-) diff --git a/src/fish_tests.cpp b/src/fish_tests.cpp index 3f0cb0b8f..6cb3c24dd 100644 --- a/src/fish_tests.cpp +++ b/src/fish_tests.cpp @@ -535,10 +535,14 @@ static void test_tokenizer() { const wchar_t *str = L"string &1 'nested \"quoted\" '(string containing subshells " L"){and,brackets}$as[$well (as variable arrays)] not_a_redirect^ ^ ^^is_a_redirect " + L"&&& ||| " + L"&& || & |" L"Compress_Newlines\n \n\t\n \nInto_Just_One"; - const int types[] = {TOK_STRING, TOK_REDIRECT, TOK_STRING, TOK_REDIRECT, TOK_STRING, - TOK_STRING, TOK_STRING, TOK_REDIRECT, TOK_REDIRECT, TOK_STRING, - TOK_STRING, TOK_END, TOK_STRING}; + const int types[] = {TOK_STRING, TOK_REDIRECT, TOK_STRING, TOK_REDIRECT, TOK_STRING, + TOK_STRING, TOK_STRING, TOK_REDIRECT, TOK_REDIRECT, TOK_STRING, + TOK_ANDAND, TOK_BACKGROUND, TOK_OROR, TOK_PIPE, TOK_ANDAND, + TOK_OROR, TOK_BACKGROUND, TOK_PIPE, TOK_STRING, TOK_END, + TOK_STRING}; say(L"Test correct tokenization"); diff --git a/src/history.cpp b/src/history.cpp index be0066a42..d2a58f676 100644 --- a/src/history.cpp +++ b/src/history.cpp @@ -1773,6 +1773,10 @@ static bool should_import_bash_history_line(const std::string &line) { if (line.find("((") != std::string::npos) return false; if (line.find("))") != std::string::npos) return false; + // Temporarily skip lines with && and || + if (line.find("&&") != std::string::npos) return false; + if (line.find("||") != std::string::npos) return false; + // Skip lines that end with a backslash. We do not handle multiline commands from bash history. if (line.back() == '\\') return false; diff --git a/src/parse_tree.cpp b/src/parse_tree.cpp index 48094446b..f99fc7951 100644 --- a/src/parse_tree.cpp +++ b/src/parse_tree.cpp @@ -213,43 +213,32 @@ wcstring parse_token_t::user_presentable_description() const { /// Convert from tokenizer_t's token type to a parse_token_t type. static inline parse_token_type_t parse_token_type_from_tokenizer_token( enum token_type tokenizer_token_type) { - parse_token_type_t result = token_type_invalid; switch (tokenizer_token_type) { - case TOK_STRING: { - result = parse_token_type_string; - break; - } - case TOK_PIPE: { - result = parse_token_type_pipe; - break; - } - case TOK_END: { - result = parse_token_type_end; - break; - } - case TOK_BACKGROUND: { - result = parse_token_type_background; - break; - } - case TOK_REDIRECT: { - result = parse_token_type_redirection; - break; - } - case TOK_ERROR: { - result = parse_special_type_tokenizer_error; - break; - } - case TOK_COMMENT: { - result = parse_special_type_comment; - break; - } - default: { - debug(0, "Bad token type %d passed to %s", (int)tokenizer_token_type, __FUNCTION__); - DIE("bad token type"); - break; - } + case TOK_NONE: + DIE("TOK_NONE passed to parse_token_type_from_tokenizer_token"); + return token_type_invalid; + case TOK_STRING: + return parse_token_type_string; + case TOK_PIPE: + return parse_token_type_pipe; + case TOK_ANDAND: + case TOK_OROR: + // Temporary while && and || support is brought up. + return parse_special_type_comment; + case TOK_END: + return parse_token_type_end; + case TOK_BACKGROUND: + return parse_token_type_background; + case TOK_REDIRECT: + return parse_token_type_redirection; + case TOK_ERROR: + return parse_special_type_tokenizer_error; + case TOK_COMMENT: + return parse_special_type_comment; } - return result; + debug(0, "Bad token type %d passed to %s", (int)tokenizer_token_type, __FUNCTION__); + DIE("bad token type"); + return token_type_invalid; } /// Helper function for parse_dump_tree(). diff --git a/src/tokenizer.cpp b/src/tokenizer.cpp index 1bf300f01..eb49cf263 100644 --- a/src/tokenizer.cpp +++ b/src/tokenizer.cpp @@ -528,16 +528,28 @@ maybe_t tokenizer_t::tok_next() { break; } case L'&': { - result.type = TOK_BACKGROUND; - result.length = 1; - this->buff++; + if (this->buff[1] == L'&') { + result.type = TOK_ANDAND; + result.length = 2; + this->buff += 2; + } else { + result.type = TOK_BACKGROUND; + result.length = 1; + this->buff++; + } break; } case L'|': { - result.type = TOK_PIPE; - result.redirected_fd = 1; - result.length = 1; - this->buff++; + if (this->buff[1] == L'|') { + result.type = TOK_OROR; + result.length = 2; + this->buff += 2; + } else { + result.type = TOK_PIPE; + result.redirected_fd = 1; + result.length = 1; + this->buff++; + } break; } case L'>': diff --git a/src/tokenizer.h b/src/tokenizer.h index 0c5226ead..e0aa58b50 100644 --- a/src/tokenizer.h +++ b/src/tokenizer.h @@ -14,6 +14,8 @@ enum token_type { TOK_ERROR, /// Error reading token TOK_STRING, /// String token TOK_PIPE, /// Pipe token + TOK_ANDAND, /// && token + TOK_OROR, /// || token TOK_END, /// End token (semicolon or newline, not literal end) TOK_REDIRECT, /// redirection token TOK_BACKGROUND, /// send job to bg token From 23d4f93556c6106bf8ef244314fecd3cb2ea5cc8 Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Thu, 1 Mar 2018 13:39:39 -0800 Subject: [PATCH 03/10] Add && and || to fish grammar This teaches the parser about && and ||, implemented via a new production "job_conjunction". These do not yet have execution support. --- src/fish_tests.cpp | 37 ++++++++++++++++++++++++++++++++++ src/parse_constants.h | 6 ++++++ src/parse_grammar.h | 18 +++++++++++++---- src/parse_grammar_elements.inc | 1 + src/parse_productions.cpp | 19 +++++++++++++++++ src/parse_tree.cpp | 37 +++++++++++++++++----------------- 6 files changed, 96 insertions(+), 22 deletions(-) diff --git a/src/fish_tests.cpp b/src/fish_tests.cpp index 6cb3c24dd..74aa08b0d 100644 --- a/src/fish_tests.cpp +++ b/src/fish_tests.cpp @@ -838,6 +838,38 @@ static void test_parser() { err(L"backgrounded 'while' conditional not reported as error"); } + if (!parse_util_detect_errors(L"true | || false")) { + err(L"bogus boolean statement error not detected on line %d", __LINE__); + } + + if (!parse_util_detect_errors(L"|| false")) { + err(L"bogus boolean statement error not detected on line %d", __LINE__); + } + + if (!parse_util_detect_errors(L"&& false")) { + err(L"bogus boolean statement error not detected on line %d", __LINE__); + } + + if (!parse_util_detect_errors(L"true ; && false")) { + err(L"bogus boolean statement error not detected on line %d", __LINE__); + } + + if (!parse_util_detect_errors(L"true ; || false")) { + err(L"bogus boolean statement error not detected on line %d", __LINE__); + } + + if (!parse_util_detect_errors(L"true || && false")) { + err(L"bogus boolean statement error not detected on line %d", __LINE__); + } + + if (!parse_util_detect_errors(L"true && || false")) { + err(L"bogus boolean statement error not detected on line %d", __LINE__); + } + + if (!parse_util_detect_errors(L"true && && false")) { + err(L"bogus boolean statement error not detected on line %d", __LINE__); + } + say(L"Testing basic evaluation"); // Ensure that we don't crash on infinite self recursion and mutual recursion. These must use @@ -3414,6 +3446,11 @@ static void test_new_parser_correctness() { {L"begin; end", true}, {L"begin if true; end; end;", true}, {L"begin if true ; echo hi ; end; end", true}, + {L"true && false || false", true}, + {L"true || false; and true", true}, + {L"true || ||", false}, + {L"|| true", false}, + {L"true || \n\n false", true}, }; for (size_t i = 0; i < sizeof parser_tests / sizeof *parser_tests; i++) { diff --git a/src/parse_constants.h b/src/parse_constants.h index 044f549e0..ff6dbec0a 100644 --- a/src/parse_constants.h +++ b/src/parse_constants.h @@ -17,6 +17,7 @@ enum parse_token_type_t { token_type_invalid = 1, // Non-terminal tokens symbol_job_list, + symbol_job_conjunction, symbol_job, symbol_job_continuation, symbol_statement, @@ -52,6 +53,8 @@ enum parse_token_type_t { parse_token_type_pipe, parse_token_type_redirection, parse_token_type_background, + parse_token_type_andand, + parse_token_type_oror, parse_token_type_end, // Special terminal type that means no more tokens forthcoming. parse_token_type_terminate, @@ -77,6 +80,8 @@ const enum_map token_enum_map[] = { {parse_token_type_pipe, L"parse_token_type_pipe"}, {parse_token_type_redirection, L"parse_token_type_redirection"}, {parse_token_type_string, L"parse_token_type_string"}, + {parse_token_type_andand, L"parse_token_type_andand"}, + {parse_token_type_oror, L"parse_token_type_oror"}, {parse_token_type_terminate, L"parse_token_type_terminate"}, {symbol_andor_job_list, L"symbol_andor_job_list"}, {symbol_argument, L"symbol_argument"}, @@ -98,6 +103,7 @@ const enum_map token_enum_map[] = { {symbol_if_clause, L"symbol_if_clause"}, {symbol_if_statement, L"symbol_if_statement"}, {symbol_job, L"symbol_job"}, + {symbol_job_conjunction, L"symbol_job_conjunction"}, {symbol_job_continuation, L"symbol_job_continuation"}, {symbol_job_list, L"symbol_job_list"}, {symbol_optional_newlines, L"symbol_optional_newlines"}, diff --git a/src/parse_grammar.h b/src/parse_grammar.h index 0526ef055..abdb56508 100644 --- a/src/parse_grammar.h +++ b/src/parse_grammar.h @@ -35,6 +35,8 @@ using tok_string = primitive; using tok_pipe = primitive; using tok_background = primitive; using tok_redirection = primitive; +using tok_andand = primitive; +using tok_oror = primitive; // Define keyword types. template @@ -197,12 +199,20 @@ struct alternative {}; // A job_list is a list of jobs, separated by semicolons or newlines DEF_ALT(job_list) { - using normal = seq; + using normal = seq; using empty_line = seq; using empty = grammar::empty; ALT_BODY(job_list, normal, empty_line, empty); }; +// A job_conjunction is a || or && continuation of a job +DEF_ALT(job_conjunction) { + using andands = seq; + using orors = seq; + using empty = grammar::empty; + ALT_BODY(job_conjunction, andands, orors, empty); +}; + // A job is a non-empty list of statements, separated by pipes. (Non-empty is useful for cases // like if statements, where we require a command). To represent "non-empty", we require a // statement, followed by a possibly empty job_continuation, and then optionally a background @@ -291,9 +301,9 @@ produces_sequence, argument, argument_list, tok_ // A boolean statement is AND or OR or NOT DEF_ALT(boolean_statement) { - using ands = seq, statement>; - using ors = seq, statement>; - using nots = seq, statement>; + using ands = seq, statement>; // foo ; and bar + using ors = seq, statement>; // foo ; or bar + using nots = seq, statement>; // not foo ALT_BODY(boolean_statement, ands, ors, nots); }; diff --git a/src/parse_grammar_elements.inc b/src/parse_grammar_elements.inc index ff087eeb7..e2e99e2f3 100644 --- a/src/parse_grammar_elements.inc +++ b/src/parse_grammar_elements.inc @@ -1,6 +1,7 @@ // Define ELEM before including this file. ELEM(job_list) ELEM(job) +ELEM(job_conjunction) ELEM(job_continuation) ELEM(statement) ELEM(if_statement) diff --git a/src/parse_productions.cpp b/src/parse_productions.cpp index e97b27208..c1acf5dd1 100644 --- a/src/parse_productions.cpp +++ b/src/parse_productions.cpp @@ -61,6 +61,19 @@ RESOLVE(job_list) { } } +RESOLVE(job_conjunction) { + UNUSED(token2); + UNUSED(out_tag); + switch (token1.type) { + case parse_token_type_andand: + return production_for(); + case parse_token_type_oror: + return production_for(); + default: + return production_for(); + } +} + RESOLVE(job_continuation) { UNUSED(token2); UNUSED(out_tag); @@ -106,6 +119,10 @@ RESOLVE(statement) { } switch (token1.type) { + case parse_token_type_andand: + case parse_token_type_oror: + return production_for(); + case parse_token_type_string: { switch (token1.keyword) { case parse_keyword_and: @@ -356,6 +373,8 @@ const production_element_t *parse_productions::production_for_token(parse_token_ case parse_token_type_pipe: case parse_token_type_redirection: case parse_token_type_background: + case parse_token_type_andand: + case parse_token_type_oror: case parse_token_type_end: case parse_token_type_terminate: { debug(0, "Terminal token type %ls passed to %s", token_type_description(node_type), diff --git a/src/parse_tree.cpp b/src/parse_tree.cpp index f99fc7951..f1f92fa5e 100644 --- a/src/parse_tree.cpp +++ b/src/parse_tree.cpp @@ -138,30 +138,29 @@ static wcstring token_type_user_presentable_description( switch (type) { // Hackish. We only support the following types. - case symbol_statement: { + case symbol_statement: return L"a command"; - } - case symbol_argument: { + case symbol_argument: return L"an argument"; - } - case parse_token_type_string: { + case symbol_job: + case symbol_job_list: + return L"a job"; + case parse_token_type_string: return L"a string"; - } - case parse_token_type_pipe: { + case parse_token_type_pipe: return L"a pipe"; - } - case parse_token_type_redirection: { + case parse_token_type_redirection: return L"a redirection"; - } - case parse_token_type_background: { + case parse_token_type_background: return L"a '&'"; - } - case parse_token_type_end: { + case parse_token_type_andand: + return L"'&&'"; + case parse_token_type_oror: + return L"'||'"; + case parse_token_type_end: return L"end of the statement"; - } - case parse_token_type_terminate: { + case parse_token_type_terminate: return L"end of the input"; - } default: { return format_string(L"a %ls", token_type_description(type)); } } } @@ -222,9 +221,9 @@ static inline parse_token_type_t parse_token_type_from_tokenizer_token( case TOK_PIPE: return parse_token_type_pipe; case TOK_ANDAND: + return parse_token_type_andand; case TOK_OROR: - // Temporary while && and || support is brought up. - return parse_special_type_comment; + return parse_token_type_oror; case TOK_END: return parse_token_type_end; case TOK_BACKGROUND: @@ -731,6 +730,8 @@ static bool type_is_terminal_type(parse_token_type_t type) { case parse_token_type_redirection: case parse_token_type_background: case parse_token_type_end: + case parse_token_type_andand: + case parse_token_type_oror: case parse_token_type_terminate: { return true; } From 8e9670ccd5c35e0d3b8288145e539d917b23a1d7 Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Thu, 1 Mar 2018 18:30:48 -0800 Subject: [PATCH 04/10] Implement execution of && and || This adds support for basic constructs. if and while is not yet supported. --- src/parse_constants.h | 1 + src/parse_execution.cpp | 36 +++++++++++++++++++++++++++++++--- src/parse_execution.h | 1 + src/parse_grammar.h | 20 +++++++++++-------- src/parse_grammar_elements.inc | 1 + src/parse_productions.cpp | 8 +++----- src/parse_util.cpp | 24 +++++++++++++---------- src/tnode.cpp | 4 ++++ src/tnode.h | 8 ++++++-- tests/andandoror.err | 0 tests/andandoror.in | 7 +++++++ tests/andandoror.out | 6 ++++++ 12 files changed, 88 insertions(+), 28 deletions(-) create mode 100644 tests/andandoror.err create mode 100644 tests/andandoror.in create mode 100644 tests/andandoror.out diff --git a/src/parse_constants.h b/src/parse_constants.h index ff6dbec0a..ef66ec9b1 100644 --- a/src/parse_constants.h +++ b/src/parse_constants.h @@ -18,6 +18,7 @@ enum parse_token_type_t { // Non-terminal tokens symbol_job_list, symbol_job_conjunction, + symbol_job_conjunction_continuation, symbol_job, symbol_job_continuation, symbol_statement, diff --git a/src/parse_execution.cpp b/src/parse_execution.cpp index 6116424ab..c23ca3fc8 100644 --- a/src/parse_execution.cpp +++ b/src/parse_execution.cpp @@ -107,7 +107,7 @@ tnode_t parse_execution_context_t::infinite_recursive_statem const wcstring &forbidden_function_name = parser->forbidden_function.back(); // Get the first job in the job list. - auto first_job = job_list.next_in_list(); + tnode_t first_job = job_list.try_get_child().child<0>(); if (!first_job) { return {}; } @@ -1220,6 +1220,35 @@ parse_execution_result_t parse_execution_context_t::run_1_job(tnode_t jo return parse_execution_success; } +parse_execution_result_t parse_execution_context_t::run_job_conjunction( + tnode_t job_expr, const block_t *associated_block) { + parse_execution_result_t result = parse_execution_success; + tnode_t cursor = job_expr; + tnode_t continuation; + while (cursor) { + if (should_cancel_execution(associated_block)) break; + bool skip = false; + if (continuation) { + // Check the conjunction type. + parse_bool_statement_type_t conj = bool_statement_type(continuation); + assert((conj == parse_bool_and || conj == parse_bool_or) && "Unexpected conjunction"); + if (conj == parse_bool_and) { + // Skip if last job failed. + skip = (proc_get_last_status() != 0); + } else if (conj == parse_bool_or) { + // Skip if last job succeeded. + skip = (proc_get_last_status() == 0); + } + } + if (! skip) { + result = run_1_job(cursor.child<0>(), associated_block); + } + continuation = cursor.child<1>(); + cursor = continuation.try_get_child(); + } + return result; +} + template parse_execution_result_t parse_execution_context_t::run_job_list(tnode_t job_list, const block_t *associated_block) { @@ -1227,9 +1256,10 @@ parse_execution_result_t parse_execution_context_t::run_job_list(tnode_t j "Not a job list"); parse_execution_result_t result = parse_execution_success; - while (tnode_t job = job_list.template next_in_list()) { + while (tnode_t job_expr = + job_list.template next_in_list()) { if (should_cancel_execution(associated_block)) break; - result = this->run_1_job(job, associated_block); + result = this->run_job_conjunction(job_expr, associated_block); } // Returns the last job executed. diff --git a/src/parse_execution.h b/src/parse_execution.h index 403d4392f..d15e69469 100644 --- a/src/parse_execution.h +++ b/src/parse_execution.h @@ -114,6 +114,7 @@ class parse_execution_context_t { io_chain_t *out_chain); parse_execution_result_t run_1_job(tnode_t job, const block_t *associated_block); + parse_execution_result_t run_job_conjunction(tnode_t job_conj, const block_t *associated_block); template parse_execution_result_t run_job_list(tnode_t job_list_node, const block_t *associated_block); diff --git a/src/parse_grammar.h b/src/parse_grammar.h index abdb56508..4dff823c2 100644 --- a/src/parse_grammar.h +++ b/src/parse_grammar.h @@ -197,20 +197,24 @@ struct alternative {}; static const production_element_t *resolve(const parse_token_t &, const parse_token_t &, \ parse_node_tag_t *); -// A job_list is a list of jobs, separated by semicolons or newlines +// A job_list is a list of job_conjunctions, separated by semicolons or newlines DEF_ALT(job_list) { - using normal = seq; + using normal = seq; using empty_line = seq; using empty = grammar::empty; ALT_BODY(job_list, normal, empty_line, empty); }; -// A job_conjunction is a || or && continuation of a job -DEF_ALT(job_conjunction) { - using andands = seq; - using orors = seq; +// A job_conjunction is a job followed by a continuation. +DEF(job_conjunction) produces_sequence { + BODY(job_conjunction); +}; + +DEF_ALT(job_conjunction_continuation) { + using andands = seq; + using orors = seq; using empty = grammar::empty; - ALT_BODY(job_conjunction, andands, orors, empty); + ALT_BODY(job_conjunction_continuation, andands, orors, empty); }; // A job is a non-empty list of statements, separated by pipes. (Non-empty is useful for cases @@ -311,7 +315,7 @@ DEF_ALT(boolean_statement) { // statement. DEF_ALT(andor_job_list) { using empty = grammar::empty; - using andor_job = seq; + using andor_job = seq; using empty_line = seq; ALT_BODY(andor_job_list, empty, andor_job, empty_line); }; diff --git a/src/parse_grammar_elements.inc b/src/parse_grammar_elements.inc index e2e99e2f3..1543a2a93 100644 --- a/src/parse_grammar_elements.inc +++ b/src/parse_grammar_elements.inc @@ -2,6 +2,7 @@ ELEM(job_list) ELEM(job) ELEM(job_conjunction) +ELEM(job_conjunction_continuation) ELEM(job_continuation) ELEM(statement) ELEM(if_statement) diff --git a/src/parse_productions.cpp b/src/parse_productions.cpp index c1acf5dd1..94c56a38b 100644 --- a/src/parse_productions.cpp +++ b/src/parse_productions.cpp @@ -61,13 +61,15 @@ RESOLVE(job_list) { } } -RESOLVE(job_conjunction) { +RESOLVE(job_conjunction_continuation) { UNUSED(token2); UNUSED(out_tag); switch (token1.type) { case parse_token_type_andand: + *out_tag = parse_bool_and; return production_for(); case parse_token_type_oror: + *out_tag = parse_bool_or; return production_for(); default: return production_for(); @@ -119,10 +121,6 @@ RESOLVE(statement) { } switch (token1.type) { - case parse_token_type_andand: - case parse_token_type_oror: - return production_for(); - case parse_token_type_string: { switch (token1.keyword) { case parse_keyword_and: diff --git a/src/parse_util.cpp b/src/parse_util.cpp index 24007a988..d201c651e 100644 --- a/src/parse_util.cpp +++ b/src/parse_util.cpp @@ -1047,6 +1047,7 @@ parser_test_error_bits_t parse_util_detect_errors_in_argument(tnode_t job, parse_error_list_t *parse_errors) { + namespace g = grammar; auto source_range = job.source_range(); if (!source_range) return false; @@ -1056,22 +1057,25 @@ static bool detect_errors_in_backgrounded_job(tnode_t job, // foo & ; or bar // if foo & ; end // while foo & ; end - if (job.try_get_parent()) { + if (job.try_get_parent()) { errored = append_syntax_error(parse_errors, source_range->start, BACKGROUND_IN_CONDITIONAL_ERROR_MSG); - } else if (job.try_get_parent()) { + } else if (job.try_get_parent()) { errored = append_syntax_error(parse_errors, source_range->start, BACKGROUND_IN_CONDITIONAL_ERROR_MSG); - } else if (auto job_list = job.try_get_parent()) { + } else if (auto jlist = + job.try_get_parent().try_get_parent()) { // This isn't very complete, e.g. we don't catch 'foo & ; not and bar'. - // Build the job list and then advance it by one. - auto first_job = job_list.next_in_list(); - assert(first_job == job && "Expected first job to be the node we found"); - (void)first_job; + // Fetch the job list and then advance it by one. + auto first_jconj = jlist.next_in_list(); + assert(first_jconj == job.try_get_parent() && + "Expected first job to be the node we found"); + (void)first_jconj; + // Try getting the next job as a boolean statement. - auto next_job = job_list.next_in_list(); - tnode_t next_stmt = next_job.child<0>(); - if (auto bool_stmt = next_stmt.try_get_child()) { + tnode_t next_job = jlist.next_in_list().child<0>(); + tnode_t next_stmt = next_job.child<0>(); + if (auto bool_stmt = next_stmt.try_get_child()) { // The next job is indeed a boolean statement. parse_bool_statement_type_t bool_type = bool_statement_type(bool_stmt); if (bool_type == parse_bool_and) { // this is not allowed diff --git a/src/tnode.cpp b/src/tnode.cpp index e0ee20ea5..64b93d124 100644 --- a/src/tnode.cpp +++ b/src/tnode.cpp @@ -50,6 +50,10 @@ enum parse_bool_statement_type_t bool_statement_type(tnode_t(stmt.tag()); } +enum parse_bool_statement_type_t bool_statement_type(tnode_t cont) { + return static_cast(cont.tag()); +} + maybe_t redirection_type(tnode_t redirection, const wcstring &src, int *out_fd, wcstring *out_target) { diff --git a/src/tnode.h b/src/tnode.h index c7edc5da6..92fde8da4 100644 --- a/src/tnode.h +++ b/src/tnode.h @@ -91,6 +91,8 @@ class tnode_t { // Helper to return whether the given tree is the same as ours. bool matches_node_tree(const parse_node_tree_t &t) const { return &t == tree; } + const parse_node_tree_t *get_tree() const { return tree; } + bool has_source() const { return nodeptr && nodeptr->has_source(); } // return the tag, or 0 if missing. @@ -147,7 +149,7 @@ class tnode_t { const parse_node_t *child = nullptr; if (nodeptr) child = tree->get_child(*nodeptr, Index); if (child && child->type == ChildType::token) return {tree, child}; - return {}; + return {tree, nullptr}; } /// assert that this is not empty and that the child at index Index has the given type, then @@ -206,7 +208,7 @@ class tnode_t { // We require that we can contain ourselves, and ItemType as well. static_assert(child_type_possible(), "Is not a list"); static_assert(child_type_possible(), "Is not a list of that type"); - if (!nodeptr) return {}; + if (!nodeptr) return {tree, nullptr}; const parse_node_t *next = tree->next_node_in_node_list(*nodeptr, ItemType::token, &nodeptr); return {tree, next}; @@ -234,6 +236,8 @@ parse_statement_decoration_t get_decoration(tnode_t st /// Return the type for a boolean statement. enum parse_bool_statement_type_t bool_statement_type(tnode_t stmt); +enum parse_bool_statement_type_t bool_statement_type(tnode_t stmt); + /// Given a redirection, get the redirection type (or none) and target (file path, or fd). maybe_t redirection_type(tnode_t redirection, const wcstring &src, int *out_fd, diff --git a/tests/andandoror.err b/tests/andandoror.err new file mode 100644 index 000000000..e69de29bb diff --git a/tests/andandoror.in b/tests/andandoror.in new file mode 100644 index 000000000..87b82d0bf --- /dev/null +++ b/tests/andandoror.in @@ -0,0 +1,7 @@ +# Test && and || support + +echo first && echo second +echo third || echo fourth +true && false ; echo "true && false: $status" +true || false ; echo "true || false: $status" +true && false || true ; echo "true && false || true: $status" diff --git a/tests/andandoror.out b/tests/andandoror.out new file mode 100644 index 000000000..a1a62e7e8 --- /dev/null +++ b/tests/andandoror.out @@ -0,0 +1,6 @@ +first +second +third +true && false: 1 +true || false: 0 +true && false || true: 0 From e1dafeab01ae5b081af8016814725d3624eb953a Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Fri, 2 Mar 2018 10:23:57 -0800 Subject: [PATCH 05/10] Add && and || support to the conditions of if and while This updates the "boolean tail" feature of the if and while conditions to know about job_conjunction, thereby respecting && and || --- src/fish_indent.cpp | 2 +- src/parse_execution.cpp | 8 ++++---- src/parse_grammar.h | 5 +++-- src/parse_util.cpp | 8 ++++---- tests/andandoror.err | 12 ++++++++++++ tests/andandoror.in | 28 +++++++++++++++++++++++++++- tests/andandoror.out | 24 ++++++++++++++++++++++++ 7 files changed, 75 insertions(+), 12 deletions(-) diff --git a/src/fish_indent.cpp b/src/fish_indent.cpp index 801b835ca..12545786d 100644 --- a/src/fish_indent.cpp +++ b/src/fish_indent.cpp @@ -121,7 +121,7 @@ static void prettify_node_recursive(const wcstring &source, const parse_node_tre const bool is_root_case_list = node_type == symbol_case_item_list && parent_type != symbol_case_item_list; const bool is_if_while_header = - (node_type == symbol_job || node_type == symbol_andor_job_list) && + (node_type == symbol_job_conjunction || node_type == symbol_andor_job_list) && (parent_type == symbol_if_clause || parent_type == symbol_while_header); if (is_root_job_list || is_root_case_list || is_if_while_header) { diff --git a/src/parse_execution.cpp b/src/parse_execution.cpp index c23ca3fc8..06f31a624 100644 --- a/src/parse_execution.cpp +++ b/src/parse_execution.cpp @@ -244,12 +244,12 @@ parse_execution_result_t parse_execution_context_t::run_if_statement( } // An if condition has a job and a "tail" of andor jobs, e.g. "foo ; and bar; or baz". - tnode_t condition_head = if_clause.child<1>(); + tnode_t condition_head = if_clause.child<1>(); tnode_t condition_boolean_tail = if_clause.child<3>(); // Check the condition and the tail. We treat parse_execution_errored here as failure, in // accordance with historic behavior. - parse_execution_result_t cond_ret = run_1_job(condition_head, ib); + parse_execution_result_t cond_ret = run_job_conjunction(condition_head, ib); if (cond_ret == parse_execution_success) { cond_ret = run_job_list(condition_boolean_tail, ib); } @@ -527,13 +527,13 @@ parse_execution_result_t parse_execution_context_t::run_while_statement( parse_execution_result_t ret = parse_execution_success; // The conditions of the while loop. - tnode_t condition_head = header.child<1>(); + tnode_t condition_head = header.child<1>(); tnode_t condition_boolean_tail = header.child<3>(); // Run while the condition is true. for (;;) { // Check the condition. - parse_execution_result_t cond_ret = this->run_1_job(condition_head, wb); + parse_execution_result_t cond_ret = this->run_job_conjunction(condition_head, wb); if (cond_ret == parse_execution_success) { cond_ret = run_job_list(condition_boolean_tail, wb); } diff --git a/src/parse_grammar.h b/src/parse_grammar.h index 4dff823c2..99b021f4a 100644 --- a/src/parse_grammar.h +++ b/src/parse_grammar.h @@ -245,7 +245,7 @@ produces_sequence, job, tok_end, andor_job_list, job_list>{ +produces_sequence, job_conjunction, tok_end, andor_job_list, job_list>{ BODY(if_clause)}; DEF_ALT(else_clause) { @@ -294,7 +294,8 @@ produces_sequence, tok_string, keyword, job, tok_end, andor_job_list>{BODY(while_header)}; +produces_sequence, job_conjunction, tok_end, andor_job_list>{ + BODY(while_header)}; DEF(begin_header) produces_single>{BODY(begin_header)}; diff --git a/src/parse_util.cpp b/src/parse_util.cpp index d201c651e..83860912a 100644 --- a/src/parse_util.cpp +++ b/src/parse_util.cpp @@ -1057,14 +1057,14 @@ static bool detect_errors_in_backgrounded_job(tnode_t job, // foo & ; or bar // if foo & ; end // while foo & ; end - if (job.try_get_parent()) { + auto job_conj = job.try_get_parent(); + if (job_conj.try_get_parent()) { errored = append_syntax_error(parse_errors, source_range->start, BACKGROUND_IN_CONDITIONAL_ERROR_MSG); - } else if (job.try_get_parent()) { + } else if (job_conj.try_get_parent()) { errored = append_syntax_error(parse_errors, source_range->start, BACKGROUND_IN_CONDITIONAL_ERROR_MSG); - } else if (auto jlist = - job.try_get_parent().try_get_parent()) { + } else if (auto jlist = job_conj.try_get_parent()) { // This isn't very complete, e.g. we don't catch 'foo & ; not and bar'. // Fetch the job list and then advance it by one. auto first_jconj = jlist.next_in_list(); diff --git a/tests/andandoror.err b/tests/andandoror.err index e69de29bb..ce5b602ba 100644 --- a/tests/andandoror.err +++ b/tests/andandoror.err @@ -0,0 +1,12 @@ + +#################### +# Basic && and || support + +#################### +# && and || in if statements + +#################### +# && and || in while statements + +#################### +# Complex scenarios diff --git a/tests/andandoror.in b/tests/andandoror.in index 87b82d0bf..f2093039b 100644 --- a/tests/andandoror.in +++ b/tests/andandoror.in @@ -1,7 +1,33 @@ -# Test && and || support +logmsg "Basic && and || support" echo first && echo second echo third || echo fourth true && false ; echo "true && false: $status" true || false ; echo "true || false: $status" true && false || true ; echo "true && false || true: $status" + +logmsg "&& and || in if statements" + +if true || false ; echo "if test 1 ok" ; end +if true && false ; else; echo "if test 2 ok" ; end +if true && false ; or true ; echo "if test 3 ok" ; end +if [ 0 = 1 ] || [ 5 -ge 3 ] ; echo "if test 4 ok"; end + +logmsg "&& and || in while statements" + +set alpha 0 +set beta 0 +set gamma 0 +set delta 0 +while [ $alpha -lt 2 ] && [ $beta -lt 3 ] + and [ $gamma -lt 4 ] || [ $delta -lt 5 ] + echo $alpha $beta $gamma + set alpha ( math $alpha + 1 ) + set beta ( math $beta + 1 ) + set gamma ( math $gamma + 1 ) + set delta ( math $delta + 1 ) +end + +logmsg "Complex scenarios" + +begin; echo 1 ; false ; end || begin ; echo 2 && echo 3 ; end diff --git a/tests/andandoror.out b/tests/andandoror.out index a1a62e7e8..e8f8f8bed 100644 --- a/tests/andandoror.out +++ b/tests/andandoror.out @@ -1,6 +1,30 @@ + +#################### +# Basic && and || support first second third true && false: 1 true || false: 0 true && false || true: 0 + +#################### +# && and || in if statements +if test 1 ok +if test 2 ok +if test 3 ok +if test 4 ok + +#################### +# && and || in while statements +0 0 0 +1 1 1 +2 2 2 +3 3 3 +4 4 4 + +#################### +# Complex scenarios +1 +2 +3 From 357d3b8c6d0bd6f1a84bca4d05d075f62bbf1ff9 Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Fri, 2 Mar 2018 18:09:16 -0800 Subject: [PATCH 06/10] Rework 'and' and 'or' to be "job decorators" This promotes "and" and "or" from a type of statement to "job decorators," as a possible prefix on a job. The point is to rationalize how they interact with && and ||. In the new world 'and' and 'or' apply to a entire job conjunction, i.e. they have "lower precedence." Example: if [ $age -ge 0 ] && [ $age -le 18 ] or [ $age -ge 75 ] && [ $age -le 100 ] echo "Child or senior" end --- src/highlight.cpp | 2 +- src/parse_constants.h | 38 +++-------------- src/parse_execution.cpp | 76 +++++++++++++++------------------- src/parse_execution.h | 7 +++- src/parse_grammar.h | 24 ++++++----- src/parse_grammar_elements.inc | 3 +- src/parse_productions.cpp | 45 ++++++++++---------- src/parse_util.cpp | 45 +++++++++++--------- src/tnode.cpp | 33 ++++++++++----- src/tnode.h | 18 ++++++-- tests/andandoror.in | 7 +++- tests/andandoror.out | 1 + 12 files changed, 151 insertions(+), 148 deletions(-) diff --git a/src/highlight.cpp b/src/highlight.cpp index cc69010ec..91ef533c2 100644 --- a/src/highlight.cpp +++ b/src/highlight.cpp @@ -1033,7 +1033,7 @@ const highlighter_t::color_array_t &highlighter_t::highlight() { case symbol_if_clause: case symbol_else_clause: case symbol_case_item: - case symbol_boolean_statement: + case symbol_not_statement: case symbol_decorated_statement: case symbol_if_statement: { this->color_children(node, parse_token_type_string, highlight_spec_command); diff --git a/src/parse_constants.h b/src/parse_constants.h index ef66ec9b1..b4f679b6a 100644 --- a/src/parse_constants.h +++ b/src/parse_constants.h @@ -19,6 +19,7 @@ enum parse_token_type_t { symbol_job_list, symbol_job_conjunction, symbol_job_conjunction_continuation, + symbol_job_decorator, symbol_job, symbol_job_continuation, symbol_statement, @@ -35,7 +36,7 @@ enum parse_token_type_t { symbol_switch_statement, symbol_case_item_list, symbol_case_item, - symbol_boolean_statement, + symbol_not_statement, symbol_decorated_statement, symbol_plain_statement, symbol_arguments_or_redirections_list, @@ -84,36 +85,9 @@ const enum_map token_enum_map[] = { {parse_token_type_andand, L"parse_token_type_andand"}, {parse_token_type_oror, L"parse_token_type_oror"}, {parse_token_type_terminate, L"parse_token_type_terminate"}, - {symbol_andor_job_list, L"symbol_andor_job_list"}, - {symbol_argument, L"symbol_argument"}, - {symbol_argument_list, L"symbol_argument_list"}, - {symbol_arguments_or_redirections_list, L"symbol_arguments_or_redirections_list"}, - {symbol_begin_header, L"symbol_begin_header"}, - {symbol_block_header, L"symbol_block_header"}, - {symbol_block_statement, L"symbol_block_statement"}, - {symbol_boolean_statement, L"symbol_boolean_statement"}, - {symbol_case_item, L"symbol_case_item"}, - {symbol_case_item_list, L"symbol_case_item_list"}, - {symbol_decorated_statement, L"symbol_decorated_statement"}, - {symbol_else_clause, L"symbol_else_clause"}, - {symbol_else_continuation, L"symbol_else_continuation"}, - {symbol_end_command, L"symbol_end_command"}, - {symbol_for_header, L"symbol_for_header"}, - {symbol_freestanding_argument_list, L"symbol_freestanding_argument_list"}, - {symbol_function_header, L"symbol_function_header"}, - {symbol_if_clause, L"symbol_if_clause"}, - {symbol_if_statement, L"symbol_if_statement"}, - {symbol_job, L"symbol_job"}, - {symbol_job_conjunction, L"symbol_job_conjunction"}, - {symbol_job_continuation, L"symbol_job_continuation"}, - {symbol_job_list, L"symbol_job_list"}, - {symbol_optional_newlines, L"symbol_optional_newlines"}, - {symbol_optional_background, L"symbol_optional_background"}, - {symbol_plain_statement, L"symbol_plain_statement"}, - {symbol_redirection, L"symbol_redirection"}, - {symbol_statement, L"symbol_statement"}, - {symbol_switch_statement, L"symbol_switch_statement"}, - {symbol_while_header, L"symbol_while_header"}, +// Define all symbols +#define ELEM(sym) {symbol_##sym, L"symbol_" #sym}, +#include "parse_grammar_elements.inc" {token_type_invalid, L"token_type_invalid"}, {token_type_invalid, NULL}}; #define token_enum_map_len (sizeof token_enum_map / sizeof *token_enum_map) @@ -165,7 +139,7 @@ enum parse_statement_decoration_t { }; // Boolean statement types, stored in node tag. -enum parse_bool_statement_type_t { parse_bool_and, parse_bool_or, parse_bool_not }; +enum parse_bool_statement_type_t { parse_bool_none, parse_bool_and, parse_bool_or }; // Whether a statement is backgrounded. enum parse_optional_background_t { parse_no_background, parse_background }; diff --git a/src/parse_execution.cpp b/src/parse_execution.cpp index 06f31a624..a54f6fb49 100644 --- a/src/parse_execution.cpp +++ b/src/parse_execution.cpp @@ -107,7 +107,7 @@ tnode_t parse_execution_context_t::infinite_recursive_statem const wcstring &forbidden_function_name = parser->forbidden_function.back(); // Get the first job in the job list. - tnode_t first_job = job_list.try_get_child().child<0>(); + tnode_t first_job = job_list.try_get_child().child<0>(); if (!first_job) { return {}; } @@ -215,7 +215,7 @@ bool parse_execution_context_t::job_is_simple_block(tnode_t job_node) co return is_empty(statement.require_get_child().child<5>()); case symbol_if_statement: return is_empty(statement.require_get_child().child<3>()); - case symbol_boolean_statement: + case symbol_not_statement: case symbol_decorated_statement: // not block statements return false; @@ -921,33 +921,10 @@ bool parse_execution_context_t::determine_io_chain(tnode_t bool_statement) { - // Handle a boolean statement. - bool skip_job = false; - switch (bool_statement_type(bool_statement)) { - case parse_bool_and: { - // AND. Skip if the last job failed. - skip_job = (proc_get_last_status() != 0); - break; - } - case parse_bool_or: { - // OR. Skip if the last job succeeded. - skip_job = (proc_get_last_status() == 0); - break; - } - case parse_bool_not: { - // NOT. Negate it. - job->set_flag(JOB_NEGATE, !job->get_flag(JOB_NEGATE)); - break; - } - } - - if (skip_job) { - return parse_execution_skipped; - } - return this->populate_job_process(job, proc, - bool_statement.require_get_child()); +parse_execution_result_t parse_execution_context_t::populate_not_process( + job_t *job, process_t *proc, tnode_t not_statement) { + job->set_flag(JOB_NEGATE, !job->get_flag(JOB_NEGATE)); + return this->populate_job_process(job, proc, not_statement.child<1>()); } template @@ -985,8 +962,8 @@ parse_execution_result_t parse_execution_context_t::populate_job_process( parse_execution_result_t result = parse_execution_success; switch (specific_statement.type) { - case symbol_boolean_statement: { - result = this->populate_boolean_process(job, proc, {&tree(), &specific_statement}); + case symbol_not_statement: { + result = this->populate_not_process(job, proc, {&tree(), &specific_statement}); break; } case symbol_block_statement: @@ -1224,6 +1201,7 @@ parse_execution_result_t parse_execution_context_t::run_job_conjunction( tnode_t job_expr, const block_t *associated_block) { parse_execution_result_t result = parse_execution_success; tnode_t cursor = job_expr; + // continuation is the parent of the cursor tnode_t continuation; while (cursor) { if (should_cancel_execution(associated_block)) break; @@ -1232,13 +1210,7 @@ parse_execution_result_t parse_execution_context_t::run_job_conjunction( // Check the conjunction type. parse_bool_statement_type_t conj = bool_statement_type(continuation); assert((conj == parse_bool_and || conj == parse_bool_or) && "Unexpected conjunction"); - if (conj == parse_bool_and) { - // Skip if last job failed. - skip = (proc_get_last_status() != 0); - } else if (conj == parse_bool_or) { - // Skip if last job succeeded. - skip = (proc_get_last_status() == 0); - } + skip = should_skip(conj); } if (! skip) { result = run_1_job(cursor.child<0>(), associated_block); @@ -1249,20 +1221,40 @@ parse_execution_result_t parse_execution_context_t::run_job_conjunction( return result; } +bool parse_execution_context_t::should_skip(parse_bool_statement_type_t type) const { + switch (type) { + case parse_bool_and: + // AND. Skip if the last job failed. + return proc_get_last_status() != 0; + case parse_bool_or: + // OR. Skip if the last job succeeded. + return proc_get_last_status() == 0; + default: + return false; + } +} + template parse_execution_result_t parse_execution_context_t::run_job_list(tnode_t job_list, const block_t *associated_block) { + // We handle both job_list and andor_job_list uniformly. static_assert(Type::token == symbol_job_list || Type::token == symbol_andor_job_list, "Not a job list"); parse_execution_result_t result = parse_execution_success; - while (tnode_t job_expr = - job_list.template next_in_list()) { + while (auto job_conj = job_list.template next_in_list()) { if (should_cancel_execution(associated_block)) break; - result = this->run_job_conjunction(job_expr, associated_block); + + // Maybe skip the job if it has a leading and/or. + // Skipping is treated as success. + if (should_skip(get_decorator(job_conj))) { + result = parse_execution_success; + } else { + result = this->run_job_conjunction(job_conj, associated_block); + } } - // Returns the last job executed. + // Returns the result of the last job executed or skipped. return result; } diff --git a/src/parse_execution.h b/src/parse_execution.h index d15e69469..dceb87235 100644 --- a/src/parse_execution.h +++ b/src/parse_execution.h @@ -72,6 +72,9 @@ class parse_execution_context_t { tnode_t job_list, wcstring *out_func_name) const; bool is_function_context() const; + /// Return whether we should skip a job with the given bool statement type. + bool should_skip(parse_bool_statement_type_t type) const; + /// Indicates whether a job is a simple block (one block, no redirections). bool job_is_simple_block(tnode_t job) const; @@ -81,8 +84,8 @@ class parse_execution_context_t { // These create process_t structures from statements. parse_execution_result_t populate_job_process(job_t *job, process_t *proc, tnode_t statement); - parse_execution_result_t populate_boolean_process( - job_t *job, process_t *proc, tnode_t bool_statement); + parse_execution_result_t populate_not_process(job_t *job, process_t *proc, + tnode_t not_statement); parse_execution_result_t populate_plain_process(job_t *job, process_t *proc, tnode_t statement); diff --git a/src/parse_grammar.h b/src/parse_grammar.h index 99b021f4a..e08ccb62d 100644 --- a/src/parse_grammar.h +++ b/src/parse_grammar.h @@ -199,12 +199,20 @@ struct alternative {}; // A job_list is a list of job_conjunctions, separated by semicolons or newlines DEF_ALT(job_list) { - using normal = seq; + using normal = seq; using empty_line = seq; using empty = grammar::empty; ALT_BODY(job_list, normal, empty_line, empty); }; +// Job decorators are 'and' and 'or'. These apply to the whole job. +DEF_ALT(job_decorator) { + using ands = single>; + using ors = single>; + using empty = grammar::empty; + ALT_BODY(job_decorator, ands, ors, empty); +}; + // A job_conjunction is a job followed by a continuation. DEF(job_conjunction) produces_sequence { BODY(job_conjunction); @@ -231,12 +239,12 @@ DEF_ALT(job_continuation) { // A statement is a normal command, or an if / while / and etc DEF_ALT(statement) { - using boolean = single; + using nots = single; using block = single; using ifs = single; using switchs = single; using decorated = single; - ALT_BODY(statement, boolean, block, ifs, switchs, decorated); + ALT_BODY(statement, nots, block, ifs, switchs, decorated); }; // A block is a conditional, loop, or begin/end @@ -304,19 +312,15 @@ DEF(function_header) produces_sequence, argument, argument_list, tok_end>{ BODY(function_header)}; -// A boolean statement is AND or OR or NOT -DEF_ALT(boolean_statement) { - using ands = seq, statement>; // foo ; and bar - using ors = seq, statement>; // foo ; or bar - using nots = seq, statement>; // not foo - ALT_BODY(boolean_statement, ands, ors, nots); +DEF(not_statement) produces_sequence, statement> { + BODY(not_statement); }; // An andor_job_list is zero or more job lists, where each starts with an `and` or `or` boolean // statement. DEF_ALT(andor_job_list) { using empty = grammar::empty; - using andor_job = seq; + using andor_job = seq; using empty_line = seq; ALT_BODY(andor_job_list, empty, andor_job, empty_line); }; diff --git a/src/parse_grammar_elements.inc b/src/parse_grammar_elements.inc index 1543a2a93..79339b6cb 100644 --- a/src/parse_grammar_elements.inc +++ b/src/parse_grammar_elements.inc @@ -1,6 +1,7 @@ // Define ELEM before including this file. ELEM(job_list) ELEM(job) +ELEM(job_decorator) ELEM(job_conjunction) ELEM(job_conjunction_continuation) ELEM(job_continuation) @@ -18,7 +19,7 @@ ELEM(for_header) ELEM(while_header) ELEM(begin_header) ELEM(function_header) -ELEM(boolean_statement) +ELEM(not_statement) ELEM(andor_job_list) ELEM(decorated_statement) ELEM(plain_statement) diff --git a/src/parse_productions.cpp b/src/parse_productions.cpp index 94c56a38b..a09e28e13 100644 --- a/src/parse_productions.cpp +++ b/src/parse_productions.cpp @@ -61,6 +61,26 @@ RESOLVE(job_list) { } } +// A job decorator is AND or OR +RESOLVE(job_decorator) { + UNUSED(token2); + + switch (token1.keyword) { + case parse_keyword_and: { + *out_tag = parse_bool_and; + return production_for(); + } + case parse_keyword_or: { + *out_tag = parse_bool_or; + return production_for(); + } + default: { + *out_tag = parse_bool_none; + return production_for(); + } + } +} + RESOLVE(job_conjunction_continuation) { UNUSED(token2); UNUSED(out_tag); @@ -123,10 +143,8 @@ RESOLVE(statement) { switch (token1.type) { case parse_token_type_string: { switch (token1.keyword) { - case parse_keyword_and: - case parse_keyword_or: case parse_keyword_not: { - return production_for(); + return production_for(); } case parse_keyword_for: case parse_keyword_while: @@ -260,27 +278,6 @@ RESOLVE(block_header) { } } -// A boolean statement is AND or OR or NOT. -RESOLVE(boolean_statement) { - UNUSED(token2); - - switch (token1.keyword) { - case parse_keyword_and: { - *out_tag = parse_bool_and; - return production_for(); - } - case parse_keyword_or: { - *out_tag = parse_bool_or; - return production_for(); - } - case parse_keyword_not: { - *out_tag = parse_bool_not; - return production_for(); - } - default: { return NO_PRODUCTION; } - } -} - RESOLVE(decorated_statement) { // If this is e.g. 'command --help' then the command is 'command' and not a decoration. If the diff --git a/src/parse_util.cpp b/src/parse_util.cpp index 83860912a..2927763ee 100644 --- a/src/parse_util.cpp +++ b/src/parse_util.cpp @@ -1072,17 +1072,15 @@ static bool detect_errors_in_backgrounded_job(tnode_t job, "Expected first job to be the node we found"); (void)first_jconj; - // Try getting the next job as a boolean statement. - tnode_t next_job = jlist.next_in_list().child<0>(); - tnode_t next_stmt = next_job.child<0>(); - if (auto bool_stmt = next_stmt.try_get_child()) { + // Try getting the next job's decorator. + if (auto next_job_dec = jlist.next_in_list()) { // The next job is indeed a boolean statement. - parse_bool_statement_type_t bool_type = bool_statement_type(bool_stmt); - if (bool_type == parse_bool_and) { // this is not allowed - errored = append_syntax_error(parse_errors, bool_stmt.source_range()->start, + parse_bool_statement_type_t bool_type = bool_statement_type(next_job_dec); + if (bool_type == parse_bool_and) { + errored = append_syntax_error(parse_errors, next_job_dec.source_range()->start, BOOL_AFTER_BACKGROUND_ERROR_MSG, L"and"); - } else if (bool_type == parse_bool_or) { // this is not allowed - errored = append_syntax_error(parse_errors, bool_stmt.source_range()->start, + } else if (bool_type == parse_bool_or) { + errored = append_syntax_error(parse_errors, next_job_dec.source_range()->start, BOOL_AFTER_BACKGROUND_ERROR_MSG, L"or"); } } @@ -1100,7 +1098,8 @@ static bool detect_errors_in_plain_statement(const wcstring &buff_src, // In a few places below, we want to know if we are in a pipeline. tnode_t st = pst.try_get_parent().try_get_parent(); - const bool is_in_pipeline = statement_is_in_pipeline(st, true /* count first */); + pipeline_position_t pipe_pos = get_pipeline_position(st); + bool is_in_pipeline = (pipe_pos != pipeline_position_t::none); // We need to know the decoration. const enum parse_statement_decoration_t decoration = get_decoration(pst); @@ -1110,6 +1109,19 @@ static bool detect_errors_in_plain_statement(const wcstring &buff_src, errored = append_syntax_error(parse_errors, source_start, EXEC_ERR_MSG, L"exec"); } + // This is a somewhat stale check that 'and' and 'or' are not in pipelines, except at the + // beginning. We can't disallow them as commands entirely because we need to support 'and + // --help', etc. + if (pipe_pos == pipeline_position_t::subsequent) { + // check if our command is 'and' or 'or'. This is very clumsy; we don't catch e.g. quoted + // commands. + wcstring command = pst.child<0>().get_source(buff_src); + if (command == L"and" || command == L"or") { + errored = + append_syntax_error(parse_errors, source_start, EXEC_ERR_MSG, command.c_str()); + } + } + if (maybe_t mcommand = command_for_plain_statement(pst, buff_src)) { wcstring command = std::move(*mcommand); // Check that we can expand the command. @@ -1254,16 +1266,9 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src, has_unclosed_block = true; } else if (node.type == symbol_statement && !node.has_source()) { // Check for a statement without source in a pipeline, i.e. unterminated pipeline. - has_unclosed_pipe |= statement_is_in_pipeline({&node_tree, &node}, false); - } else if (node.type == symbol_boolean_statement) { - // 'or' and 'and' can be in a pipeline, as long as they're first. - tnode_t gbs{&node_tree, &node}; - parse_bool_statement_type_t type = bool_statement_type(gbs); - if ((type == parse_bool_and || type == parse_bool_or) && - statement_is_in_pipeline(gbs.try_get_parent(), - false /* don't count first */)) { - errored = append_syntax_error(&parse_errors, node.source_start, EXEC_ERR_MSG, - (type == parse_bool_and) ? L"and" : L"or"); + auto pipe_pos = get_pipeline_position({&node_tree, &node}); + if (pipe_pos != pipeline_position_t::none) { + has_unclosed_pipe = true; } } else if (node.type == symbol_argument) { tnode_t arg{&node_tree, &node}; diff --git a/src/tnode.cpp b/src/tnode.cpp index 64b93d124..8e4a447fb 100644 --- a/src/tnode.cpp +++ b/src/tnode.cpp @@ -46,7 +46,7 @@ enum parse_statement_decoration_t get_decoration(tnode_t stmt) { +enum parse_bool_statement_type_t bool_statement_type(tnode_t stmt) { return static_cast(stmt.tag()); } @@ -111,24 +111,35 @@ bool job_node_is_background(tnode_t job) { return bg.tag() == parse_background; } -bool statement_is_in_pipeline(tnode_t st, bool include_first) { +parse_bool_statement_type_t get_decorator(tnode_t conj) { + using namespace grammar; + tnode_t dec; + // We have two possible parents: job_list and andor_job_list. + if (auto p = conj.try_get_parent()) { + dec = p.require_get_child(); + } else if (auto p = conj.try_get_parent()) { + dec = p.require_get_child(); + } + // note this returns 0 (none) if dec is empty. + return bool_statement_type(dec); +} + +pipeline_position_t get_pipeline_position(tnode_t st) { using namespace grammar; if (!st) { - return false; + return pipeline_position_t::none; } // If we're part of a job continuation, we're definitely in a pipeline. if (st.try_get_parent()) { - return true; + return pipeline_position_t::subsequent; } - // If include_first is set, check if we're the beginning of a job, and if so, whether that job + // Check if we're the beginning of a job, and if so, whether that job // has a non-empty continuation. - if (include_first) { - tnode_t jc = st.try_get_parent().child<1>(); - if (jc.try_get_child()) { - return true; - } + tnode_t jc = st.try_get_parent().child<1>(); + if (jc.try_get_child()) { + return pipeline_position_t::first; } - return false; + return pipeline_position_t::none; } diff --git a/src/tnode.h b/src/tnode.h index 92fde8da4..60fac38b6 100644 --- a/src/tnode.h +++ b/src/tnode.h @@ -234,7 +234,7 @@ maybe_t command_for_plain_statement(tnode_t parse_statement_decoration_t get_decoration(tnode_t stmt); /// Return the type for a boolean statement. -enum parse_bool_statement_type_t bool_statement_type(tnode_t stmt); +enum parse_bool_statement_type_t bool_statement_type(tnode_t stmt); enum parse_bool_statement_type_t bool_statement_type(tnode_t stmt); @@ -253,9 +253,19 @@ arguments_node_list_t get_argument_nodes(tnode_t); -/// Return whether the statement is part of a pipeline. If include_first is set, the first command -/// in a pipeline is considered part of it; otherwise only the second or additional commands are. -bool statement_is_in_pipeline(tnode_t st, bool include_first); +/// If the conjunction is has a decorator (and/or), return it; otherwise return none. This only +/// considers the leading conjunction, e.g. in `and true || false` only the 'true' conjunction will +/// return 'and'. +parse_bool_statement_type_t get_decorator(tnode_t); + +/// Return whether the statement is part of a pipeline. +/// This doesn't detect e.g. pipelines involving our parent's block statements. +enum class pipeline_position_t { + none, // not part of a pipeline + first, // first command in a pipeline + subsequent // second or further command in a pipeline +}; +pipeline_position_t get_pipeline_position(tnode_t st); /// Check whether an argument_list is a root list. inline bool argument_list_is_root(tnode_t list) { diff --git a/tests/andandoror.in b/tests/andandoror.in index f2093039b..f739cc1ea 100644 --- a/tests/andandoror.in +++ b/tests/andandoror.in @@ -20,7 +20,7 @@ set beta 0 set gamma 0 set delta 0 while [ $alpha -lt 2 ] && [ $beta -lt 3 ] - and [ $gamma -lt 4 ] || [ $delta -lt 5 ] + or [ $gamma -lt 4 ] || [ $delta -lt 5 ] echo $alpha $beta $gamma set alpha ( math $alpha + 1 ) set beta ( math $beta + 1 ) @@ -31,3 +31,8 @@ end logmsg "Complex scenarios" begin; echo 1 ; false ; end || begin ; echo 2 && echo 3 ; end + +if false && true + or not false + echo 4 +end diff --git a/tests/andandoror.out b/tests/andandoror.out index e8f8f8bed..84f9f04f2 100644 --- a/tests/andandoror.out +++ b/tests/andandoror.out @@ -28,3 +28,4 @@ if test 4 ok 1 2 3 +4 From f83742d57993b651f9d97c2d508f7e6ffb6d30f6 Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Sun, 4 Mar 2018 15:03:56 -0800 Subject: [PATCH 07/10] Highlight && and || as operators This also switches 'and' and 'or' to operators as well. --- src/fish_tests.cpp | 21 ++++++++++++++++++--- src/highlight.cpp | 10 ++++++++++ src/parse_tree.h | 2 +- 3 files changed, 29 insertions(+), 4 deletions(-) diff --git a/src/fish_tests.cpp b/src/fish_tests.cpp index 74aa08b0d..17cf062a5 100644 --- a/src/fish_tests.cpp +++ b/src/fish_tests.cpp @@ -3951,10 +3951,23 @@ static void test_highlighting() { {L"2>", highlight_spec_redirection}, {NULL, -1}}; + const highlight_component_t components15[] = {{L"if", highlight_spec_command}, + {L"true", highlight_spec_command}, + {L"&&", highlight_spec_operator}, + {L"false", highlight_spec_command}, + {L";", highlight_spec_statement_terminator}, + {L"or", highlight_spec_operator}, + {L"false", highlight_spec_command}, + {L"||", highlight_spec_operator}, + {L"true", highlight_spec_command}, + {L";", highlight_spec_statement_terminator}, + {L"end", highlight_spec_command}, + {NULL, -1}}; + const highlight_component_t *tests[] = {components1, components2, components3, components4, components5, components6, components7, components8, components9, components10, components11, components12, - components13, components14}; + components13, components14, components15}; for (size_t which = 0; which < sizeof tests / sizeof *tests; which++) { const highlight_component_t *components = tests[which]; // Count how many we have. @@ -3990,8 +4003,10 @@ static void test_highlighting() { if (expected_colors.at(i) != colors.at(i)) { const wcstring spaces(i, L' '); - err(L"Wrong color at index %lu in text (expected %#x, actual %#x):\n%ls\n%ls^", i, - expected_colors.at(i), colors.at(i), text.c_str(), spaces.c_str()); + err(L"Wrong color in test %lu at index %lu in text (expected %#x, actual " + L"%#x):\n%ls\n%ls^", + which + 1, i, expected_colors.at(i), colors.at(i), text.c_str(), + spaces.c_str()); } } } diff --git a/src/highlight.cpp b/src/highlight.cpp index 91ef533c2..0deff0590 100644 --- a/src/highlight.cpp +++ b/src/highlight.cpp @@ -1059,6 +1059,16 @@ const highlighter_t::color_array_t &highlighter_t::highlight() { this->color_argument(fhead.child<1>()); break; } + + case parse_token_type_andand: + case parse_token_type_oror: + this->color_node(node, highlight_spec_operator); + break; + + case symbol_job_decorator: + this->color_node(node, highlight_spec_operator); + break; + case parse_token_type_pipe: case parse_token_type_background: case parse_token_type_end: diff --git a/src/parse_tree.h b/src/parse_tree.h index ac46d6500..184037f1c 100644 --- a/src/parse_tree.h +++ b/src/parse_tree.h @@ -95,7 +95,7 @@ class parse_node_t { // This is used to store e.g. the statement decoration. parse_node_tag_t tag : 4; // Description - wcstring describe(void) const; + wcstring describe() const; // Constructor explicit parse_node_t(parse_token_type_t ty) From c7f16439bfda5a69e1924aeccdbe3d67b2828dad Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Sun, 4 Mar 2018 16:06:32 -0800 Subject: [PATCH 08/10] Add support for ! as an analog to 'not' ! and not are effectively interchangeable now. Mark them both as operators for syntax highlighting. --- src/fish_tests.cpp | 5 +++++ src/highlight.cpp | 5 ++++- src/parse_constants.h | 29 +++++++++++++++++++---------- src/parse_execution.cpp | 3 ++- src/parse_grammar.h | 6 ++++-- src/parse_productions.cpp | 16 +++++++++++++++- src/parse_tree.cpp | 2 +- tests/andandoror.err | 3 +++ tests/andandoror.in | 8 ++++++++ tests/andandoror.out | 9 +++++++++ 10 files changed, 70 insertions(+), 16 deletions(-) diff --git a/src/fish_tests.cpp b/src/fish_tests.cpp index 17cf062a5..ff01bdbe3 100644 --- a/src/fish_tests.cpp +++ b/src/fish_tests.cpp @@ -3961,6 +3961,11 @@ static void test_highlighting() { {L"||", highlight_spec_operator}, {L"true", highlight_spec_command}, {L";", highlight_spec_statement_terminator}, + {L"and", highlight_spec_operator}, + {L"not", highlight_spec_operator}, + {L"!", highlight_spec_operator}, + {L"true", highlight_spec_command}, + {L";", highlight_spec_statement_terminator}, {L"end", highlight_spec_command}, {NULL, -1}}; diff --git a/src/highlight.cpp b/src/highlight.cpp index 0deff0590..1c91172d7 100644 --- a/src/highlight.cpp +++ b/src/highlight.cpp @@ -1033,7 +1033,6 @@ const highlighter_t::color_array_t &highlighter_t::highlight() { case symbol_if_clause: case symbol_else_clause: case symbol_case_item: - case symbol_not_statement: case symbol_decorated_statement: case symbol_if_statement: { this->color_children(node, parse_token_type_string, highlight_spec_command); @@ -1065,6 +1064,10 @@ const highlighter_t::color_array_t &highlighter_t::highlight() { this->color_node(node, highlight_spec_operator); break; + case symbol_not_statement: + this->color_children(node, parse_token_type_string, highlight_spec_operator); + break; + case symbol_job_decorator: this->color_node(node, highlight_spec_operator); break; diff --git a/src/parse_constants.h b/src/parse_constants.h index b4f679b6a..9deb0efc3 100644 --- a/src/parse_constants.h +++ b/src/parse_constants.h @@ -111,21 +111,30 @@ enum parse_keyword_t { parse_keyword_if, parse_keyword_in, parse_keyword_not, + parse_keyword_exclam, parse_keyword_or, parse_keyword_switch, parse_keyword_while, } __packed; -const enum_map keyword_enum_map[] = { - {parse_keyword_and, L"and"}, {parse_keyword_begin, L"begin"}, - {parse_keyword_builtin, L"builtin"}, {parse_keyword_case, L"case"}, - {parse_keyword_command, L"command"}, {parse_keyword_else, L"else"}, - {parse_keyword_end, L"end"}, {parse_keyword_exec, L"exec"}, - {parse_keyword_for, L"for"}, {parse_keyword_function, L"function"}, - {parse_keyword_if, L"if"}, {parse_keyword_in, L"in"}, - {parse_keyword_not, L"not"}, {parse_keyword_or, L"or"}, - {parse_keyword_switch, L"switch"}, {parse_keyword_while, L"while"}, - {parse_keyword_none, NULL}}; +const enum_map keyword_enum_map[] = {{parse_keyword_exclam, L"!"}, + {parse_keyword_and, L"and"}, + {parse_keyword_begin, L"begin"}, + {parse_keyword_builtin, L"builtin"}, + {parse_keyword_case, L"case"}, + {parse_keyword_command, L"command"}, + {parse_keyword_else, L"else"}, + {parse_keyword_end, L"end"}, + {parse_keyword_exec, L"exec"}, + {parse_keyword_for, L"for"}, + {parse_keyword_function, L"function"}, + {parse_keyword_if, L"if"}, + {parse_keyword_in, L"in"}, + {parse_keyword_not, L"not"}, + {parse_keyword_or, L"or"}, + {parse_keyword_switch, L"switch"}, + {parse_keyword_while, L"while"}, + {parse_keyword_none, NULL}}; #define keyword_enum_map_len (sizeof keyword_enum_map / sizeof *keyword_enum_map) // Node tag values. diff --git a/src/parse_execution.cpp b/src/parse_execution.cpp index a54f6fb49..c9bcd6f58 100644 --- a/src/parse_execution.cpp +++ b/src/parse_execution.cpp @@ -924,7 +924,8 @@ bool parse_execution_context_t::determine_io_chain(tnode_t not_statement) { job->set_flag(JOB_NEGATE, !job->get_flag(JOB_NEGATE)); - return this->populate_job_process(job, proc, not_statement.child<1>()); + return this->populate_job_process(job, proc, + not_statement.require_get_child()); } template diff --git a/src/parse_grammar.h b/src/parse_grammar.h index e08ccb62d..488bb9314 100644 --- a/src/parse_grammar.h +++ b/src/parse_grammar.h @@ -312,8 +312,10 @@ DEF(function_header) produces_sequence, argument, argument_list, tok_end>{ BODY(function_header)}; -DEF(not_statement) produces_sequence, statement> { - BODY(not_statement); +DEF_ALT(not_statement) { + using nots = seq, statement>; + using exclams = seq, statement>; + ALT_BODY(not_statement, nots, exclams); }; // An andor_job_list is zero or more job lists, where each starts with an `and` or `or` boolean diff --git a/src/parse_productions.cpp b/src/parse_productions.cpp index a09e28e13..fefb0377b 100644 --- a/src/parse_productions.cpp +++ b/src/parse_productions.cpp @@ -143,7 +143,8 @@ RESOLVE(statement) { switch (token1.type) { case parse_token_type_string: { switch (token1.keyword) { - case parse_keyword_not: { + case parse_keyword_not: + case parse_keyword_exclam: { return production_for(); } case parse_keyword_for: @@ -215,6 +216,19 @@ RESOLVE(case_item_list) { return production_for(); } +RESOLVE(not_statement) { + UNUSED(token2); + UNUSED(out_tag); + switch (token1.keyword) { + case parse_keyword_not: + return production_for(); + case parse_keyword_exclam: + return production_for(); + default: + return NO_PRODUCTION; + } +} + RESOLVE(andor_job_list) { UNUSED(out_tag); diff --git a/src/parse_tree.cpp b/src/parse_tree.cpp index f1f92fa5e..9c51025f4 100644 --- a/src/parse_tree.cpp +++ b/src/parse_tree.cpp @@ -960,7 +960,7 @@ static inline parse_keyword_t keyword_with_name(const wchar_t *name) { static bool is_keyword_char(wchar_t c) { return (c >= L'a' && c <= L'z') || (c >= L'A' && c <= L'Z') || (c >= L'0' && c <= L'9') || - c == L'\'' || c == L'"' || c == L'\\' || c == '\n'; + c == L'\'' || c == L'"' || c == L'\\' || c == '\n' || c == L'!'; } /// Given a token, returns the keyword it matches, or parse_keyword_none. diff --git a/tests/andandoror.err b/tests/andandoror.err index ce5b602ba..60f61565c 100644 --- a/tests/andandoror.err +++ b/tests/andandoror.err @@ -8,5 +8,8 @@ #################### # && and || in while statements +#################### +# Nots + #################### # Complex scenarios diff --git a/tests/andandoror.in b/tests/andandoror.in index f739cc1ea..fff23d3b9 100644 --- a/tests/andandoror.in +++ b/tests/andandoror.in @@ -28,6 +28,14 @@ while [ $alpha -lt 2 ] && [ $beta -lt 3 ] set delta ( math $delta + 1 ) end +logmsg "Nots" + +true && ! false ; echo $status +not true && ! false ; echo $status +not not not true ; echo $status +not ! ! not true ; echo $status +not ! echo not ! ; echo $status + logmsg "Complex scenarios" begin; echo 1 ; false ; end || begin ; echo 2 && echo 3 ; end diff --git a/tests/andandoror.out b/tests/andandoror.out index 84f9f04f2..649267ebd 100644 --- a/tests/andandoror.out +++ b/tests/andandoror.out @@ -23,6 +23,15 @@ if test 4 ok 3 3 3 4 4 4 +#################### +# Nots +0 +1 +1 +0 +not ! +0 + #################### # Complex scenarios 1 From 9bb2d1e79f75684c55d534ffcdccd2580c93d3ba Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Sun, 4 Mar 2018 17:29:17 -0800 Subject: [PATCH 09/10] Note support for && || ! in CHANGELOG --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 56830a217..3bab1c405 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -43,6 +43,7 @@ This section is for changes merged to the `major` branch that are not also merge - The names `argparse`, `read`, `set`, `status`, `test` and `[` are now reserved and not allowed as function names. This prevents users unintentionally breaking stuff (#3000). - Wrapping completions (from `complete -w` or `function -w`) can now inject arguments. For example, `complete gco -w 'git checkout'` now works properly (#1976). The `alias` function has been updated to respect this behavior. - The `jobs` builtin now has a `-q` and `--quiet` option to silence the output. +- fish now supports `&&`, `||`, and `!` (#4620). ## Other significant changes - Command substitution output is now limited to 10 MB by default (#3822). From 0938c9f4272ae1fb2207a3e2d59f66475148fd35 Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Sun, 4 Mar 2018 20:17:28 -0800 Subject: [PATCH 10/10] Document && and || --- doc_src/faq.hdr | 2 +- doc_src/tutorial.hdr | 12 +++++++++--- lexicon_filter.in | 9 +++++---- 3 files changed, 15 insertions(+), 8 deletions(-) diff --git a/doc_src/faq.hdr b/doc_src/faq.hdr index 45756f77e..133ac438c 100644 --- a/doc_src/faq.hdr +++ b/doc_src/faq.hdr @@ -281,7 +281,7 @@ The fish user community extends fish in unique and useful ways via scripts that - Oh My Fish - Tacklebox -This is not an exhaustive list and the fish project has no opinion regarding the merits of the repositories listed above or the scripts found therein. We mention these only because you may find within them a solution to a need you have such as supporting the `&&` and `||` operators or improved integration with other tools that you use. +This is not an exhaustive list and the fish project has no opinion regarding the merits of the repositories listed above or the scripts found therein. \htmlonly[block] diff --git a/doc_src/tutorial.hdr b/doc_src/tutorial.hdr index 061792af7..832d4d95e 100644 --- a/doc_src/tutorial.hdr +++ b/doc_src/tutorial.hdr @@ -420,17 +420,23 @@ echo chips \section tut_combiners Combiners (And, Or, Not) -Unlike other shells, `fish` does not have special syntax like && or || to combine commands. Instead it has commands `and`, `or`, and `not`. +fish supports the familiar `&&` and `||` to combine commands, and `!` to negate them: \fish{cli-dark} ->_ cp file1.txt file1_bak.txt; and echo "Backup successful"; or echo "Backup failed" +>_ ./configure && make && sudo make install +\endfish + +fish also supports `and`, `or`, and `not`. The first two are job modifiers and have lower precedence. Example usage: + +\fish{cli-dark} +>_ cp file1.txt file1_bak.txt && cp file2.txt file2_bak.txt ; and echo "Backup successful"; or echo "Backup failed" Backup failed \endfish As mentioned in the section on the semicolon, this can also be written in multiple lines, like so: \fish -cp file1.txt file1_bak.txt +cp file1.txt file1_bak.txt && cp file2.txt file2_bak.txt and echo "Backup successful" or echo "Backup failed" \endfish diff --git a/lexicon_filter.in b/lexicon_filter.in index 19c33d3ff..211a5dc43 100644 --- a/lexicon_filter.in +++ b/lexicon_filter.in @@ -584,10 +584,11 @@ x # A special case. Tidy up after performing command substitution. # Redirectors s/\([^{|] *\)|/\1@redr{|}/g -s/\&@EOL$/@redr{@amp}@EOL/g -s/@amp@EOL$/@redr{@amp}@EOL/g -s/\([<>]\)@amp\([0-9]\)/@redr{\1@amp\2}/g -s/\([^{&] *\)&[^@a-z]/\1@redr{\&}/g +#s/\&@EOL$/@redr{@amp}@EOL/g +#s/@amp@EOL$/@redr{@amp}@EOL/g +#s/\([<>]\)@amp\([0-9]\)/@redr{\1@amp\2}/g +s/@amp&/@optr{@amp@amp}/g +#s/\([^{&] *\)&[^@a-z]/\1@redr{\&}/g s/\([^{<>^] *\)\([0-9]* *[<>^][<>^]*[^@][a-zA-Z0-9./_-]*\)/\1@redr{\2}/g s/\\}/}\\/g #.