표준 오류 스트림을 잡는 방법 (stderr)? –libdir=/usr/lib

오디오 클립의 메타 정보를 얻기 위해 ffmpeg를 사용하고 있습니다. 그러나 나는 그것을 잡을 수 없습니다.

    $ ffmpeg -i 01-Daemon.mp3  |grep -i Duration
    FFmpeg version SVN-r15261, Copyright (c) 2000-2008 Fabrice Bellard, et al.
      configuration: --prefix=/usr --bindir=/usr/bin
      --datadir=/usr/share/ffmpeg --incdir=/usr/include/ffmpeg --libdir=/usr/lib
      --mandir=/usr/share/man --arch=i386 --extra-cflags=-O2
      ...

이 ffmpeg 출력은 stderr로 전달됩니다.

$ ffmpeg -i 01-Daemon.mp3 2> /dev/null

그래서 grep이 일치하는 줄을 잡기 위해 오류 스트림을 읽을 수 없다고 생각합니다. grep이 오류 스트림을 읽도록하려면 어떻게해야합니까?

nixCraft 링크를 사용하여 표준 오류 스트림을 표준 출력 스트림으로 리디렉션 한 다음 grep이 작동했습니다.

$ ffmpeg -i 01-Daemon.mp3 2>&1 | grep -i Duration
  Duration: 01:15:12.33, start: 0.000000, bitrate: 64 kb/s

그러나 stderr을 stdout으로 리디렉션하지 않으려면 어떻게해야합니까?



답변

bash익명 파이프를 사용 하지 않는 이유를 사용하는 경우 본질적으로 phunehehe가 말한 것에 대한 약어입니다.

ffmpeg -i 01-Daemon.mp3 2> >(grep -i Duration)


답변

일반적인 쉘 (zsh조차 포함)은 stdout에서 stdin까지의 파이프를 허용하지 않습니다. 그러나 모든 Bourne 스타일 쉘은 파일 디스크립터 재 할당 (에서와 같이 1>&2)을 지원합니다 . 따라서 일시적으로 stdout을 fd 3으로, stderr을 stdout으로 전환 한 다음 나중에 fd 3을 stdout에 다시 넣을 수 있습니다. 경우 stuff표준 출력에 약간의 출력과 표준 에러에 대한 몇 가지 출력을 생성하고, 적용 할 filter표준 출력은 그대로두고 오류 출력에, 당신은 사용할 수 있습니다 { stuff 2>&1 1>&3 | filter 1>&2; } 3>&1.

$ stuff () {
  echo standard output
  echo more output
  echo standard error 1>&2
  echo more error 1>&2
}
$ filter () {
  grep a
}
$ { stuff 2>&1 1>&3 | filter 1>&2; } 3>&1
standard output
more output
standard error

답변

이것은 phunehehe의 “temp file trick”과 유사하지만 대신 명명 된 파이프를 사용하여 결과를 출력 할 때 약간 더 가까운 결과를 얻을 수 있으므로 장기 실행 명령에 유용 할 수 있습니다.

$ mkfifo mypipe
$ command 2> mypipe | grep "pattern" mypipe

이 구성에서 stderr은 “mypipe”라는 파이프로 연결됩니다. 때문에 grep파일 인수로 호출 된, 그것의 입력을 STDIN에 보이지 않는 것입니다. 불행히도, 일단 당신이 한 후에는 그 명명 된 파이프를 정리해야합니다.

Bash 4를 사용하는 경우에 대한 바로 가기 구문 command1 2>&1 | command2command1 |& command2있습니다. 그러나 이것은 순전히 구문 바로 가기라고 생각하며 STDERR을 STDOUT으로 리디렉션합니다.


답변

Gilles와 Stefan Lasiewski의 답변은 모두 좋지만이 방법은 더 간단합니다.

ffmpeg -i 01-Daemon.mp3 2>&1 >/dev/null | grep "pattern"

ffmpeg'sstdout을 인쇄 하고 싶지 않다고 가정합니다 .

