mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-03 16:48:45 +00:00
Support explicitly separated output on stderr
In principle this would allow 'string split' or whatever to output to stderr and not lose the item separation. In practice this is not used but it fixes a TODO.
This commit is contained in:
parent
bcfc54fdaa
commit
81a39be0bb
2 changed files with 9 additions and 8 deletions
15
src/exec.cpp
15
src/exec.cpp
|
@ -469,12 +469,6 @@ static bool handle_builtin_output(parser_t &parser, const std::shared_ptr<job_t>
|
||||||
// If we are directing output to a buffer, then we can just transfer it directly without needing
|
// If we are directing output to a buffer, then we can just transfer it directly without needing
|
||||||
// to write to the bufferfill pipe. Note this is how we handle explicitly separated stdout
|
// to write to the bufferfill pipe. Note this is how we handle explicitly separated stdout
|
||||||
// output (i.e. string split0) which can't really be sent through a pipe.
|
// output (i.e. string split0) which can't really be sent through a pipe.
|
||||||
// TODO: we're sloppy about handling explicitly separated output.
|
|
||||||
// Theoretically we could have explicitly separated output on stdout and also stderr output; in
|
|
||||||
// that case we ought to thread the exp-sep output through to the io buffer. We're getting away
|
|
||||||
// with this because the only thing that can output exp-sep output is `string split0` which
|
|
||||||
// doesn't also produce stderr. Also note that we never send stderr to a buffer, so there's no
|
|
||||||
// need for a similar check for stderr.
|
|
||||||
bool stdout_done = false;
|
bool stdout_done = false;
|
||||||
if (output_buffer && stdout_io && stdout_io->io_mode == io_mode_t::bufferfill) {
|
if (output_buffer && stdout_io && stdout_io->io_mode == io_mode_t::bufferfill) {
|
||||||
auto stdout_buffer = dynamic_cast<const io_bufferfill_t *>(stdout_io.get())->buffer();
|
auto stdout_buffer = dynamic_cast<const io_bufferfill_t *>(stdout_io.get())->buffer();
|
||||||
|
@ -482,13 +476,20 @@ static bool handle_builtin_output(parser_t &parser, const std::shared_ptr<job_t>
|
||||||
stdout_done = true;
|
stdout_done = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool stderr_done = false;
|
||||||
|
if (errput_buffer && stderr_io && stderr_io->io_mode == io_mode_t::bufferfill) {
|
||||||
|
auto stderr_buffer = dynamic_cast<const io_bufferfill_t *>(stderr_io.get())->buffer();
|
||||||
|
stderr_buffer->append_from_wide_buffer(*errput_buffer);
|
||||||
|
stderr_done = true;
|
||||||
|
}
|
||||||
|
|
||||||
// Figure out any data remaining to write. We may have none in which case we can short-circuit.
|
// Figure out any data remaining to write. We may have none in which case we can short-circuit.
|
||||||
std::string outbuff;
|
std::string outbuff;
|
||||||
if (output_buffer && !stdout_done) {
|
if (output_buffer && !stdout_done) {
|
||||||
outbuff = wcs2string(output_buffer->newline_serialized());
|
outbuff = wcs2string(output_buffer->newline_serialized());
|
||||||
}
|
}
|
||||||
std::string errbuff;
|
std::string errbuff;
|
||||||
if (errput_buffer) {
|
if (errput_buffer && !stderr_done) {
|
||||||
errbuff = wcs2string(errput_buffer->newline_serialized());
|
errbuff = wcs2string(errput_buffer->newline_serialized());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
2
src/io.h
2
src/io.h
|
@ -313,7 +313,7 @@ class io_buffer_t {
|
||||||
/// Lock for appending.
|
/// Lock for appending.
|
||||||
std::mutex append_lock_{};
|
std::mutex append_lock_{};
|
||||||
|
|
||||||
/// Read a bit, filling the buffer. The append lock must be held.
|
/// Read some, filling the buffer. The append lock must be held.
|
||||||
/// \return positive on success, 0 if closed, -1 on error (in which case errno will be set).
|
/// \return positive on success, 0 if closed, -1 on error (in which case errno will be set).
|
||||||
ssize_t read_once(int fd);
|
ssize_t read_once(int fd);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue