Refactor how redirections are represented by the tokenizer

Prior to this fix, each redirection type was a separate token_type.
Unify these under a single type TOK_REDIRECT and break the redirection
type out into a new sub-type redirection_type_t.
This commit is contained in:
ridiculousfish 2018-02-23 15:19:58 -08:00
parent 6673fe5457
commit 99fb7bb6aa
8 changed files with 131 additions and 147 deletions

View file

@ -536,10 +536,9 @@ static void test_tokenizer() {
L"string <redirection 2>&1 'nested \"quoted\" '(string containing subshells " L"string <redirection 2>&1 'nested \"quoted\" '(string containing subshells "
L"){and,brackets}$as[$well (as variable arrays)] not_a_redirect^ ^ ^^is_a_redirect " L"){and,brackets}$as[$well (as variable arrays)] not_a_redirect^ ^ ^^is_a_redirect "
L"Compress_Newlines\n \n\t\n \nInto_Just_One"; L"Compress_Newlines\n \n\t\n \nInto_Just_One";
const int types[] = {TOK_STRING, TOK_REDIRECT_IN, TOK_STRING, TOK_REDIRECT_FD, const int types[] = {TOK_STRING, TOK_REDIRECT, TOK_STRING, TOK_REDIRECT, TOK_STRING,
TOK_STRING, TOK_STRING, TOK_STRING, TOK_REDIRECT_OUT, TOK_STRING, TOK_STRING, TOK_REDIRECT, TOK_REDIRECT, TOK_STRING,
TOK_REDIRECT_APPEND, TOK_STRING, TOK_STRING, TOK_END, TOK_STRING, TOK_END, TOK_STRING};
TOK_STRING};
say(L"Test correct tokenization"); say(L"Test correct tokenization");
@ -594,25 +593,25 @@ static void test_tokenizer() {
} }
// Test redirection_type_for_string. // Test redirection_type_for_string.
if (redirection_type_for_string(L"<") != TOK_REDIRECT_IN) if (redirection_type_for_string(L"<") != redirection_type_t::input)
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
if (redirection_type_for_string(L"^") != TOK_REDIRECT_OUT) if (redirection_type_for_string(L"^") != redirection_type_t::overwrite)
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
if (redirection_type_for_string(L">") != TOK_REDIRECT_OUT) if (redirection_type_for_string(L">") != redirection_type_t::overwrite)
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
if (redirection_type_for_string(L"2>") != TOK_REDIRECT_OUT) if (redirection_type_for_string(L"2>") != redirection_type_t::overwrite)
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
if (redirection_type_for_string(L">>") != TOK_REDIRECT_APPEND) if (redirection_type_for_string(L">>") != redirection_type_t::append)
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
if (redirection_type_for_string(L"2>>") != TOK_REDIRECT_APPEND) if (redirection_type_for_string(L"2>>") != redirection_type_t::append)
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
if (redirection_type_for_string(L"2>?") != TOK_REDIRECT_NOCLOB) if (redirection_type_for_string(L"2>?") != redirection_type_t::noclob)
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
if (redirection_type_for_string(L"9999999999999999>?") != TOK_NONE) if (redirection_type_for_string(L"9999999999999999>?"))
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
if (redirection_type_for_string(L"2>&3") != TOK_REDIRECT_FD) if (redirection_type_for_string(L"2>&3") != redirection_type_t::fd)
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
if (redirection_type_for_string(L"2>|") != TOK_NONE) if (redirection_type_for_string(L"2>|"))
err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); err(L"redirection_type_for_string failed on line %ld", (long)__LINE__);
} }

View file

