오디오 클립의 메타 정보를 얻기 위해 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 | command2
이 command1 |& command2
있습니다. 그러나 이것은 순전히 구문 바로 가기라고 생각하며 STDERR을 STDOUT으로 리디렉션합니다.
답변
Gilles와 Stefan Lasiewski의 답변은 모두 좋지만이 방법은 더 간단합니다.
ffmpeg -i 01-Daemon.mp3 2>&1 >/dev/null | grep "pattern"
ffmpeg's
stdout을 인쇄 하고 싶지 않다고 가정합니다 .
작동 방식 :
- 먼저 파이프
- 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