작동 방식 :

  • 먼저 파이프
    • ffmpeg 및 grep이 시작되고 ffmpeg의 stdout이 grep의 stdin으로 이동
  • 다음에서 왼쪽에서 오른쪽으로 리디렉션
    • ffmpeg의 stderr는 stdout이 무엇이든 (현재 파이프) 설정됩니다.
    • ffmpeg의 stdout이 / dev / null로 설정되었습니다.

답변

이 테스트에 사용 된 스크립트는 아래를 참조하십시오.

Grep은 stdin에서만 작동 할 수 있으므로 Grep가 구문 분석 할 수있는 형식으로 stderr 스트림을 변환해야합니다.

일반적으로 stdout 및 stderr은 모두 화면에 인쇄됩니다.

$ ./stdout-stderr.sh
./stdout-stderr.sh: Printing to stdout
./stdout-stderr.sh: Printing to stderr

stdout을 숨기고 여전히 stderr를 인쇄하려면 다음과 같이하십시오.

$ ./stdout-stderr.sh >/dev/null
./stdout-stderr.sh: Printing to stderr

그러나 grep은 stderr에서 작동하지 않습니다! 다음 명령은 ‘err’가 포함 된 행을 표시하지 않을 것으로 예상하지만 그렇지 않습니다.

$ ./stdout-stderr.sh >/dev/null |grep --invert-match err
./stdout-stderr.sh: Printing to stderr

해결책은 다음과 같습니다.

다음 Bash 구문은 출력을 stdout에 숨기지 만 stderr은 계속 표시합니다. 먼저 stdout을 / dev / null로 파이프 한 다음 stderr을 stdout으로 변환합니다. Unix 파이프는 stdout에서만 작동하기 때문입니다. 여전히 텍스트를 잡을 수 있습니다.

$ ./stdout-stderr.sh 2>&1 >/dev/null | grep err
./stdout-stderr.sh: Printing to stderr

(상기 명령 유의 다른./command >/dev/null 2>&1매우 일반적인 명령이다).

테스트에 사용되는 스크립트는 다음과 같습니다. 이것은 stdout에 한 줄을 stderr에 한 줄을 인쇄합니다.

#!/bin/sh

# Print a message to stdout
echo "$0: Printing to stdout"
# Print a message to stderr
echo "$0: Printing to stderr" >&2

exit 0

답변

을 사용하여 한 명령의 출력을 다른 명령으로 파이프하면 |표준 출력 만 리디렉션됩니다. 그 이유를 설명해야합니다

ffmpeg -i 01-Daemon.mp3 | grep -i Duration

원하는 것을 출력하지 않습니다 (그러나 작동합니다).

오류 출력을 표준 출력으로 리디렉션하지 않으려면 오류 출력을 파일로 리디렉션 한 다음 나중에 grep 할 수 있습니다

ffmpeg -i 01-Daemon.mp3 2> /tmp/ffmpeg-error
grep -i Duration /tmp/ffmpeg-error

답변

스트림을 교환 할 수 있습니다. 이를 grep통해 원래 표준 오류 스트림으로 이동하면서 원래 터미널의 표준 출력으로 이동 한 출력을 얻을 수 있습니다.

somecommand 3>&2 2>&1 1>&3- | grep 'pattern'

이것은 먼저 출력을 위해 열린 새 파일 디스크립터 (3)를 작성하고 표준 오류 스트림 ( 3>&2)으로 설정하여 작동합니다 . 그런 다음 표준 오류를 표준 출력 ( 2>&1)으로 리디렉션합니다 . 마지막으로 표준 출력이 원래 표준 오류로 리디렉션되고 새 파일 설명자가 닫힙니다 ( 1>&3-).

귀하의 경우 :

ffmpeg -i 01-Daemon.mp3 3>&2 2>&1 1>&3- | grep -i Duration

그것을 테스트 :

$ ( echo "error" >&2; echo "output" ) 3>&2 2>&1 1>&3- | grep "error"
output
error

$ ( echo "error" >&2; echo "output" ) 3>&2 2>&1 1>&3- | grep -v "error"
output