Pipe only STDERR through a filter

后端 未结 7 997
夕颜
夕颜 2020-11-28 20:53

Is there any way, in bash, to pipe STDERR through a filter before unifying it with STDOUT? That is, I want

STDOUT ────────────────┐
                       ├         


        
相关标签:
7条回答
  • 2020-11-28 21:07

    TL;DR: (bash and zsh)

    $ cmd 2> >(stderr-filter >&2)
    

    Example:

    % cat /non-existant 2> >(tr o X >&2)
    cat: /nXn-existant: NX such file Xr directXry
    %
    

    Many answers on the StackExchange network have the form:

    cat /non-existant 3>&1 1>&2 2>&3 3>&- | sed 's/e/E/g'
    

    This has a built-in assumption: that file descriptor 3 isn't being used for something else.

    Instead, use a named file descriptor, and {ba,z}sh will allocate the next available file descriptor >= 10:

    cat /non-existant {tmp}>&1 1>&2 2>&$tmp {tmp}>&- | sed 's/e/E/g'
    

    Note that named file descriptors aren't supported by POSIX sh.

    The other issue with the above is that the command cannot be piped to further commands without again swapping STDOUT and STDERR back to their original values.

    To allow onward piping in POSIX sh, (and still assuming FD 3 is not it use) it gets complicated:

    (cmd 2>&1 >&3 3>&- | stderr-filter >&2 3>&-) 3>&1
    

    So, Given the assumption and gnarly syntax of this, you're likely to be better off using the simpler bash/zsh syntax shown in the TL;DR above, and explained here.

    0 讨论(0)
  • 2020-11-28 21:08

    TL;DR:

    $ cmd 2> >(stderr-filter >&2)
    

    Example:

    % cat /non-existant 2> >(tr o X >&2)
    cat: /nXn-existant: NX such file Xr directXry
    %
    

    This will work in both bash and zsh. Bash is pretty much ubiquitous these days, however, if you really do need a (really gnarly) solution for POSIX sh, then see here.


    Explanation

    By far, the easiest way to do this is to redirect STDERR via process substitution:

    Process substitution allows a process’s input or output to be referred to using a filename. It takes the form of

    >(list)
    

    The process list is run asynchronously, and its input or output appears as a filename.

    So what you get with process substituion is a filename.

    Just like you could do:

    $ cmd 2> filename
    

    you can do

    $ cmd 2> >(filter >&2)
    

    The >&2 redirect's filter's STDOUT back to the original STDERR.

    0 讨论(0)
  • 2020-11-28 21:11

    Here's an example, modeled after how to swap file descriptors in bash . The output of a.out is the following, without the 'STDXXX: ' prefix.

    STDERR: stderr output
    STDOUT: more regular
    
    ./a.out 3>&1 1>&2 2>&3 3>&- | sed 's/e/E/g'
    more regular
    stdErr output
    

    Quoting from the above link:

    1. First save stdout as &3 (&1 is duped into 3)
    2. Next send stdout to stderr (&2 is duped into 1)
    3. Send stderr to &3 (stdout) (&3 is duped into 2)
    4. close &3 (&- is duped into 3)
    0 讨论(0)
  • 2020-11-28 21:13

    I find the use of bash process substitution easier to remember and use as it reflects the original intention almost verbatim. For example:

    $ cat ./p
    echo stdout
    echo stderr >&2
    $ ./p 2> >(sed -e 's/s/S/') | sed 's/t/T/'
    sTdout
    STderr
    

    uses the first sed command as a filter on stderr only and the second sed command to modify the joined output.

    Note that the white space after 2> is mandatory for the command to be parsed correctly.

    0 讨论(0)
  • 2020-11-28 21:19

    A naive use of process substitution seems to allow filtering of stderr separately from stdout:

    :; ( echo out ; echo err >&2 ) 2> >( sed s/^/e:/ >&2 )
    out
    e:err
    

    Note that stderr comes out on stderr and stdout on stdout, which we can see by wrapping the whole thing in another subshell and redirecting to files o and e

    ( ( echo out ; echo err >&2 ) 2> >( sed s/^/e:/ >&2 ) ) 1>o 2>e
    
    0 讨论(0)
  • 2020-11-28 21:27

    Take a look at named pipes:

    $ mkfifo err
    $ cmd1 2>err |cat - err |cmd2
    
    0 讨论(0)
提交回复
热议问题