@ -836,11 +836,11 @@ void highlighter_t::color_redirection(tnode_t<g::redirection> redirection_node)
if (redir_prim) { if (redir_prim) {
wcstring target; wcstring target;
const enum token_type redirect_type = const maybe_t<redirection_type_t> redirect_type =
redirection_type(redirection_node, this->buff, nullptr, &target); redirection_type(redirection_node, this->buff, nullptr, &target);
// We may get a TOK_NONE redirection type, e.g. if the redirection is invalid. // We may get a missing redirection type if the redirection is invalid.
auto hl = redirect_type == TOK_NONE ? highlight_spec_error : highlight_spec_redirection; auto hl = redirect_type ? highlight_spec_redirection : highlight_spec_error;
this->color_node(redir_prim, hl); this->color_node(redir_prim, hl);
// Check if the argument contains a command substitution. If so, highlight it as a param // Check if the argument contains a command substitution. If so, highlight it as a param
@ -852,7 +852,10 @@ void highlighter_t::color_redirection(tnode_t<g::redirection> redirection_node)
// disallow redirections into a non-existent directory. // disallow redirections into a non-existent directory.
bool target_is_valid = true; bool target_is_valid = true;
if (!this->io_ok) { if (!redirect_type) {
// not a valid redirection
target_is_valid = false;
} else if (!this->io_ok) {
// I/O is disallowed, so we don't have much hope of catching anything but gross // I/O is disallowed, so we don't have much hope of catching anything but gross
// errors. Assume it's valid. // errors. Assume it's valid.
target_is_valid = true; target_is_valid = true;
@ -865,22 +868,22 @@ void highlighter_t::color_redirection(tnode_t<g::redirection> redirection_node)
// redirections). Note that the target is now unescaped. // redirections). Note that the target is now unescaped.
const wcstring target_path = const wcstring target_path =
path_apply_working_directory(target, this->working_directory); path_apply_working_directory(target, this->working_directory);
switch (redirect_type) { switch (*redirect_type) {
case TOK_REDIRECT_FD: { case redirection_type_t::fd: {
int fd = fish_wcstoi(target.c_str()); int fd = fish_wcstoi(target.c_str());
target_is_valid = !errno && fd >= 0; target_is_valid = !errno && fd >= 0;
break; break;
} }
case TOK_REDIRECT_IN: { case redirection_type_t::input: {
// Input redirections must have a readable non-directory. // Input redirections must have a readable non-directory.
struct stat buf = {}; struct stat buf = {};
target_is_valid = !waccess(target_path, R_OK) && target_is_valid = !waccess(target_path, R_OK) &&
!wstat(target_path, &buf) && !S_ISDIR(buf.st_mode); !wstat(target_path, &buf) && !S_ISDIR(buf.st_mode);
break; break;
} }
case TOK_REDIRECT_OUT: case redirection_type_t::overwrite:
case TOK_REDIRECT_APPEND: case redirection_type_t::append:
case TOK_REDIRECT_NOCLOB: { case redirection_type_t::noclob: {
// Test whether the file exists, and whether it's writable (possibly after // Test whether the file exists, and whether it's writable (possibly after
// creating it). access() returns failure if the file does not exist. // creating it). access() returns failure if the file does not exist.
bool file_exists = false, file_is_writable = false; bool file_exists = false, file_is_writable = false;
@ -922,14 +925,9 @@ void highlighter_t::color_redirection(tnode_t<g::redirection> redirection_node)
} }
// NOCLOB means that we must not overwrite files that exist. // NOCLOB means that we must not overwrite files that exist.
target_is_valid = file_is_writable && target_is_valid =
!(file_exists && redirect_type == TOK_REDIRECT_NOCLOB); file_is_writable &&
break; !(file_exists && redirect_type == redirection_type_t::noclob);
}
default: {
// We should not get here, since the node was marked as a redirection, but
// treat it as an error for paranoia.
target_is_valid = false;
break; break;
} }
} }

View file

@ -871,8 +871,7 @@ bool parse_execution_context_t::determine_io_chain(tnode_t<g::arguments_or_redir
while (auto redirect_node = node.next_in_list<g::redirection>()) { while (auto redirect_node = node.next_in_list<g::redirection>()) {
int source_fd = -1; // source fd int source_fd = -1; // source fd
wcstring target; // file path or target fd wcstring target; // file path or target fd
enum token_type redirect_type = auto redirect_type = redirection_type(redirect_node, pstree->src, &source_fd, &target);
redirection_type(redirect_node, pstree->src, &source_fd, &target);
// PCA: I can't justify this EXPAND_SKIP_VARIABLES flag. It was like this when I got here. // PCA: I can't justify this EXPAND_SKIP_VARIABLES flag. It was like this when I got here.
bool target_expanded = expand_one(target, no_exec ? EXPAND_SKIP_VARIABLES : 0, NULL); bool target_expanded = expand_one(target, no_exec ? EXPAND_SKIP_VARIABLES : 0, NULL);
@ -884,9 +883,9 @@ bool parse_execution_context_t::determine_io_chain(tnode_t<g::arguments_or_redir
// Generate the actual IO redirection. // Generate the actual IO redirection.
shared_ptr<io_data_t> new_io; shared_ptr<io_data_t> new_io;
assert(redirect_type != TOK_NONE); assert(redirect_type && "expected to have a valid redirection");
switch (redirect_type) { switch (*redirect_type) {
case TOK_REDIRECT_FD: { case redirection_type_t::fd: {
if (target == L"-") { if (target == L"-") {
new_io.reset(new io_close_t(source_fd)); new_io.reset(new io_close_t(source_fd));
} else { } else {
@ -902,21 +901,12 @@ bool parse_execution_context_t::determine_io_chain(tnode_t<g::arguments_or_redir
} }
break; break;
} }
case TOK_REDIRECT_OUT: default: {
case TOK_REDIRECT_APPEND: int oflags = oflags_for_redirection_type(*redirect_type);
case TOK_REDIRECT_IN:
case TOK_REDIRECT_NOCLOB: {
int oflags = oflags_for_redirection_type(redirect_type);
io_file_t *new_io_file = new io_file_t(source_fd, target, oflags); io_file_t *new_io_file = new io_file_t(source_fd, target, oflags);
new_io.reset(new_io_file); new_io.reset(new_io_file);
break; break;
} }
default: {
// Should be unreachable.
debug(0, "Unexpected redirection type %ld.", (long)redirect_type);
PARSER_DIE();
break;
}
} }
// Append the new_io if we got one. // Append the new_io if we got one.

View file

@ -231,11 +231,7 @@ static inline parse_token_type_t parse_token_type_from_tokenizer_token(
result = parse_token_type_background; result = parse_token_type_background;
break; break;
} }
case TOK_REDIRECT_OUT: case TOK_REDIRECT: {
case TOK_REDIRECT_APPEND:
case TOK_REDIRECT_IN:
case TOK_REDIRECT_FD:
case TOK_REDIRECT_NOCLOB: {
result = parse_token_type_redirection; result = parse_token_type_redirection;
break; break;
} }

View file

@ -50,10 +50,11 @@ enum parse_bool_statement_type_t bool_statement_type(tnode_t<grammar::boolean_st
return static_cast<parse_bool_statement_type_t>(stmt.tag()); return static_cast<parse_bool_statement_type_t>(stmt.tag());
} }
enum token_type redirection_type(tnode_t<grammar::redirection> redirection, const wcstring &src, maybe_t<redirection_type_t> redirection_type(tnode_t<grammar::redirection> redirection,
int *out_fd, wcstring *out_target) { const wcstring &src, int *out_fd,
wcstring *out_target) {
assert(redirection && "redirection is missing"); assert(redirection && "redirection is missing");
enum token_type result = TOK_NONE; maybe_t<redirection_type_t> result{};
tnode_t<grammar::tok_redirection> prim = redirection.child<0>(); // like 2> tnode_t<grammar::tok_redirection> prim = redirection.child<0>(); // like 2>
assert(prim && "expected to have primitive"); assert(prim && "expected to have primitive");

View file

@ -234,9 +234,10 @@ parse_statement_decoration_t get_decoration(tnode_t<grammar::plain_statement> st
/// Return the type for a boolean statement. /// Return the type for a boolean statement.
enum parse_bool_statement_type_t bool_statement_type(tnode_t<grammar::boolean_statement> stmt); enum parse_bool_statement_type_t bool_statement_type(tnode_t<grammar::boolean_statement> stmt);
/// Given a redirection, get the redirection type (or TOK_NONE) and target (file path, or fd). /// Given a redirection, get the redirection type (or none) and target (file path, or fd).
enum token_type redirection_type(tnode_t<grammar::redirection> redirection, const wcstring &src, maybe_t<redirection_type_t> redirection_type(tnode_t<grammar::redirection> redirection,
int *out_fd, wcstring *out_target); const wcstring &src, int *out_fd,
wcstring *out_target);
/// Return the arguments under an arguments_list or arguments_or_redirection_list /// Return the arguments under an arguments_list or arguments_or_redirection_list
/// Do not return more than max. /// Do not return more than max.

View file

@ -323,15 +323,25 @@ tok_t tokenizer_t::read_string() {
return result; return result;
} }
/// Reads a redirection or an "fd pipe" (like 2>|) from a string. Returns how many characters were // Reads a redirection or an "fd pipe" (like 2>|) from a string.
/// consumed. If zero, then this string was not a redirection. Also returns by reference the // Returns the parsed pipe or redirection, or none() on error.
/// redirection mode, and the fd to redirection. If there is overflow, *out_fd is set to -1. struct parsed_redir_or_pipe_t {
static size_t read_redirection_or_fd_pipe(const wchar_t *buff, // Number of characters consumed.
enum token_type *out_redirection_mode, int *out_fd) { size_t consumed{0};
bool errored = false;
int fd = 0;
enum token_type redirection_mode = TOK_NONE;
// The token type, always either TOK_PIPE or TOK_REDIRECT.
token_type type{TOK_REDIRECT};
// The redirection mode if the type is TOK_REDIRECT.
redirection_type_t redirection_mode{redirection_type_t::overwrite};
// The redirected fd, or -1 on overflow.
int fd{0};
};
static maybe_t<parsed_redir_or_pipe_t> read_redirection_or_fd_pipe(const wchar_t *buff) {
bool errored = false;
parsed_redir_or_pipe_t result;
size_t idx = 0; size_t idx = 0;
// Determine the fd. This may be specified as a prefix like '2>...' or it may be implicit like // Determine the fd. This may be specified as a prefix like '2>...' or it may be implicit like
@ -343,21 +353,21 @@ static size_t read_redirection_or_fd_pipe(const wchar_t *buff,
if (big_fd <= INT_MAX) big_fd = big_fd * 10 + (buff[idx] - L'0'); if (big_fd <= INT_MAX) big_fd = big_fd * 10 + (buff[idx] - L'0');
} }
fd = (big_fd > INT_MAX ? -1 : static_cast<int>(big_fd)); result.fd = (big_fd > INT_MAX ? -1 : static_cast<int>(big_fd));
if (idx == 0) { if (idx == 0) {
// We did not find a leading digit, so there's no explicit fd. Infer it from the type. // We did not find a leading digit, so there's no explicit fd. Infer it from the type.
switch (buff[idx]) { switch (buff[idx]) {
case L'>': { case L'>': {
fd = STDOUT_FILENO; result.fd = STDOUT_FILENO;
break; break;
} }
case L'<': { case L'<': {
fd = STDIN_FILENO; result.fd = STDIN_FILENO;
break; break;
} }
case L'^': { case L'^': {
fd = STDERR_FILENO; result.fd = STDERR_FILENO;
break; break;
} }
default: { default: {
@ -371,55 +381,49 @@ static size_t read_redirection_or_fd_pipe(const wchar_t *buff,
// Don't allow an fd with a caret redirection - see #1873 // Don't allow an fd with a caret redirection - see #1873
wchar_t redirect_char = buff[idx++]; // note increment of idx wchar_t redirect_char = buff[idx++]; // note increment of idx
if (redirect_char == L'>' || (redirect_char == L'^' && idx == 1)) { if (redirect_char == L'>' || (redirect_char == L'^' && idx == 1)) {
redirection_mode = TOK_REDIRECT_OUT; result.redirection_mode = redirection_type_t::overwrite;
if (buff[idx] == redirect_char) { if (buff[idx] == redirect_char) {
// Doubled up like ^^ or >>. That means append. // Doubled up like ^^ or >>. That means append.
redirection_mode = TOK_REDIRECT_APPEND; result.redirection_mode = redirection_type_t::append;
idx++; idx++;
} }
} else if (redirect_char == L'<') { } else if (redirect_char == L'<') {
redirection_mode = TOK_REDIRECT_IN; result.redirection_mode = redirection_type_t::input;
} else { } else {
// Something else. // Something else.
errored = true; errored = true;
} }
// Don't return valid-looking stuff on error. // Bail on error.
if (errored) { if (errored) {
idx = 0; return none();
redirection_mode = TOK_NONE;
} else {
// Optional characters like & or ?, or the pipe char |.
wchar_t opt_char = buff[idx];
if (opt_char == L'&') {
redirection_mode = TOK_REDIRECT_FD;
idx++;
} else if (opt_char == L'?') {
redirection_mode = TOK_REDIRECT_NOCLOB;
idx++;
} else if (opt_char == L'|') {
// So the string looked like '2>|'. This is not a redirection - it's a pipe! That gets
// handled elsewhere.
redirection_mode = TOK_PIPE;
idx++;
}
} }
// Return stuff. // Optional characters like & or ?, or the pipe char |.
if (out_redirection_mode != NULL) *out_redirection_mode = redirection_mode; wchar_t opt_char = buff[idx];
if (out_fd != NULL) *out_fd = fd; if (opt_char == L'&') {
result.redirection_mode = redirection_type_t::fd;
idx++;
} else if (opt_char == L'?') {
result.redirection_mode = redirection_type_t::noclob;
idx++;
} else if (opt_char == L'|') {
// So the string looked like '2>|'. This is not a redirection - it's a pipe! That gets
// handled elsewhere.
result.type = TOK_PIPE;
idx++;
}
return idx; result.consumed = idx;
return result;
} }
enum token_type redirection_type_for_string(const wcstring &str, int *out_fd) { maybe_t<redirection_type_t> redirection_type_for_string(const wcstring &str, int *out_fd) {
enum token_type mode = TOK_NONE; auto v = read_redirection_or_fd_pipe(str.c_str());
int fd = 0;
read_redirection_or_fd_pipe(str.c_str(), &mode, &fd);
// Redirections only, no pipes. // Redirections only, no pipes.
if (mode == TOK_PIPE || fd < 0) mode = TOK_NONE; if (!v || v->type != TOK_REDIRECT || v->fd < 0) return none();
if (out_fd != NULL) *out_fd = fd; if (out_fd) *out_fd = v->fd;
return mode; return v->redirection_mode;
} }
int fd_redirected_by_pipe(const wcstring &str) { int fd_redirected_by_pipe(const wcstring &str) {
@ -427,27 +431,22 @@ int fd_redirected_by_pipe(const wcstring &str) {
if (str == L"|") { if (str == L"|") {
return STDOUT_FILENO; return STDOUT_FILENO;
} }
auto v = read_redirection_or_fd_pipe(str.c_str());
enum token_type mode = TOK_NONE; return (v && v->type == TOK_PIPE) ? v->fd : -1;
int fd = 0;
read_redirection_or_fd_pipe(str.c_str(), &mode, &fd);
// Pipes only.
if (mode != TOK_PIPE || fd < 0) fd = -1;
return fd;
} }
int oflags_for_redirection_type(enum token_type type) { int oflags_for_redirection_type(redirection_type_t type) {
switch (type) { switch (type) {
case TOK_REDIRECT_APPEND: { case redirection_type_t::append: {
return O_CREAT | O_APPEND | O_WRONLY; return O_CREAT | O_APPEND | O_WRONLY;
} }
case TOK_REDIRECT_OUT: { case redirection_type_t::overwrite: {
return O_CREAT | O_WRONLY | O_TRUNC; return O_CREAT | O_WRONLY | O_TRUNC;
} }
case TOK_REDIRECT_NOCLOB: { case redirection_type_t::noclob: {
return O_CREAT | O_EXCL | O_WRONLY; return O_CREAT | O_EXCL | O_WRONLY;
} }
case TOK_REDIRECT_IN: { case redirection_type_t::input: {
return O_RDONLY; return O_RDONLY;
} }
default: { return -1; } default: { return -1; }
@ -551,39 +550,35 @@ maybe_t<tok_t> tokenizer_t::tok_next() {
case L'^': { case L'^': {
// There's some duplication with the code in the default case below. The key difference // There's some duplication with the code in the default case below. The key difference
// here is that we must never parse these as a string; a failed redirection is an error! // here is that we must never parse these as a string; a failed redirection is an error!
enum token_type mode = TOK_NONE; auto redir_or_pipe = read_redirection_or_fd_pipe(this->buff);
int fd = -1; if (!redir_or_pipe || redir_or_pipe->fd < 0) {
size_t consumed = read_redirection_or_fd_pipe(this->buff, &mode, &fd);
if (consumed == 0 || fd < 0) {
return this->call_error(TOK_INVALID_REDIRECT, this->buff, this->buff); return this->call_error(TOK_INVALID_REDIRECT, this->buff, this->buff);
} }
result.type = mode; result.type = redir_or_pipe->type;
result.redirected_fd = fd; result.redirected_fd = redir_or_pipe->fd;
result.length = consumed; result.length = redir_or_pipe->consumed;
this->buff += consumed; this->buff += redir_or_pipe->consumed;
break; break;
} }
default: { default: {
// Maybe a redirection like '2>&1', maybe a pipe like 2>|, maybe just a string. // Maybe a redirection like '2>&1', maybe a pipe like 2>|, maybe just a string.
const wchar_t *error_location = this->buff; const wchar_t *error_location = this->buff;
size_t consumed = 0; maybe_t<parsed_redir_or_pipe_t> redir_or_pipe;
enum token_type mode = TOK_NONE;
int fd = -1;
if (iswdigit(*this->buff)) { if (iswdigit(*this->buff)) {
consumed = read_redirection_or_fd_pipe(this->buff, &mode, &fd); redir_or_pipe = read_redirection_or_fd_pipe(this->buff);
} }
if (consumed > 0) { if (redir_or_pipe && redir_or_pipe->consumed > 0) {
// It looks like a redirection or a pipe. But we don't support piping fd 0. Note // It looks like a redirection or a pipe. But we don't support piping fd 0. Note
// that fd 0 may be -1, indicating overflow; but we don't treat that as a tokenizer // that fd 0 may be -1, indicating overflow; but we don't treat that as a tokenizer
// error. // error.
if (mode == TOK_PIPE && fd == 0) { if (redir_or_pipe->type == TOK_PIPE && redir_or_pipe->fd == 0) {
return this->call_error(TOK_INVALID_PIPE, error_location, error_location); return this->call_error(TOK_INVALID_PIPE, error_location, error_location);
} }
result.type = mode; result.type = redir_or_pipe->type;
result.redirected_fd = fd; result.redirected_fd = redir_or_pipe->fd;
result.length = consumed; result.length = redir_or_pipe->consumed;
this->buff += consumed; this->buff += redir_or_pipe->consumed;
} else { } else {
// Not a redirection or pipe, so just a string. // Not a redirection or pipe, so just a string.
result = this->read_string(); result = this->read_string();

View file

@ -10,18 +10,14 @@
/// Token types. /// Token types.
enum token_type { enum token_type {
TOK_NONE, /// Tokenizer not yet constructed TOK_NONE, /// Tokenizer not yet constructed
TOK_ERROR, /// Error reading token TOK_ERROR, /// Error reading token
TOK_STRING, /// String token TOK_STRING, /// String token
TOK_PIPE, /// Pipe token TOK_PIPE, /// Pipe token
TOK_END, /// End token (semicolon or newline, not literal end) TOK_END, /// End token (semicolon or newline, not literal end)
TOK_REDIRECT_OUT, /// redirection token TOK_REDIRECT, /// redirection token
TOK_REDIRECT_APPEND, /// redirection append token TOK_BACKGROUND, /// send job to bg token
TOK_REDIRECT_IN, /// input redirection token TOK_COMMENT /// comment token
TOK_REDIRECT_FD, /// redirection to new fd token
TOK_REDIRECT_NOCLOB, /// redirection token
TOK_BACKGROUND, /// send job to bg token
TOK_COMMENT /// comment token
}; };
/// Tokenizer error types. /// Tokenizer error types.
@ -35,6 +31,14 @@ enum tokenizer_error {
TOK_INVALID_PIPE TOK_INVALID_PIPE
}; };
enum class redirection_type_t {
overwrite, // normal redirection: > file.txt
append, // appending redirection: >> file.txt
input, // input redirection: < file.txt
fd, // fd redirection: 2>&1
noclob // noclobber redirection: >? file.txt
};
/// Flag telling the tokenizer to accept incomplete parameters, i.e. parameters with mismatching /// Flag telling the tokenizer to accept incomplete parameters, i.e. parameters with mismatching
/// paranthesis, etc. This is useful for tab-completion. /// paranthesis, etc. This is useful for tab-completion.
#define TOK_ACCEPT_UNFINISHED 1 #define TOK_ACCEPT_UNFINISHED 1
@ -127,13 +131,13 @@ wcstring tok_first(const wcstring &str);
/// Helper function to determine redirection type from a string, or TOK_NONE if the redirection is /// Helper function to determine redirection type from a string, or TOK_NONE if the redirection is
/// invalid. Also returns the fd by reference. /// invalid. Also returns the fd by reference.
enum token_type redirection_type_for_string(const wcstring &str, int *out_fd = NULL); maybe_t<redirection_type_t> redirection_type_for_string(const wcstring &str, int *out_fd = NULL);
/// Helper function to determine which fd is redirected by a pipe. /// Helper function to determine which fd is redirected by a pipe.
int fd_redirected_by_pipe(const wcstring &str); int fd_redirected_by_pipe(const wcstring &str);
/// Helper function to return oflags (as in open(2)) for a redirection type. /// Helper function to return oflags (as in open(2)) for a redirection type.
int oflags_for_redirection_type(enum token_type type); int oflags_for_redirection_type(redirection_type_t type);
enum move_word_style_t { enum move_word_style_t {
move_word_style_punctuation, // stop at punctuation move_word_style_punctuation, // stop at punctuation