/** \file parser.c The fish parser. Contains functions for parsing and evaluating code. */ #include "config.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include "fallback.h" #include "util.h" #include "common.h" #include "wutil.h" #include "proc.h" #include "parser.h" #include "parser_keywords.h" #include "tokenizer.h" #include "exec.h" #include "wildcard.h" #include "function.h" #include "builtin.h" #include "env.h" #include "expand.h" #include "reader.h" #include "sanity.h" #include "env_universal.h" #include "event.h" #include "intern.h" #include "parse_util.h" #include "path.h" #include "signal.h" #include "complete.h" /** Maximum number of function calls, i.e. recursion depth. */ #define MAX_RECURSION_DEPTH 128 /** Error message for unknown builtin */ #define UNKNOWN_BUILTIN_ERR_MSG _(L"Unknown builtin '%ls'") /** Error message for improper use of the exec builtin */ #define EXEC_ERR_MSG _(L"This command can not be used in a pipeline") /** Error message for tokenizer error. The tokenizer message is appended to this message. */ #define TOK_ERR_MSG _( L"Tokenizer error: '%ls'") /** Error message for short circuit command error. */ #define COND_ERR_MSG _( L"An additional command is required" ) /** Error message on a function that calls itself immediately */ #define INFINITE_RECURSION_ERR_MSG _( L"The function calls itself immediately, which would result in an infinite loop.") /** Error message on reaching maximum recursion depth */ #define OVERFLOW_RECURSION_ERR_MSG _( L"Maximum recursion depth reached. Accidental infinite loop?") /** Error message used when the end of a block can't be located */ #define BLOCK_END_ERR_MSG _( L"Could not locate end of block. The 'end' command is missing, misspelled or a ';' is missing.") /** Error message on reaching maximum number of block calls */ #define BLOCK_ERR_MSG _( L"Maximum number of nested blocks reached.") /** Error message when a non-string token is found when expecting a command name */ #define CMD_ERR_MSG _( L"Expected a command name, got token of type '%ls'") /** Error message when a non-string token is found when expecting a command name */ #define CMD_OR_ERR_MSG _( L"Expected a command name, got token of type '%ls'. Did you mean 'COMMAND; or COMMAND'? See the help section for the 'or' builtin command by typing 'help or'.") /** Error message when a non-string token is found when expecting a command name */ #define CMD_AND_ERR_MSG _( L"Expected a command name, got token of type '%ls'. Did you mean 'COMMAND; and COMMAND'? See the help section for the 'and' builtin command by typing 'help and'.") /** Error message when encountering an illegal command name */ #define ILLEGAL_CMD_ERR_MSG _( L"Illegal command name '%ls'") /** Error message when encountering an illegal file descriptor */ #define ILLEGAL_FD_ERR_MSG _( L"Illegal file descriptor '%ls'") /** Error message for wildcards with no matches */ #define WILDCARD_ERR_MSG _( L"No matches for wildcard '%ls'.") /** Error when using case builtin outside of switch block */ #define INVALID_CASE_ERR_MSG _( L"'case' builtin not inside of switch block") /** Error when using loop control builtins (break or continue) outside of loop */ #define INVALID_LOOP_ERR_MSG _( L"Loop control command while not inside of loop" ) /** Error when using return builtin outside of function definition */ #define INVALID_RETURN_ERR_MSG _( L"'return' builtin command outside of function definition" ) /** Error when using else builtin outside of if block */ #define INVALID_ELSE_ERR_MSG _( L"'%ls' builtin not inside of if block" ) /** Error when using 'else if' past a naked 'else' */ #define INVALID_ELSEIF_PAST_ELSE_ERR_MSG _( L"'%ls' used past terminating 'else'" ) /** Error when using end builtin outside of block */ #define INVALID_END_ERR_MSG _( L"'end' command outside of block") /** Error message for Posix-style assignment: foo=bar */ #define COMMAND_ASSIGN_ERR_MSG _( L"Unknown command '%ls'. Did you mean 'set %ls %ls'? See the help section on the set command by typing 'help set'.") /** Error for invalid redirection token */ #define REDIRECT_TOKEN_ERR_MSG _( L"Expected redirection specification, got token of type '%ls'") /** Error when encountering redirection without a command */ #define INVALID_REDIRECTION_ERR_MSG _( L"Encountered redirection when expecting a command name. Fish does not allow a redirection operation before a command.") /** Error for evaluating null pointer */ #define EVAL_NULL_ERR_MSG _( L"Tried to evaluate null pointer." ) /** Error for evaluating in illegal scope */ #define INVALID_SCOPE_ERR_MSG _( L"Tried to evaluate commands using invalid block type '%ls'" ) /** Error for wrong token type */ #define UNEXPECTED_TOKEN_ERR_MSG _( L"Unexpected token of type '%ls'") /** While block description */ #define WHILE_BLOCK N_( L"'while' block" ) /** For block description */ #define FOR_BLOCK N_( L"'for' block" ) /** Breakpoint block */ #define BREAKPOINT_BLOCK N_( L"Block created by breakpoint" ) /** If block description */ #define IF_BLOCK N_( L"'if' conditional block" ) /** Function definition block description */ #define FUNCTION_DEF_BLOCK N_( L"function definition block" ) /** Function invocation block description */ #define FUNCTION_CALL_BLOCK N_( L"function invocation block" ) /** Function invocation block description */ #define FUNCTION_CALL_NO_SHADOW_BLOCK N_( L"function invocation block with no variable shadowing" ) /** Switch block description */ #define SWITCH_BLOCK N_( L"'switch' block" ) /** Fake block description */ #define FAKE_BLOCK N_( L"unexecutable block" ) /** Top block description */ #define TOP_BLOCK N_( L"global root block" ) /** Command substitution block description */ #define SUBST_BLOCK N_( L"command substitution block" ) /** Begin block description */ #define BEGIN_BLOCK N_( L"'begin' unconditional block" ) /** Source block description */ #define SOURCE_BLOCK N_( L"Block created by the . builtin" ) /** Source block description */ #define EVENT_BLOCK N_( L"event handler block" ) /** Unknown block description */ #define UNKNOWN_BLOCK N_( L"unknown/invalid block" ) /** Datastructure to describe a block type, like while blocks, command substitution blocks, etc. */ struct block_lookup_entry { /** The block type id. The legal values are defined in parser.h. */ block_type_t type; /** The name of the builtin that creates this type of block, if any. */ const wchar_t *name; /** A description of this block type */ const wchar_t *desc; } ; /** List of all legal block types */ static const struct block_lookup_entry block_lookup[]= { { WHILE, L"while", WHILE_BLOCK }, { FOR, L"for", FOR_BLOCK }, { IF, L"if", IF_BLOCK }, { FUNCTION_DEF, L"function", FUNCTION_DEF_BLOCK }, { FUNCTION_CALL, 0, FUNCTION_CALL_BLOCK }, { FUNCTION_CALL_NO_SHADOW, 0, FUNCTION_CALL_NO_SHADOW_BLOCK }, { SWITCH, L"switch", SWITCH_BLOCK }, { FAKE, 0, FAKE_BLOCK }, { TOP, 0, TOP_BLOCK }, { SUBST, 0, SUBST_BLOCK }, { BEGIN, L"begin", BEGIN_BLOCK }, { SOURCE, L".", SOURCE_BLOCK }, { EVENT, 0, EVENT_BLOCK }, { BREAKPOINT, L"breakpoint", BREAKPOINT_BLOCK }, { (block_type_t)0, 0, 0 } }; static bool job_should_skip_elseif(const job_t *job, const block_t *current_block); parser_t::parser_t(enum parser_type_t type, bool errors) : parser_type(type), show_errors(errors), error_code(0), err_pos(0), current_tokenizer(NULL), current_tokenizer_pos(0), job_start_pos(0), eval_level(-1), current_block(NULL), block_io(shared_ptr()) { } /* A pointer to the principal parser (which is a static local) */ static parser_t *s_principal_parser = NULL; parser_t &parser_t::principal_parser(void) { ASSERT_IS_NOT_FORKED_CHILD(); ASSERT_IS_MAIN_THREAD(); static parser_t parser(PARSER_TYPE_GENERAL, true); if (! s_principal_parser) { s_principal_parser = &parser; } return parser; } void parser_t::skip_all_blocks(void) { /* Tell all blocks to skip */ if (s_principal_parser) { //write(2, "Cancelling blocks\n", strlen("Cancelling blocks\n")); block_t *c = s_principal_parser->current_block; while (c) { c->skip = true; //fprintf(stderr, " Cancelled %p\n", c); c = c->outer; } } } void parser_t::push_block(block_t *newv) { const enum block_type_t type = newv->type(); newv->src_lineno = parser_t::get_lineno(); newv->src_filename = parser_t::current_filename()?intern(parser_t::current_filename()):0; newv->outer = current_block; if (current_block && current_block->skip) newv->mark_as_fake(); /* New blocks should be skipped if the outer block is skipped, except TOP ans SUBST block, which open up new environments. Fake blocks should always be skipped. Rather complicated... :-( */ newv->skip=current_block?current_block->skip:0; /* Type TOP and SUBST are never skipped */ if (type == TOP || type == SUBST) { newv->skip = 0; } /* Fake blocks and function definition blocks are never executed */ if (type == FAKE || type == FUNCTION_DEF) { newv->skip = 1; } newv->job = 0; newv->loop_status=LOOP_NORMAL; current_block = newv; if ((newv->type() != FUNCTION_DEF) && (newv->type() != FAKE) && (newv->type() != TOP)) { env_push(type == FUNCTION_CALL); newv->wants_pop_env = true; } } void parser_t::pop_block() { block_t *old = current_block; if (!current_block) { debug(1, L"function %s called on empty block stack.", __func__); bugreport(); return; } current_block = current_block->outer; if (old->wants_pop_env) env_pop(); delete old; } const wchar_t *parser_t::get_block_desc(int block) const { for (size_t i=0; block_lookup[i].desc; i++) { if (block_lookup[i].type == block) { return _(block_lookup[i].desc); } } return _(UNKNOWN_BLOCK); } /** Returns 1 if the specified command is a builtin that may not be used in a pipeline */ static int parser_is_pipe_forbidden(const wcstring &word) { return contains(word, L"exec", L"case", L"break", L"return", L"continue"); } /** Search the text for the end of the current block */ static const wchar_t *parser_find_end(const wchar_t * buff) { int had_cmd=0; int count = 0; int error=0; int mark=0; CHECK(buff, 0); tokenizer_t tok(buff, 0); for (; tok_has_next(&tok) && !error; tok_next(&tok)) { int last_type = tok_last_type(&tok); switch (last_type) { case TOK_STRING: { if (!had_cmd) { if (wcscmp(tok_last(&tok), L"end")==0) { count--; } else if (parser_keywords_is_block(tok_last(&tok))) { count++; } if (count < 0) { error = 1; } had_cmd = 1; } break; } case TOK_END: { had_cmd = 0; break; } case TOK_PIPE: case TOK_BACKGROUND: { if (had_cmd) { had_cmd = 0; } else { error = 1; } break; } case TOK_ERROR: error = 1; break; default: break; } if (!count) { tok_next(&tok); mark = tok_get_pos(&tok); break; } } if (!count && !error) { return buff+mark; } return 0; } void parser_t::forbid_function(const wcstring &function) { forbidden_function.push_back(function); } void parser_t::allow_function() { /* if( al_peek( &forbidden_function) ) debug( 2, L"Allow %ls\n", al_peek( &forbidden_function) ); */ forbidden_function.pop_back(); } void parser_t::error(int ec, int p, const wchar_t *str, ...) { va_list va; CHECK(str,); error_code = ec; err_pos = p; va_start(va, str); err_buff = vformat_string(str, va); va_end(va); } /** Print profiling information to the specified stream */ static void print_profile(const std::vector &items, FILE *out) { size_t pos; for (pos = 0; pos < items.size(); pos++) { const profile_item_t *me, *prev; size_t i; int my_time; me = items.at(pos); if (!me->skipped) { my_time=me->parse+me->exec; for (i=pos+1; iskipped) { continue; } if (prev->level <= me->level) { break; } if (prev->level > me->level+1) { continue; } my_time -= prev->parse; my_time -= prev->exec; } if (me->cmd.size() > 0) { if (fwprintf(out, L"%d\t%d\t", my_time, me->parse+me->exec) < 0) { wperror(L"fwprintf"); return; } for (i=0; ilevel; i++) { if (fwprintf(out, L"-") < 0) { wperror(L"fwprintf"); return; } } if (fwprintf(out, L"> %ls\n", me->cmd.c_str()) < 0) { wperror(L"fwprintf"); return; } } delete me; } } } void parser_t::destroy() { if (profile) { /* Save profiling information. OK to not use CLO_EXEC here because this is called while fish is dying (and hence will not fork) */ FILE *f = fopen(profile, "w"); if (!f) { debug(1, _(L"Could not write profiling information to file '%s'"), profile); } else { if (fwprintf(f, _(L"Time\tSum\tCommand\n"), profile_items.size()) < 0) { wperror(L"fwprintf"); } else { print_profile(profile_items, f); } if (fclose(f)) { wperror(L"fclose"); } } } lineinfo.clear(); forbidden_function.clear(); } /** Print error message to string if an error has occured while parsing \param target the buffer to write to \param prefix: The string token to prefix the each line with. Usually the name of the command trying to parse something. */ void parser_t::print_errors(wcstring &target, const wchar_t *prefix) { CHECK(prefix,); if (error_code && ! err_buff.empty()) { int tmp; append_format(target, L"%ls: %ls\n", prefix, err_buff.c_str()); tmp = current_tokenizer_pos; current_tokenizer_pos = err_pos; append_format(target, L"%ls", this->current_line()); current_tokenizer_pos=tmp; } } /** Print error message to stderr if an error has occured while parsing */ void parser_t::print_errors_stderr() { if (error_code && ! err_buff.empty()) { debug(0, L"%ls", err_buff.c_str()); int tmp; tmp = current_tokenizer_pos; current_tokenizer_pos = err_pos; fwprintf(stderr, L"%ls", this->current_line()); current_tokenizer_pos=tmp; } } void parser_t::eval_args(const wchar_t *line, std::vector &args) { expand_flags_t eflags = 0; if (! show_errors) eflags |= EXPAND_NO_DESCRIPTIONS; if (this->parser_type != PARSER_TYPE_GENERAL) eflags |= EXPAND_SKIP_CMDSUBST; bool do_loop=1; if (! line) return; // PCA we need to suppress calling proc_push_interactive off of the main thread. I'm not sure exactly what it does. if (this->parser_type == PARSER_TYPE_GENERAL) proc_push_interactive(0); tokenizer_t tok(line, (show_errors ? 0 : TOK_SQUASH_ERRORS)); /* eval_args may be called while evaulating another command, so we save the previous tokenizer and restore it on exit */ scoped_push tokenizer_push(¤t_tokenizer, &tok); scoped_push tokenizer_pos_push(¤t_tokenizer_pos, 0); error_code=0; for (; do_loop && tok_has_next(&tok) ; tok_next(&tok)) { current_tokenizer_pos = tok_get_pos(&tok); switch (tok_last_type(&tok)) { case TOK_STRING: { const wcstring tmp = tok_last(&tok); if (expand_string(tmp, args, eflags) == EXPAND_ERROR) { err_pos=tok_get_pos(&tok); do_loop=0; } break; } case TOK_END: { break; } case TOK_ERROR: { if (show_errors) error(SYNTAX_ERROR, tok_get_pos(&tok), TOK_ERR_MSG, tok_last(&tok)); do_loop=0; break; } default: { if (show_errors) error(SYNTAX_ERROR, tok_get_pos(&tok), UNEXPECTED_TOKEN_ERR_MSG, tok_get_desc(tok_last_type(&tok))); do_loop=0; break; } } } if (show_errors) this->print_errors_stderr(); if (this->parser_type == PARSER_TYPE_GENERAL) proc_pop_interactive(); } void parser_t::stack_trace(block_t *b, wcstring &buff) { /* Check if we should end the recursion */ if (!b) return; if (b->type()==EVENT) { /* This is an event handler */ const event_block_t *eb = static_cast(b); wcstring description = event_get_desc(eb->event); append_format(buff, _(L"in event handler: %ls\n"), description.c_str()); buff.append(L"\n"); /* Stop recursing at event handler. No reason to believe that any other code is relevant. It might make sense in the future to continue printing the stack trace of the code that invoked the event, if this is a programmatic event, but we can't currently detect that. */ return; } if (b->type() == FUNCTION_CALL || b->type()==SOURCE || b->type()==SUBST) { /* These types of blocks should be printed */ int i; switch (b->type()) { case SOURCE: { const source_block_t *sb = static_cast(b); const wchar_t *source_dest = sb->source_file; append_format(buff, _(L"in . (source) call of file '%ls',\n"), source_dest); break; } case FUNCTION_CALL: { const function_block_t *fb = static_cast(b); append_format(buff, _(L"in function '%ls',\n"), fb->name.c_str()); break; } case SUBST: { append_format(buff, _(L"in command substitution\n")); break; } default: /* Can't get here */ break; } const wchar_t *file = b->src_filename; if (file) { append_format(buff, _(L"\tcalled on line %d of file '%ls',\n"), b->src_lineno, file); } else { append_format(buff, _(L"\tcalled on standard input,\n")); } if (b->type() == FUNCTION_CALL) { const function_block_t *fb = static_cast(b); const process_t * const process = fb->process; if (process->argv(1)) { wcstring tmp; for (i=1; process->argv(i); i++) { if (i > 1) tmp.push_back(L' '); tmp.append(process->argv(i)); } append_format(buff, _(L"\twith parameter list '%ls'\n"), tmp.c_str()); } } append_format(buff, L"\n"); } /* Recursively print the next block */ parser_t::stack_trace(b->outer, buff); } /** Returns the name of the currently evaluated function if we are currently evaluating a function, null otherwise. This is tested by moving down the block-scope-stack, checking every block if it is of type FUNCTION_CALL. */ const wchar_t *parser_t::is_function() const { // PCA: Have to make this a string somehow ASSERT_IS_MAIN_THREAD(); wcstring result; block_t *b = current_block; while (1) { if (!b) { return NULL; } if (b->type() == FUNCTION_CALL) { const function_block_t *fb = static_cast(b); return fb->name.c_str(); } b=b->outer; } } int parser_t::get_lineno() const { int lineno; if (! current_tokenizer || ! tok_string(current_tokenizer)) return -1; lineno = current_tokenizer->line_number_of_character_at_offset(current_tokenizer_pos); const wchar_t *function_name; if ((function_name = is_function())) { lineno += function_get_definition_offset(function_name); } return lineno; } int parser_t::line_number_of_character_at_offset(size_t idx) const { if (! current_tokenizer) return -1; int result = current_tokenizer->line_number_of_character_at_offset(idx); //assert(result == parse_util_lineno(tok_string( current_tokenizer ), idx)); return result; } const wchar_t *parser_t::current_filename() const { /* We query a global array for the current file name, so it only makes sense to ask this on the principal parser. */ ASSERT_IS_MAIN_THREAD(); assert(this == &principal_parser()); block_t *b = current_block; while (1) { if (!b) { return reader_current_filename(); } if (b->type() == FUNCTION_CALL) { const function_block_t *fb = static_cast(b); return function_get_definition_file(fb->name); } b=b->outer; } } /** Calculates the on-screen width of the specified substring of the specified string. This function takes into account the width and alignment of the tab character, but other wise behaves like repeatedly calling wcwidth. */ static int printed_width(const wchar_t *str, int len) { int res=0; int i; CHECK(str, 0); for (i=0; str[i] && i= (size_t)min_match && (wcsncmp(L"--help", s, len) == 0)); } job_t *parser_t::job_create() { job_t *res = new job_t(acquire_job_id(), this->block_io); this->my_job_list.push_front(res); job_set_flag(res, JOB_CONTROL, (job_control_mode==JOB_CONTROL_ALL) || ((job_control_mode == JOB_CONTROL_INTERACTIVE) && (get_is_interactive()))); return res; } bool parser_t::job_remove(job_t *j) { job_list_t::iterator iter = std::find(my_job_list.begin(), my_job_list.end(), j); if (iter != my_job_list.end()) { my_job_list.erase(iter); return true; } else { debug(1, _(L"Job inconsistency")); sanity_lose(); return false; } } void parser_t::job_promote(job_t *job) { signal_block(); job_list_t::iterator loc = std::find(my_job_list.begin(), my_job_list.end(), job); assert(loc != my_job_list.end()); /* Move the job to the beginning */ my_job_list.splice(my_job_list.begin(), my_job_list, loc); signal_unblock(); } job_t *parser_t::job_get(job_id_t id) { job_iterator_t jobs(my_job_list); job_t *job; while ((job = jobs.next())) { if (id <= 0 || job->job_id == id) return job; } return NULL; } job_t *parser_t::job_get_from_pid(int pid) { job_iterator_t jobs; job_t *job; while ((job = jobs.next())) { if (job->pgid == pid) return job; } return 0; } /** Parse options for the specified job \param p the process to parse options for \param j the job to which the process belongs to \param tok the tokenizer to read options from \param args the argument list to insert options into \param args unskip whether we should ignore current_block->skip. Big hack because of our dumb handling of if statements. */ void parser_t::parse_job_argument_list(process_t *p, job_t *j, tokenizer_t *tok, std::vector &args, bool unskip) { int is_finished=0; int proc_is_count=0; int matched_wildcard = 0, unmatched_wildcard = 0; wcstring unmatched; int unmatched_pos=0; /* The set of IO redirections that we construct for the process */ io_chain_t process_io_chain; /* Test if this is the 'count' command. We need to special case count in the shell, since it should display a help message on 'count -h', but not on 'set foo -h; count $foo'. This is an ugly workaround and a huge hack, but as near as I can tell, the alternatives are worse. */ proc_is_count = (args.at(0).completion == L"count"); while (1) { switch (tok_last_type(tok)) { case TOK_PIPE: { wchar_t *end; if (p->type == INTERNAL_EXEC) { error(SYNTAX_ERROR, tok_get_pos(tok), EXEC_ERR_MSG); return; } errno = 0; p->pipe_write_fd = fish_wcstoi(tok_last(tok), &end, 10); if (p->pipe_write_fd < 0 || errno || *end) { error(SYNTAX_ERROR, tok_get_pos(tok), ILLEGAL_FD_ERR_MSG, tok_last(tok)); return; } p->set_argv(completions_to_wcstring_list(args)); p->next = new process_t(); tok_next(tok); /* Don't do anything on failure. parse_job will notice the error flag and report any errors for us */ parse_job(p->next, j, tok); is_finished = 1; break; } case TOK_BACKGROUND: { job_set_flag(j, JOB_FOREGROUND, 0); // PCA note fall through, this is deliberate. The background modifier & terminates a command } case TOK_END: { if (!p->get_argv()) p->set_argv(completions_to_wcstring_list(args)); if (tok_has_next(tok)) tok_next(tok); is_finished = 1; break; } case TOK_STRING: { int skip=0; if (job_get_flag(j, JOB_SKIP)) { skip = 1; } else if (current_block->skip && ! unskip) { /* If this command should be skipped, we do not expand the arguments */ skip=1; /* But if this is in fact a case statement or an elseif statement, then it should be evaluated */ block_type_t type = current_block->type(); if (type == SWITCH && args.at(0).completion == L"case" && p->type == INTERNAL_BUILTIN) { skip=0; } else if (job_get_flag(j, JOB_ELSEIF) && ! job_should_skip_elseif(j, current_block)) { skip=0; } } else { /* If this is an else if, and we should skip it, then don't expand any arguments */ if (job_get_flag(j, JOB_ELSEIF) && job_should_skip_elseif(j, current_block)) { skip = 1; } } if (!skip) { if ((proc_is_count) && (args.size() == 1) && (parser_t::is_help(tok_last(tok), 0)) && (p->type == INTERNAL_BUILTIN)) { /* Display help for count */ p->count_help_magic = 1; } switch (expand_string(tok_last(tok), args, 0)) { case EXPAND_ERROR: { err_pos=tok_get_pos(tok); if (error_code == 0) { error(SYNTAX_ERROR, tok_get_pos(tok), _(L"Could not expand string '%ls'"), tok_last(tok)); } break; } case EXPAND_WILDCARD_NO_MATCH: { unmatched_wildcard = 1; if (unmatched.empty()) { unmatched = tok_last(tok); unmatched_pos = tok_get_pos(tok); } break; } case EXPAND_WILDCARD_MATCH: { matched_wildcard = 1; break; } case EXPAND_OK: { break; } } } break; } case TOK_REDIRECT_OUT: case TOK_REDIRECT_IN: case TOK_REDIRECT_APPEND: case TOK_REDIRECT_FD: case TOK_REDIRECT_NOCLOB: { int type = tok_last_type(tok); shared_ptr new_io; wcstring target; bool has_target = false; wchar_t *end; /* Don't check redirections in skipped part Otherwise, bogus errors may be the result. (Do check that token is string, though) */ if (current_block->skip && ! unskip) { tok_next(tok); if (tok_last_type(tok) != TOK_STRING) { error(SYNTAX_ERROR, tok_get_pos(tok), REDIRECT_TOKEN_ERR_MSG, tok_get_desc(tok_last_type(tok))); } break; } errno = 0; int fd = fish_wcstoi(tok_last(tok), &end, 10); if (fd < 0 || errno || *end) { error(SYNTAX_ERROR, tok_get_pos(tok), ILLEGAL_FD_ERR_MSG, tok_last(tok)); } else { tok_next(tok); switch (tok_last_type(tok)) { case TOK_STRING: { target = tok_last(tok); has_target = expand_one(target, no_exec ? EXPAND_SKIP_VARIABLES : 0); if (! has_target && error_code == 0) { error(SYNTAX_ERROR, tok_get_pos(tok), REDIRECT_TOKEN_ERR_MSG, tok_last(tok)); } break; } default: error(SYNTAX_ERROR, tok_get_pos(tok), REDIRECT_TOKEN_ERR_MSG, tok_get_desc(tok_last_type(tok))); } if (! has_target || target.empty()) { if (error_code == 0) error(SYNTAX_ERROR, tok_get_pos(tok), _(L"Invalid IO redirection")); tok_next(tok); } else if (type == TOK_REDIRECT_FD) { if (target == L"-") { new_io.reset(new io_close_t(fd)); } else { wchar_t *end; errno = 0; int old_fd = fish_wcstoi(target.c_str(), &end, 10); if (old_fd < 0 || errno || *end) { error(SYNTAX_ERROR, tok_get_pos(tok), _(L"Requested redirection to something that is not a file descriptor %ls"), target.c_str()); tok_next(tok); } else { new_io.reset(new io_fd_t(fd, old_fd)); } } } else { int flags = 0; switch (type) { case TOK_REDIRECT_APPEND: flags = O_CREAT | O_APPEND | O_WRONLY; break; case TOK_REDIRECT_OUT: flags = O_CREAT | O_WRONLY | O_TRUNC; break; case TOK_REDIRECT_NOCLOB: flags = O_CREAT | O_EXCL | O_WRONLY; break; case TOK_REDIRECT_IN: flags = O_RDONLY; break; } io_file_t *new_io_file = new io_file_t(fd, target, flags); new_io.reset(new_io_file); } } if (new_io.get() != NULL) { process_io_chain.push_back(new_io); } } break; case TOK_ERROR: { error(SYNTAX_ERROR, tok_get_pos(tok), TOK_ERR_MSG, tok_last(tok)); return; } default: error(SYNTAX_ERROR, tok_get_pos(tok), UNEXPECTED_TOKEN_ERR_MSG, tok_get_desc(tok_last_type(tok))); tok_next(tok); break; } if ((is_finished) || (error_code != 0)) break; tok_next(tok); } if (!error_code) { if (unmatched_wildcard && !matched_wildcard) { job_set_flag(j, JOB_WILDCARD_ERROR, 1); proc_set_last_status(STATUS_UNMATCHED_WILDCARD); if (get_is_interactive() && !is_block) { int tmp; debug(1, WILDCARD_ERR_MSG, unmatched.c_str()); tmp = current_tokenizer_pos; current_tokenizer_pos = unmatched_pos; fwprintf(stderr, L"%ls", parser_t::current_line()); current_tokenizer_pos=tmp; } } } /* Store our IO chain. The existing chain should be empty. */ assert(p->io_chain().empty()); p->set_io_chain(process_io_chain); } /* static void print_block_stack( block_t *b ) { if( !b ) return; print_block_stack( b->outer ); debug( 0, L"Block type %ls, skip: %d", parser_get_block_desc( b->type ), b->skip ); } */ /** Fully parse a single job. Does not call exec on it, but any command substitutions in the job will be executed. \param p The process structure that should be used to represent the first process in the job. \param j The job structure to contain the parsed job \param tok tokenizer to read from f \return 1 on success, 0 on error */ int parser_t::parse_job(process_t *p, job_t *j, tokenizer_t *tok) { std::vector args; // The list that will become the argv array for the program int use_function = 1; // May functions be considered when checking what action this command represents int use_builtin = 1; // May builtins be considered when checking what action this command represents int use_command = 1; // May commands be considered when checking what action this command represents int is_new_block=0; // Does this command create a new block? bool unskip = false; // Maybe we are an elseif inside an if block; if so we may want to evaluate this even if the if block is currently set to skip bool allow_bogus_command = false; // If we are an elseif that will not be executed, or an AND or OR that will have been short circuited, don't complain about non-existent commands block_t *prev_block = current_block; scoped_push tokenizer_pos_push(¤t_tokenizer_pos, tok_get_pos(tok)); while (args.empty()) { wcstring nxt; bool has_nxt = false; bool consumed = false; // Set to one if the command requires a second command, like e.g. while does int mark; // Use to save the position of the beginning of the token switch (tok_last_type(tok)) { case TOK_STRING: { nxt = tok_last(tok); has_nxt = expand_one(nxt, EXPAND_SKIP_CMDSUBST | EXPAND_SKIP_VARIABLES); if (! has_nxt) { error(SYNTAX_ERROR, tok_get_pos(tok), ILLEGAL_CMD_ERR_MSG, tok_last(tok)); return 0; } break; } case TOK_ERROR: { error(SYNTAX_ERROR, tok_get_pos(tok), TOK_ERR_MSG, tok_last(tok)); return 0; } case TOK_PIPE: { const wchar_t *str = tok_string(tok); if (tok_get_pos(tok)>0 && str[tok_get_pos(tok)-1] == L'|') { error(SYNTAX_ERROR, tok_get_pos(tok), CMD_OR_ERR_MSG, tok_get_desc(tok_last_type(tok))); } else { error(SYNTAX_ERROR, tok_get_pos(tok), CMD_ERR_MSG, tok_get_desc(tok_last_type(tok))); } return 0; } default: { error(SYNTAX_ERROR, tok_get_pos(tok), CMD_ERR_MSG, tok_get_desc(tok_last_type(tok))); return 0; } } mark = tok_get_pos(tok); if (contains(nxt, L"command", L"builtin", L"not", L"and", L"or", L"exec")) { int sw; int is_exec = nxt == L"exec"; if (is_exec && (p != j->first_process)) { error(SYNTAX_ERROR, tok_get_pos(tok), EXEC_ERR_MSG); return 0; } tok_next(tok); sw = parser_keywords_is_switch(tok_last(tok)); if (sw == ARG_SWITCH) { tok_set_pos(tok, mark); } else { if (sw == ARG_SKIP) { tok_next(tok); } consumed = true; if (nxt == L"command" || nxt == L"builtin") { use_function = 0; if (nxt == L"command") { use_builtin = 0; use_command = 1; } else { use_builtin = 1; use_command = 0; } } else if (nxt == L"not") { job_set_flag(j, JOB_NEGATE, !job_get_flag(j, JOB_NEGATE)); } else if (nxt == L"and") { bool skip = (proc_get_last_status() != 0); job_set_flag(j, JOB_SKIP, skip); allow_bogus_command = skip; } else if (nxt == L"or") { bool skip = (proc_get_last_status() == 0); job_set_flag(j, JOB_SKIP, skip); allow_bogus_command = skip; } else if (is_exec) { use_function = 0; use_builtin=0; p->type=INTERNAL_EXEC; tokenizer_pos_push.restore(); } } } else if (nxt == L"while") { bool new_block = false; tok_next(tok); while_block_t *wb = NULL; if ((current_block->type() != WHILE)) { new_block = true; } else if ((wb = static_cast(current_block))->status == WHILE_TEST_AGAIN) { wb->status = WHILE_TEST_FIRST; } else { new_block = true; } if (new_block) { while_block_t *wb = new while_block_t(); wb->status = WHILE_TEST_FIRST; wb->tok_pos = mark; this->push_block(wb); } consumed = true; is_new_block=1; } else if (nxt == L"if") { tok_next(tok); if_block_t *ib = new if_block_t(); this->push_block(ib); ib->tok_pos = mark; is_new_block=1; consumed = true; } else if (nxt == L"else") { /* Record where the else is for error reporting */ const int else_pos = tok_get_pos(tok); /* See if we have any more arguments, that is, whether we're ELSE IF ... or just ELSE. */ tok_next(tok); if (tok_last_type(tok) == TOK_STRING && current_block->type() == IF) { const if_block_t *ib = static_cast(current_block); /* If we've already encountered an else, complain */ if (ib->else_evaluated) { error(SYNTAX_ERROR, else_pos, INVALID_ELSEIF_PAST_ELSE_ERR_MSG, L"else if"); } else { job_set_flag(j, JOB_ELSEIF, 1); consumed = true; /* We're at the IF. Go past it. */ tok_next(tok); /* We want to execute this ELSEIF if the IF expression was evaluated, it failed, and so has every other ELSEIF (if any) */ unskip = (ib->if_expr_evaluated && ! ib->any_branch_taken); /* But if we're not executing it, don't complain about its command if it doesn't exist */ if (! unskip) allow_bogus_command = true; } } } /* Test if we need another command */ if (consumed) { /* Yes we do, around in the loop for another lap, then! */ continue; } if (use_function && (unskip || ! current_block->skip)) { bool nxt_forbidden=false; wcstring forbid; int is_function_call=0; /* This is a bit fragile. It is a test to see if we are inside of function call, but not inside a block in that function call. If, in the future, the rules for what block scopes are pushed on function invocation changes, then this check will break. */ if ((current_block->type() == TOP) && (current_block->outer) && (current_block->outer->type() == FUNCTION_CALL)) is_function_call = 1; /* If we are directly in a function, and this is the first command of the block, then the function we are executing may not be called, since that would mean an infinite recursion. */ if (is_function_call && !current_block->had_command) { forbid = forbidden_function.empty() ? wcstring(L"") : forbidden_function.back(); if (forbid == nxt) { /* Infinite recursive loop */ nxt_forbidden = true; error(SYNTAX_ERROR, tok_get_pos(tok), INFINITE_RECURSION_ERR_MSG); } } if (!nxt_forbidden && has_nxt && function_exists(nxt)) { /* Check if we have reached the maximum recursion depth */ if (forbidden_function.size() > MAX_RECURSION_DEPTH) { error(SYNTAX_ERROR, tok_get_pos(tok), OVERFLOW_RECURSION_ERR_MSG); } else { p->type = INTERNAL_FUNCTION; } } } args.push_back(completion_t(nxt)); } if (error_code == 0) { if (!p->type) { if (use_builtin && builtin_exists(args.at(0).completion)) { p->type = INTERNAL_BUILTIN; is_new_block |= parser_keywords_is_block(args.at(0).completion); } } if ((!p->type || (p->type == INTERNAL_EXEC))) { /* If we are not executing the current block, allow non-existent commands. */ if (current_block->skip && ! unskip) allow_bogus_command = true; //note this may already be true for other reasons if (allow_bogus_command) { p->actual_cmd.clear(); } else { int err; bool has_command = path_get_path(args.at(0).completion, &p->actual_cmd); err = errno; bool use_implicit_cd = false; if (! has_command) { /* If the specified command does not exist, try using an implicit cd. */ wcstring implicit_cd_path; use_implicit_cd = path_can_be_implicit_cd(args.at(0).completion, &implicit_cd_path); if (use_implicit_cd) { args.clear(); args.push_back(completion_t(L"cd")); args.push_back(completion_t(implicit_cd_path)); /* If we have defined a wrapper around cd, use it, otherwise use the cd builtin */ if (use_function && function_exists(L"cd")) p->type = INTERNAL_FUNCTION; else p->type = INTERNAL_BUILTIN; } } // Disabled pending discussion in https://github.com/fish-shell/fish-shell/issues/367 #if 0 if (! has_command && ! use_implicit_cd) { if (fish_openSUSE_dbus_hack_hack_hack_hack(&args)) { has_command = true; p->type = INTERNAL_BUILTIN; } } #endif /* Check if the specified command exists */ if (! has_command && ! use_implicit_cd) { const wchar_t *cmd = args.at(0).completion.c_str(); /* We couldn't find the specified command. What we want to happen now is that the specified job won't get executed, and an error message is printed on-screen, but otherwise, the parsing/execution of the file continues. Because of this, we don't want to call error(), since that would stop execution of the file. Instead we let p->actual_command be 0 (null), which will cause the job to silently not execute. We also print an error message and set the status to 127 (This is the standard number for this, used by other shells like bash and zsh). */ const wchar_t * const equals_ptr = wcschr(cmd, L'='); if (equals_ptr != NULL) { /* Try to figure out if this is a pure variable assignment (foo=bar), or if this appears to be running a command (foo=bar ruby...) */ const wcstring name_str = wcstring(cmd, equals_ptr - cmd); //variable name, up to the = const wcstring val_str = wcstring(equals_ptr + 1); //variable value, past the = wcstring next_str; if (tok_peek_next(tok, &next_str) == TOK_STRING && ! next_str.empty()) { wcstring ellipsis_str = wcstring(1, ellipsis_char); if (ellipsis_str == L"$") ellipsis_str = L"..."; /* Looks like a command */ debug(0, _(L"Unknown command '%ls'. Did you mean to run %ls with a modified environment? Try 'env %ls=%ls %ls%ls'. See the help section on the set command by typing 'help set'."), cmd, next_str.c_str(), name_str.c_str(), val_str.c_str(), next_str.c_str(), ellipsis_str.c_str()); } else { debug(0, COMMAND_ASSIGN_ERR_MSG, cmd, name_str.c_str(), val_str.c_str()); } } else if (cmd[0]==L'$' || cmd[0] == VARIABLE_EXPAND || cmd[0] == VARIABLE_EXPAND_SINGLE) { const env_var_t val_wstr = env_get_string(cmd+1); const wchar_t *val = val_wstr.missing() ? NULL : val_wstr.c_str(); if (val) { debug(0, _(L"Variables may not be used as commands. Instead, define a function like 'function %ls; %ls $argv; end' or use the eval builtin instead, like 'eval %ls'. See the help section for the function command by typing 'help function'."), cmd+1, val, cmd, cmd); } else { debug(0, _(L"Variables may not be used as commands. Instead, define a function or use the eval builtin instead, like 'eval %ls'. See the help section for the function command by typing 'help function'."), cmd, cmd); } } else if (wcschr(cmd, L'$')) { debug(0, _(L"Commands may not contain variables. Use the eval builtin instead, like 'eval %ls'. See the help section for the eval command by typing 'help eval'."), cmd, cmd); } else if (err!=ENOENT) { debug(0, _(L"The file '%ls' is not executable by this user"), cmd?cmd:L"UNKNOWN"); } else { /* Handle unrecognized commands with standard command not found handler that can make better error messages */ wcstring_list_t event_args; event_args.push_back(args.at(0).completion); event_fire_generic(L"fish_command_not_found", &event_args); } int tmp = current_tokenizer_pos; current_tokenizer_pos = tok_get_pos(tok); fwprintf(stderr, L"%ls", parser_t::current_line()); current_tokenizer_pos=tmp; job_set_flag(j, JOB_SKIP, 1); proc_set_last_status(err==ENOENT?STATUS_UNKNOWN_COMMAND:STATUS_NOT_EXECUTABLE); } } } if ((p->type == EXTERNAL) && !use_command) { error(SYNTAX_ERROR, tok_get_pos(tok), UNKNOWN_BUILTIN_ERR_MSG, args.back().completion.c_str()); } } if (is_new_block) { const wchar_t *end=parser_find_end(tok_string(tok) + current_tokenizer_pos); int make_sub_block = j->first_process != p; if (!end) { error(SYNTAX_ERROR, tok_get_pos(tok), BLOCK_END_ERR_MSG); } else { if (!make_sub_block) { int done=0; tokenizer_t subtok(end, 0); for (; ! done && tok_has_next(&subtok); tok_next(&subtok)) { switch (tok_last_type(&subtok)) { case TOK_END: done = 1; break; case TOK_REDIRECT_OUT: case TOK_REDIRECT_NOCLOB: case TOK_REDIRECT_APPEND: case TOK_REDIRECT_IN: case TOK_REDIRECT_FD: case TOK_PIPE: { done = 1; make_sub_block = 1; break; } case TOK_STRING: { break; } default: { done = 1; error(SYNTAX_ERROR, current_tokenizer_pos, BLOCK_END_ERR_MSG); } } } } if (make_sub_block) { long end_pos = end-tok_string(tok); const wcstring sub_block(tok_string(tok) + current_tokenizer_pos, end_pos - current_tokenizer_pos); p->type = INTERNAL_BLOCK; args.at(0) = completion_t(sub_block); tok_set_pos(tok, (int)end_pos); while (prev_block != current_block) { parser_t::pop_block(); } } else tok_next(tok); } } else tok_next(tok); if (!error_code) { if (p->type == INTERNAL_BUILTIN && parser_keywords_skip_arguments(args.at(0).completion)) { if (!p->get_argv()) p->set_argv(completions_to_wcstring_list(args)); } else { parse_job_argument_list(p, j, tok, args, unskip); } } if (!error_code) { if (!is_new_block) { current_block->had_command = true; } } if (error_code) { /* Make sure the block stack is consistent */ while (prev_block != current_block) { parser_t::pop_block(); } } return !error_code; } /** Do skipped execution of command. This means that only limited execution of block level commands such as end and switch should be preformed. \param j the job to execute */ void parser_t::skipped_exec(job_t * j) { process_t *p; /* Handle other skipped guys */ for (p = j->first_process; p; p=p->next) { if (p->type == INTERNAL_BUILTIN) { if ((wcscmp(p->argv0(), L"for")==0) || (wcscmp(p->argv0(), L"switch")==0) || (wcscmp(p->argv0(), L"begin")==0) || (wcscmp(p->argv0(), L"function")==0)) { this->push_block(new fake_block_t()); } else if (wcscmp(p->argv0(), L"end")==0) { if (!current_block->outer->skip) { exec_job(*this, j); return; } parser_t::pop_block(); } else if (wcscmp(p->argv0(), L"else")==0) { if (current_block->type() == IF) { /* Evaluate this ELSE if the IF expression failed, and so has every ELSEIF (if any) expression thus far */ const if_block_t *ib = static_cast(current_block); if (ib->if_expr_evaluated && ! ib->any_branch_taken) { exec_job(*this, j); return; } } } else if (wcscmp(p->argv0(), L"case")==0) { if (current_block->type() == SWITCH) { exec_job(*this, j); return; } } } } job_free(j); } /* Return whether we should skip the current block, if it is an elseif. */ static bool job_should_skip_elseif(const job_t *job, const block_t *current_block) { if (current_block->type() != IF) { /* Not an IF block, so just honor the skip property */ return current_block->skip; } else { /* We are an IF block */ const if_block_t *ib = static_cast(current_block); /* Execute this ELSEIF if the IF expression has been evaluated, it evaluated to false, and all ELSEIFs so far have evaluated to false. */ bool execute_elseif = (ib->if_expr_evaluated && ! ib->any_branch_taken); /* Invert the sense */ return ! execute_elseif; } } /** Evaluates a job from the specified tokenizer. First calls parse_job to parse the job and then calls exec to execute it. \param tok The tokenizer to read tokens from */ void parser_t::eval_job(tokenizer_t *tok) { ASSERT_IS_MAIN_THREAD(); int start_pos = job_start_pos = tok_get_pos(tok); long long t1=0, t2=0, t3=0; profile_item_t *profile_item = NULL; bool skip = false; int job_begin_pos; const bool do_profile = profile; if (do_profile) { profile_item = new profile_item_t(); profile_item->skipped = 1; profile_items.push_back(profile_item); t1 = get_time(); } switch (tok_last_type(tok)) { case TOK_STRING: { job_t *j = this->job_create(); job_set_flag(j, JOB_FOREGROUND, 1); job_set_flag(j, JOB_TERMINAL, job_get_flag(j, JOB_CONTROL)); job_set_flag(j, JOB_TERMINAL, job_get_flag(j, JOB_CONTROL) \ && (!is_subshell && !is_event)); job_set_flag(j, JOB_SKIP_NOTIFICATION, is_subshell \ || is_block \ || is_event \ || (!get_is_interactive())); current_block->job = j; if (get_is_interactive()) { if (tcgetattr(0, &j->tmodes)) { tok_next(tok); wperror(L"tcgetattr"); job_free(j); break; } } j->first_process = new process_t(); job_begin_pos = tok_get_pos(tok); if (parse_job(j->first_process, j, tok) && j->first_process->get_argv()) { if (job_start_pos < tok_get_pos(tok)) { long stop_pos = tok_get_pos(tok); const wchar_t *newline = wcschr(tok_string(tok)+start_pos, L'\n'); if (newline) stop_pos = mini(stop_pos, newline - tok_string(tok)); j->set_command(wcstring(tok_string(tok)+start_pos, stop_pos-start_pos)); } else j->set_command(L""); if (do_profile) { t2 = get_time(); profile_item->cmd = j->command(); profile_item->skipped=current_block->skip; } /* If we're an ELSEIF, then we may want to unskip, if we're skipping because of an IF */ if (job_get_flag(j, JOB_ELSEIF)) { bool skip_elseif = job_should_skip_elseif(j, current_block); /* Record that we're entering an elseif */ if (! skip_elseif) { /* We must be an IF block here */ assert(current_block->type() == IF); static_cast(current_block)->is_elseif_entry = true; } /* Record that in the block too. This is similar to what builtin_else does. */ current_block->skip = skip_elseif; } skip = skip || current_block->skip; skip = skip || job_get_flag(j, JOB_WILDCARD_ERROR); skip = skip || job_get_flag(j, JOB_SKIP); if (!skip) { int was_builtin = 0; if (j->first_process->type==INTERNAL_BUILTIN && !j->first_process->next) was_builtin = 1; scoped_push tokenizer_pos_push(¤t_tokenizer_pos, job_begin_pos); exec_job(*this, j); /* Only external commands require a new fishd barrier */ if (!was_builtin) set_proc_had_barrier(false); } else { this->skipped_exec(j); } if (do_profile) { t3 = get_time(); profile_item->level=eval_level; profile_item->parse = (int)(t2-t1); profile_item->exec=(int)(t3-t2); } if (current_block->type() == WHILE) { while_block_t *wb = static_cast(current_block); switch (wb->status) { case WHILE_TEST_FIRST: { // PCA I added the 'wb->skip ||' part because we couldn't reliably // control-C out of loops like this: while test 1 -eq 1; end wb->skip = wb->skip || proc_get_last_status()!= 0; wb->status = WHILE_TESTED; } break; } } if (current_block->type() == IF) { if_block_t *ib = static_cast(current_block); if (ib->skip) { /* Nothing */ } else if (! ib->if_expr_evaluated) { /* Execute the IF */ bool if_result = (proc_get_last_status() == 0); ib->any_branch_taken = if_result; /* Don't execute if the expression failed */ current_block->skip = ! if_result; ib->if_expr_evaluated = true; } else if (ib->is_elseif_entry && ! ib->any_branch_taken) { /* Maybe mark an ELSEIF branch as taken */ bool elseif_taken = (proc_get_last_status() == 0); ib->any_branch_taken = elseif_taken; current_block->skip = ! elseif_taken; ib->is_elseif_entry = false; } } } else { /* This job could not be properly parsed. We free it instead, and set the status to 1. This should be rare, since most errors should be detected by the ahead of time validator. */ job_free(j); proc_set_last_status(1); } current_block->job = 0; break; } case TOK_END: { if (tok_has_next(tok)) tok_next(tok); break; } case TOK_BACKGROUND: { const wchar_t *str = tok_string(tok); if (tok_get_pos(tok)>0 && str[tok_get_pos(tok)-1] == L'&') { error(SYNTAX_ERROR, tok_get_pos(tok), CMD_AND_ERR_MSG, tok_get_desc(tok_last_type(tok))); } else { error(SYNTAX_ERROR, tok_get_pos(tok), CMD_ERR_MSG, tok_get_desc(tok_last_type(tok))); } return; } case TOK_ERROR: { error(SYNTAX_ERROR, tok_get_pos(tok), TOK_ERR_MSG, tok_last(tok)); return; } default: { error(SYNTAX_ERROR, tok_get_pos(tok), CMD_ERR_MSG, tok_get_desc(tok_last_type(tok))); return; } } job_reap(0); } int parser_t::eval(const wcstring &cmdStr, const io_chain_t &io, enum block_type_t block_type) { const wchar_t * const cmd = cmdStr.c_str(); size_t forbid_count; int code; block_t *start_current_block = current_block; /* Record the current chain so we can put it back later */ scoped_push block_io_push(&block_io, io); scoped_push forbidden_function_push(&forbidden_function); if (block_type == SUBST) { forbidden_function.clear(); } CHECK_BLOCK(1); forbid_count = forbidden_function.size(); job_reap(0); debug(4, L"eval: %ls", cmd); if (!cmd) { debug(1, EVAL_NULL_ERR_MSG); bugreport(); return 1; } if ((block_type != TOP) && (block_type != SUBST)) { debug(1, INVALID_SCOPE_ERR_MSG, parser_t::get_block_desc(block_type)); bugreport(); return 1; } eval_level++; this->push_block(new scope_block_t(block_type)); tokenizer_t local_tokenizer(cmd, 0); scoped_push tokenizer_push(¤t_tokenizer, &local_tokenizer); error_code = 0; event_fire(NULL); while (tok_has_next(current_tokenizer) && !error_code && !sanity_check() && !exit_status()) { this->eval_job(current_tokenizer); event_fire(NULL); } parser_t::pop_block(); while (start_current_block != current_block) { if (current_block == 0) { debug(0, _(L"End of block mismatch. Program terminating.")); bugreport(); FATAL_EXIT(); break; } if ((!error_code) && (!exit_status()) && (!proc_get_last_status())) { //debug( 2, L"Status %d\n", proc_get_last_status() ); debug(1, L"%ls", parser_t::get_block_desc(current_block->type())); debug(1, BLOCK_END_ERR_MSG); fwprintf(stderr, L"%ls", parser_t::current_line()); const wcstring h = builtin_help_get(*this, L"end"); if (h.size()) fwprintf(stderr, L"%ls", h.c_str()); break; } parser_t::pop_block(); } this->print_errors_stderr(); tokenizer_push.restore(); while (forbidden_function.size() > forbid_count) parser_t::allow_function(); /* Restore previous eval state */ eval_level--; code=error_code; error_code=0; job_reap(0); return code; } /** \return the block type created by the specified builtin, or -1 on error. */ block_type_t parser_get_block_type(const wcstring &cmd) { for (size_t i=0; block_lookup[i].desc; i++) { if (block_lookup[i].name && cmd == block_lookup[i].name) { return block_lookup[i].type; } } return (block_type_t)-1; } /** \return the block command that createa the specified block type, or null on error. */ const wchar_t *parser_get_block_command(int type) { for (size_t i=0; block_lookup[i].desc; i++) { if (block_lookup[i].type == type) { return block_lookup[i].name; } } return NULL; } /** Test if this argument contains any errors. Detected errors include syntax errors in command substitutions, improperly escaped characters and improper use of the variable expansion operator. */ int parser_t::parser_test_argument(const wchar_t *arg, wcstring *out, const wchar_t *prefix, int offset) { int err=0; wchar_t *paran_begin, *paran_end; wchar_t *arg_cpy; int do_loop = 1; CHECK(arg, 1); arg_cpy = wcsdup(arg); while (do_loop) { switch (parse_util_locate_cmdsubst(arg_cpy, ¶n_begin, ¶n_end, false)) { case -1: err=1; if (out) { error(SYNTAX_ERROR, offset, L"Mismatched parenthesis"); this->print_errors(*out, prefix); } free(arg_cpy); return err; case 0: do_loop = 0; break; case 1: { wchar_t *subst = wcsndup(paran_begin+1, paran_end-paran_begin-1); wcstring tmp; tmp.append(arg_cpy, paran_begin - arg_cpy); tmp.push_back(INTERNAL_SEPARATOR); tmp.append(paran_end+1); // debug( 1, L"%ls -> %ls %ls", arg_cpy, subst, tmp.buff ); err |= parser_t::detect_errors(subst, out, prefix); free(subst); free(arg_cpy); arg_cpy = wcsdup(tmp.c_str()); /* Do _not_ call sb_destroy on this stringbuffer - it's buffer is used as the new 'arg_cpy'. It is free'd at the end of the loop. */ break; } } } wcstring unesc; if (! unescape_string(arg_cpy, &unesc, UNESCAPE_SPECIAL)) { if (out) { error(SYNTAX_ERROR, offset, L"Invalid token '%ls'", arg_cpy); print_errors(*out, prefix); } return 1; } else { /* Check for invalid variable expansions */ const size_t unesc_size = unesc.size(); for (size_t idx = 0; idx < unesc_size; idx++) { switch (unesc.at(idx)) { case VARIABLE_EXPAND: case VARIABLE_EXPAND_SINGLE: { wchar_t next_char = (idx + 1 < unesc_size ? unesc.at(idx + 1) : L'\0'); if (next_char != VARIABLE_EXPAND && next_char != VARIABLE_EXPAND_SINGLE && ! wcsvarchr(next_char)) { err=1; if (out) { expand_variable_error(*this, unesc, idx, offset); print_errors(*out, prefix); } } break; } } } } free(arg_cpy); return err; } int parser_t::test_args(const wchar_t * buff, wcstring *out, const wchar_t *prefix) { int do_loop = 1; int err = 0; CHECK(buff, 1); tokenizer_t tok(buff, 0); scoped_push tokenizer_push(¤t_tokenizer, &tok); scoped_push tokenizer_pos_push(¤t_tokenizer_pos); for (; do_loop && tok_has_next(&tok); tok_next(&tok)) { current_tokenizer_pos = tok_get_pos(&tok); switch (tok_last_type(&tok)) { case TOK_STRING: { err |= parser_test_argument(tok_last(&tok), out, prefix, tok_get_pos(&tok)); break; } case TOK_END: { break; } case TOK_ERROR: { if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), TOK_ERR_MSG, tok_last(&tok)); print_errors(*out, prefix); } err=1; do_loop=0; break; } default: { if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), UNEXPECTED_TOKEN_ERR_MSG, tok_get_desc(tok_last_type(&tok))); print_errors(*out, prefix); } err=1; do_loop=0; break; } } } error_code=0; return err; } // helper type used in parser::test below struct block_info_t { int position; //tokenizer position block_type_t type; //type of the block }; parser_test_error_bits_t parser_t::detect_errors(const wchar_t *buff, wcstring *out, const wchar_t *prefix) { ASSERT_IS_MAIN_THREAD(); /* Set to one if a command name has been given for the currently parsed process specification */ int had_cmd=0; int err=0; int unfinished = 0; // This is very nearly a stack, but sometimes we have to inspect non-top elements (e.g. return) std::vector block_infos; /* Set to 1 if the current command is inside a pipeline */ int is_pipeline = 0; /* Set to one if the currently specified process can not be used inside a pipeline */ int forbid_pipeline = 0; /* Set to one if an additional process specification is needed */ bool needs_cmd = false; /* Counter on the number of arguments this function has encountered so far. Is set to -1 when the count is unknown, i.e. after encountering an argument that contains substitutions that can expand to more/less arguemtns then 1. */ int arg_count=0; /* The currently validated command. */ wcstring command; bool has_command = false; CHECK(buff, 1); tokenizer_t tok(buff, 0); scoped_push tokenizer_push(¤t_tokenizer, &tok); scoped_push tokenizer_pos_push(¤t_tokenizer_pos); for (;; tok_next(&tok)) { current_tokenizer_pos = tok_get_pos(&tok); int last_type = tok_last_type(&tok); int end_of_cmd = 0; switch (last_type) { case TOK_STRING: { if (!had_cmd) { int mark = tok_get_pos(&tok); had_cmd = 1; arg_count=0; command = tok_last(&tok); // Pass SKIP_HOME_DIRECTORIES for https://github.com/fish-shell/fish-shell/issues/512 has_command = expand_one(command, EXPAND_SKIP_CMDSUBST | EXPAND_SKIP_VARIABLES | EXPAND_SKIP_HOME_DIRECTORIES); if (! has_command) { command = L""; err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), ILLEGAL_CMD_ERR_MSG, tok_last(&tok)); print_errors(*out, prefix); } break; } if (needs_cmd) { /* end is not a valid command when a followup command is needed, such as after 'and' or 'while' */ if (contains(command, L"end")) { err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), COND_ERR_MSG); print_errors(*out, prefix); } } needs_cmd = false; } /* Decrement block count on end command */ if (command == L"end") { tok_next(&tok); tok_set_pos(&tok, mark); /* Test that end is not used when not inside any block */ if (block_infos.empty()) { err = 1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), INVALID_END_ERR_MSG); print_errors(*out, prefix); const wcstring h = builtin_help_get(*this, L"end"); if (! h.empty()) append_format(*out, L"%ls", h.c_str()); } } else { block_infos.pop_back(); } } /* Handle block commands */ if (parser_keywords_is_block(command)) { struct block_info_t info = {current_tokenizer_pos, parser_get_block_type(command)}; block_infos.push_back(info); tok_next(&tok); tok_set_pos(&tok, mark); } /* If parser_keywords_is_subcommand is true, the command accepts a second command as it's first argument. If parser_skip_arguments is true, the second argument is optional. */ if (parser_keywords_is_subcommand(command) && !parser_keywords_skip_arguments(command)) { needs_cmd = true; had_cmd = 0; } if (contains(command, L"or", L"and")) { /* 'or' and 'and' can not be used inside pipelines */ if (is_pipeline) { err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), EXEC_ERR_MSG); print_errors(*out, prefix); } } } /* There are a lot of situations where pipelines are forbidden, including when using the exec builtin. */ if (parser_is_pipe_forbidden(command)) { if (is_pipeline) { err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), EXEC_ERR_MSG); print_errors(*out, prefix); } } forbid_pipeline = 1; } /* Test that the case builtin is only used directly in a switch block */ if (command == L"case") { if (block_infos.empty() || block_infos.back().type != SWITCH) { err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), INVALID_CASE_ERR_MSG); print_errors(*out, prefix); const wcstring h = builtin_help_get(*this, L"case"); if (h.size()) append_format(*out, L"%ls", h.c_str()); } } } /* Test that the return bultin is only used within function definitions */ if (command == L"return") { bool found_func = false; size_t block_idx = block_infos.size(); while (block_idx--) { if (block_infos.at(block_idx).type == FUNCTION_DEF) { found_func = true; break; } } if (!found_func) { /* Peek to see if the next argument is --help, in which case we'll allow it to show the help. */ int old_pos = tok_get_pos(&tok); int is_help = 0; tok_next(&tok); if (tok_last_type(&tok) == TOK_STRING) { wcstring first_arg = tok_last(&tok); if (expand_one(first_arg, EXPAND_SKIP_CMDSUBST) && parser_t::is_help(first_arg.c_str(), 3)) { is_help = 1; } } tok_set_pos(&tok, old_pos); if (!is_help) { err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), INVALID_RETURN_ERR_MSG); print_errors(*out, prefix); } } } } /* Test that break and continue are only used within loop blocks */ if (contains(command, L"break", L"continue")) { bool found_loop = false; size_t block_idx = block_infos.size(); while (block_idx--) { block_type_t type = block_infos.at(block_idx).type; if (type == WHILE || type == FOR) { found_loop = true; break; } } if (!found_loop) { /* Peek to see if the next argument is --help, in which case we'll allow it to show the help. */ int old_pos = tok_get_pos(&tok); int is_help = 0; tok_next(&tok); if (tok_last_type(&tok) == TOK_STRING) { wcstring first_arg = tok_last(&tok); if (expand_one(first_arg, EXPAND_SKIP_CMDSUBST) && parser_t::is_help(first_arg.c_str(), 3)) { is_help = 1; } } tok_set_pos(&tok, old_pos); if (!is_help) { err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), INVALID_LOOP_ERR_MSG); print_errors(*out, prefix); } } } } /* Test that else and else-if are only used directly in an if-block */ if (command == L"else") { if (block_infos.empty() || block_infos.back().type != IF) { err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), INVALID_ELSE_ERR_MSG, command.c_str()); print_errors(*out, prefix); } } } } else { err |= parser_test_argument(tok_last(&tok), out, prefix, tok_get_pos(&tok)); /* If possible, keep track of number of supplied arguments */ if (arg_count >= 0 && expand_is_clean(tok_last(&tok))) { arg_count++; } else { arg_count = -1; } if (has_command) { /* Try to make sure the second argument to 'for' is 'in' */ if (command == L"for") { if (arg_count == 1) { if (wcsvarname(tok_last(&tok))) { err = 1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), BUILTIN_FOR_ERR_NAME, L"for", tok_last(&tok)); print_errors(*out, prefix); } } } else if (arg_count == 2) { if (wcscmp(tok_last(&tok), L"in") != 0) { err = 1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), BUILTIN_FOR_ERR_IN, L"for"); print_errors(*out, prefix); } } } } else if (command == L"else") { if (arg_count == 1) { /* Any second argument must be "if" */ if (wcscmp(tok_last(&tok), L"if") != 0) { err = 1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), BUILTIN_ELSEIF_ERR_ARGUMENT, L"else"); print_errors(*out, prefix); } } else { /* Successfully detected "else if". Now we need a new command. */ needs_cmd = true; had_cmd = false; } } } } } break; } case TOK_REDIRECT_OUT: case TOK_REDIRECT_IN: case TOK_REDIRECT_APPEND: case TOK_REDIRECT_FD: case TOK_REDIRECT_NOCLOB: { if (!had_cmd) { err = 1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), INVALID_REDIRECTION_ERR_MSG); print_errors(*out, prefix); } } break; } case TOK_END: { if (needs_cmd && !had_cmd) { err = 1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), CMD_ERR_MSG, tok_get_desc(tok_last_type(&tok))); print_errors(*out, prefix); } } needs_cmd = false; had_cmd = 0; is_pipeline=0; forbid_pipeline=0; end_of_cmd = 1; break; } case TOK_PIPE: { if (!had_cmd) { err=1; if (out) { if (tok_get_pos(&tok)>0 && buff[tok_get_pos(&tok)-1] == L'|') { error(SYNTAX_ERROR, tok_get_pos(&tok), CMD_OR_ERR_MSG, tok_get_desc(tok_last_type(&tok))); } else { error(SYNTAX_ERROR, tok_get_pos(&tok), CMD_ERR_MSG, tok_get_desc(tok_last_type(&tok))); } print_errors(*out, prefix); } } else if (forbid_pipeline) { err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), EXEC_ERR_MSG); print_errors(*out, prefix); } } else { needs_cmd = true; is_pipeline=1; had_cmd=0; end_of_cmd = 1; } break; } case TOK_BACKGROUND: { if (!had_cmd) { err = 1; if (out) { if (tok_get_pos(&tok)>0 && buff[tok_get_pos(&tok)-1] == L'&') { error(SYNTAX_ERROR, tok_get_pos(&tok), CMD_AND_ERR_MSG, tok_get_desc(tok_last_type(&tok))); } else { error(SYNTAX_ERROR, tok_get_pos(&tok), CMD_ERR_MSG, tok_get_desc(tok_last_type(&tok))); } print_errors(*out, prefix); } } had_cmd = 0; end_of_cmd = 1; break; } case TOK_ERROR: default: if (tok_get_error(&tok) == TOK_UNTERMINATED_QUOTE) { unfinished = 1; } else { // Only print errors once if (out && ! err) { error(SYNTAX_ERROR, tok_get_pos(&tok), TOK_ERR_MSG, tok_last(&tok)); print_errors(*out, prefix); } err = 1; } break; } if (end_of_cmd) { if (has_command && command == L"for") { if (arg_count >= 0 && arg_count < 2) { /* Not enough arguments to the for builtin */ err = 1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), BUILTIN_FOR_ERR_COUNT, L"for", arg_count); print_errors(*out, prefix); } } } else if (has_command && command == L"else") { if (arg_count == 1) { /* If we have any arguments, we must have at least two...either "else" or "else if foo..." */ err = true; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), BUILTIN_ELSEIF_ERR_COUNT, L"else", arg_count); print_errors(*out, prefix); } } } } if (!tok_has_next(&tok)) break; } if (needs_cmd) { err=1; if (out) { error(SYNTAX_ERROR, tok_get_pos(&tok), COND_ERR_MSG); print_errors(*out, prefix); } } if (out != NULL && ! block_infos.empty()) { const wchar_t *cmd; int bad_pos = block_infos.back().position; block_type_t bad_type = block_infos.back().type; error(SYNTAX_ERROR, bad_pos, BLOCK_END_ERR_MSG); print_errors(*out, prefix); cmd = parser_get_block_command(bad_type); if (cmd) { const wcstring h = builtin_help_get(*this, cmd); if (h.size()) { append_format(*out, L"%ls", h.c_str()); } } } /* Calculate exit status */ if (! block_infos.empty()) unfinished = 1; parser_test_error_bits_t res = 0; if (err) res |= PARSER_TEST_ERROR; if (unfinished) res |= PARSER_TEST_INCOMPLETE; /* Cleanup */ error_code=0; return res; } block_t::block_t(block_type_t t) : block_type(t), made_fake(false), skip(), had_command(), tok_pos(), loop_status(), job(), src_filename(), src_lineno(), wants_pop_env(false), event_blocks(), outer(NULL) { } block_t::~block_t() { } /* Various block constructors */ if_block_t::if_block_t() : block_t(IF), if_expr_evaluated(false), is_elseif_entry(false), any_branch_taken(false), else_evaluated(false) { } event_block_t::event_block_t(const event_t &evt) : block_t(EVENT), event(evt) { } function_block_t::function_block_t(const process_t *p, const wcstring &n, bool shadows) : block_t(shadows ? FUNCTION_CALL : FUNCTION_CALL_NO_SHADOW), process(p), name(n) { } source_block_t::source_block_t(const wchar_t *src) : block_t(SOURCE), source_file(src) { } for_block_t::for_block_t(const wcstring &var) : block_t(FOR), variable(var), sequence() { } while_block_t::while_block_t() : block_t(WHILE), status(0) { } switch_block_t::switch_block_t(const wcstring &sv) : block_t(SWITCH), switch_taken(false), switch_value(sv) { } fake_block_t::fake_block_t() : block_t(FAKE) { } function_def_block_t::function_def_block_t() : block_t(FUNCTION_DEF), function_data() { } scope_block_t::scope_block_t(block_type_t type) : block_t(type) { assert(type == BEGIN || type == TOP || type == SUBST); } breakpoint_block_t::breakpoint_block_t() : block_t(BREAKPOINT) { }