Reallow "2>>&1" as a redirection

Appending to an fd doesn't really make sense, but we allowed the
syntax previously and it was actually used.

It's not too harmful to allow it, so let's just do that again.

For the record: Zsh also allows it, bash doesn't.

Fixes #6614

(cherry picked from commit aba900a71f)
This commit is contained in:
Fabian Homborg 2020-02-17 08:52:41 +01:00 committed by David Adam
parent 971a837031
commit 2af174513e
2 changed files with 8 additions and 2 deletions

View file

@ -362,6 +362,7 @@ maybe_t<pipe_or_redir_t> pipe_or_redir_t::from_string(const wchar_t *buff) {
} }
case L'>': { case L'>': {
consume(L'>'); consume(L'>');
if (try_consume(L'>')) result.mode = redirection_mode_t::append;
if (try_consume(L'|')) { if (try_consume(L'|')) {
// Note we differ from bash here. // Note we differ from bash here.
// Consider `echo foo 2>| bar` // Consider `echo foo 2>| bar`
@ -374,6 +375,7 @@ maybe_t<pipe_or_redir_t> pipe_or_redir_t::from_string(const wchar_t *buff) {
: STDOUT_FILENO; // like >| : STDOUT_FILENO; // like >|
} else if (try_consume(L'&')) { } else if (try_consume(L'&')) {
// This is a redirection to an fd. // This is a redirection to an fd.
// Note that we allow ">>&", but it's still just writing to the fd - "appending" to it doesn't make sense.
result.mode = redirection_mode_t::fd; result.mode = redirection_mode_t::fd;
result.fd = has_fd ? parse_fd(fd_start, fd_end) // like 1>&2 result.fd = has_fd ? parse_fd(fd_start, fd_end) // like 1>&2
: STDOUT_FILENO; // like >&2 : STDOUT_FILENO; // like >&2
@ -381,11 +383,10 @@ maybe_t<pipe_or_redir_t> pipe_or_redir_t::from_string(const wchar_t *buff) {
// This is a redirection to a file. // This is a redirection to a file.
result.fd = has_fd ? parse_fd(fd_start, fd_end) // like 1> file.txt result.fd = has_fd ? parse_fd(fd_start, fd_end) // like 1> file.txt
: STDOUT_FILENO; // like > file.txt : STDOUT_FILENO; // like > file.txt
if (result.mode != redirection_mode_t::append) result.mode = redirection_mode_t::overwrite;
// Note 'echo abc >>? file' is valid: it means append and noclobber. // Note 'echo abc >>? file' is valid: it means append and noclobber.
// But here "noclobber" means the file must not exist, so appending // But here "noclobber" means the file must not exist, so appending
// can be ignored. // can be ignored.
result.mode = redirection_mode_t::overwrite;
if (try_consume(L'>')) result.mode = redirection_mode_t::append;
if (try_consume(L'?')) result.mode = redirection_mode_t::noclob; if (try_consume(L'?')) result.mode = redirection_mode_t::noclob;
} }
break; break;

View file

@ -8,6 +8,11 @@ end
outnerr 0 &| count outnerr 0 &| count
#CHECK: 2 #CHECK: 2
outnerr appendfd 2>>&1
#CHECK: out appendfd
#CHECK: err appendfd
set -l tmpdir (mktemp -d) set -l tmpdir (mktemp -d)
outnerr overwrite &>$tmpdir/file.txt outnerr overwrite &>$tmpdir/file.txt
cat $tmpdir/file.txt cat $tmpdir/file.txt