mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-14 05:53:59 +00:00
Fix comment parsing inside command substitutions and brackets
This commit is contained in:
parent
627033f447
commit
2ef12af60e
7 changed files with 52 additions and 6 deletions
|
@ -93,6 +93,7 @@ Scripting improvements
|
|||
- ``abbr -q`` returns the correct exit status when given multiple abbreviation names as arguments (:issue:`8431`).
|
||||
- ``command -v`` returns an exit status of 127 instead of 1 if no command was found (:issue:`8547`).
|
||||
- ``argparse`` with ``--ignore-unknown`` no longer breaks with multiple unknown options in a short option group (:issue:`8637`).
|
||||
- Comments inside command substitutions or brackets now correctly ignore parentheses, quotes, and brackets (:issue:`7866`, :issue:`8022`).
|
||||
|
||||
Interactive improvements
|
||||
------------------------
|
||||
|
|
|
@ -523,6 +523,13 @@ const wchar_t *quote_end(const wchar_t *pos, wchar_t quote) {
|
|||
return nullptr;
|
||||
}
|
||||
|
||||
const wchar_t *comment_end(const wchar_t *pos) {
|
||||
do {
|
||||
pos++;
|
||||
} while (*pos && *pos != L'\n');
|
||||
return pos;
|
||||
}
|
||||
|
||||
void fish_setlocale() {
|
||||
// Use various Unicode symbols if they can be encoded using the current locale, else a simple
|
||||
// ASCII char alternative. All of the can_be_encoded() invocations should return the same
|
||||
|
|
|
@ -451,6 +451,11 @@ std::unique_ptr<T> make_unique(Args &&...args) {
|
|||
/// \param quote the quote to use, usually pointed to by \c pos.
|
||||
const wchar_t *quote_end(const wchar_t *pos, wchar_t quote);
|
||||
|
||||
/// This functions returns the end of the comment substring beginning at \c pos.
|
||||
///
|
||||
/// \param pos the position where the comment starts, including the '#' symbol.
|
||||
const wchar_t *comment_end(const wchar_t *pos);
|
||||
|
||||
/// This function should be called after calling `setlocale()` to perform fish specific locale
|
||||
/// initialization.
|
||||
void fish_setlocale();
|
||||
|
|
|
@ -90,6 +90,8 @@ size_t parse_util_get_offset(const wcstring &str, int line, long line_offset) {
|
|||
static int parse_util_locate_cmdsub(const wchar_t *in, const wchar_t **begin, const wchar_t **end,
|
||||
bool allow_incomplete, bool *inout_is_quoted) {
|
||||
bool escaped = false;
|
||||
bool is_first = true;
|
||||
bool is_token_begin = true;
|
||||
bool syntax_error = false;
|
||||
int paran_count = 0;
|
||||
std::vector<int> quoted_cmdsubs;
|
||||
|
@ -122,6 +124,10 @@ static int parse_util_locate_cmdsub(const wchar_t *in, const wchar_t **begin, co
|
|||
if (!escaped) {
|
||||
if (*pos == L'\'' || *pos == L'"') {
|
||||
if (!process_opening_quote(*pos)) break;
|
||||
} else if (*pos == L'\\') {
|
||||
escaped = true;
|
||||
} else if (*pos == L'#' && is_token_begin) {
|
||||
pos = comment_end(pos) - 1;
|
||||
} else {
|
||||
if (*pos == L'(') {
|
||||
if ((paran_count == 0) && (paran_begin == nullptr)) {
|
||||
|
@ -161,12 +167,12 @@ static int parse_util_locate_cmdsub(const wchar_t *in, const wchar_t **begin, co
|
|||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (*pos == '\\') {
|
||||
escaped = !escaped;
|
||||
is_token_begin = is_token_delimiter(pos[0], is_first, pos[1]);
|
||||
} else {
|
||||
escaped = false;
|
||||
is_token_begin = false;
|
||||
}
|
||||
is_first = false;
|
||||
}
|
||||
|
||||
syntax_error |= (paran_count < 0);
|
||||
|
|
|
@ -154,6 +154,7 @@ tok_t tokenizer_t::read_string() {
|
|||
int slice_offset = 0;
|
||||
const wchar_t *const buff_start = this->token_cursor;
|
||||
bool is_first = true;
|
||||
bool is_token_begin = true;
|
||||
|
||||
auto process_opening_quote = [&](wchar_t quote) -> const wchar_t * {
|
||||
const wchar_t *end = quote_end(this->token_cursor, quote);
|
||||
|
@ -192,6 +193,8 @@ tok_t tokenizer_t::read_string() {
|
|||
// has been explicitly ignored (escaped).
|
||||
else if (c == L'\\') {
|
||||
mode |= tok_modes::char_escape;
|
||||
} else if (c == L'#' && is_token_begin) {
|
||||
this->token_cursor = comment_end(this->token_cursor) - 1;
|
||||
} else if (c == L'(') {
|
||||
paran_offsets.push_back(this->token_cursor - this->start);
|
||||
expecting.push_back(L')');
|
||||
|
@ -278,8 +281,9 @@ tok_t tokenizer_t::read_string() {
|
|||
FLOGF(error, msg.c_str(), c, c, int(mode_begin), int(mode));
|
||||
#endif
|
||||
|
||||
this->token_cursor++;
|
||||
is_token_begin = is_token_delimiter(this->token_cursor[0], is_first, this->token_cursor[1]);
|
||||
is_first = false;
|
||||
this->token_cursor++;
|
||||
}
|
||||
|
||||
if (!this->accept_unfinished && (mode != tok_modes::regular_text)) {
|
||||
|
@ -540,8 +544,7 @@ maybe_t<tok_t> tokenizer_t::next() {
|
|||
while (*this->token_cursor == L'#') {
|
||||
// We have a comment, walk over the comment.
|
||||
const wchar_t *comment_start = this->token_cursor;
|
||||
while (this->token_cursor[0] != L'\n' && this->token_cursor[0] != L'\0')
|
||||
this->token_cursor++;
|
||||
this->token_cursor = comment_end(this->token_cursor);
|
||||
size_t comment_len = this->token_cursor - comment_start;
|
||||
|
||||
// If we are going to continue after the comment, skip any trailing newline.
|
||||
|
@ -679,6 +682,10 @@ maybe_t<tok_t> tokenizer_t::next() {
|
|||
return result;
|
||||
}
|
||||
|
||||
bool is_token_delimiter(wchar_t c, bool is_first, maybe_t<wchar_t> next) {
|
||||
return c == L'(' || !tok_is_string_character(c, is_first, next);
|
||||
}
|
||||
|
||||
wcstring tok_first(const wcstring &str) {
|
||||
tokenizer_t t(str.c_str(), 0);
|
||||
if (auto token = t.next()) {
|
||||
|
|
|
@ -133,6 +133,9 @@ class tokenizer_t : noncopyable_t {
|
|||
}
|
||||
};
|
||||
|
||||
/// Tests if this character can delimit tokens.
|
||||
bool is_token_delimiter(wchar_t c, bool is_first, maybe_t<wchar_t> next);
|
||||
|
||||
/// Returns only the first token from the specified string. This is a convenience function, used to
|
||||
/// retrieve the first token of a string. This can be useful for error messages, etc. On failure,
|
||||
/// returns the empty string.
|
||||
|
|
|
@ -533,6 +533,23 @@ echo banana
|
|||
echo (echo hello\\)
|
||||
# CHECK: hello\
|
||||
|
||||
# This used to be a parse error - #7866.
|
||||
echo (echo foo;#)
|
||||
)
|
||||
# CHECK: foo
|
||||
echo (echo bar #'
|
||||
)
|
||||
# CHECK: bar
|
||||
echo (#"
|
||||
echo baz)
|
||||
# CHECK: baz
|
||||
|
||||
# Make sure we don't match up brackets within comments (#8022).
|
||||
$fish -c 'echo f[oo # not valid, no matching ]'
|
||||
# CHECKERR: fish: Unexpected end of string, square brackets do not match
|
||||
# CHECKERR: echo f[oo # not valid, no matching ]
|
||||
# CHECKERR: {{ }}^
|
||||
|
||||
# Should fail because $PWD is read-only.
|
||||
for PWD in foo bar
|
||||
true
|
||||
|
|
Loading…
Reference in a new issue