작업 중인 두 개의 bash 스크립트에 문제가 있습니다.
스크립트 파일
#!/bin/bash
while true; do
read -p "script: read: var: " var
echo "script: write: var: $var"
done
파이프.sh
#!/bin/bash
while read line; do
echo "pipe: read: line: (( $line ))"
read -p "pipe: read var: " var < /dev/tty
echo "pipe: write: var: $var"
done< <(cat)
실행되고 script.sh
출력이 나에게 파이프 되면 pipe.sh
다음과 같은 출력을 얻습니다.
$ ./script.sh | ./pipe.sh
1: script: read: var: 123 # user entering '123'
2: script: read: var: pipe: read: line: (( script: write: var: 123 ))
3: pipe: read var: 321 # user entering '321'
4: script: read: var: 456 # user entering '456'
5: pipe: write: var: 456
6: pipe: read: line: (( script: write: var: 321 ))
7: pipe: read var:
보시다시피 4번째 줄에 도달할 때까지는 모든 것이 괜찮은 것 같습니다. 나는 원래 4번째 줄 pipe: write: var: 321
이 pipe.sh
. 대신 에서 팁을 얻습니다 script.sh
.
문자열 "456"을 입력하면 이전에 예상했던 줄이 실행되지만 잘못된 문자열이 포함됩니다(예상: "321", "456" 발생). 또한 6행에서는 "456"을 인쇄하는 대신 "321"을 인쇄합니다.
여기서 뭔가가 완전히 잘못되었습니다. 이 문제를 해결하는 방법과 이런 일이 발생하는 이유에 대한 제안이 있으십니까?
고쳐 쓰다:
기본적으로 파이프라인이 아래 코드와 동일한 방식으로 작동하기를 원합니다.
script1.sh
#!/bin/bash
while true; do
read -p "val1: " val1
script2.sh "${val1}"
done
script2.sh
#!/bin/bash
val1="${1}"
read -p "val2: " val2
echo "${val1} ${val2}"
하지만 script2.sh
.script1.sh
script2.sh
script1.sh
답변1
read -p
호출과 읽기는 모두 현재 터미널 script.sh
에서 pipe.sh
수행되며 파이프라인의 명령은 병렬로 실행되므로 어느 명령이 사용자가 먼저 입력한 데이터를 가져올지 추측할 수 없습니다.
from은 프롬프트를 표시할 수 있지만 사용자가 입력한 문자열은 from에서 읽을 수 있으며 read -p
그 반대의 경우도 마찬가지입니다.script.sh
read -p
pipe.sh
이와 같은 파이프에서는 a | b
계속 하기 전에 b
입력을 기다리는 것이 쉽지만 a
그 반대의 경우는 아닙니다. 파이프가 버퍼링 중이므로 a
읽은 데이터가 없다는 것을 알기 전에 많은 데이터를 작성해야 합니다.b
이 문제를 해결하는 한 가지 방법은 일종의 "루프 파이프"로 b
stdout과 stdin을 연결 하고 ( ) 처럼 stdin의 입력을 기다리도록 수정하는 것 입니다. a
a
script.sh
b
pipe.sh
쉘 언어 제한으로 인해 다음을 사용해야 합니다.명명된 파이프그런 이유로. 간단한 예:
cat > circpipe <<'EOT'; chmod 755 circpipe
fifo=$(mktemp -u)
mkfifo "$fifo" || exit 1
exec 3<>"$fifo" >"$fifo" <"$fifo"
rm "$fifo"
echo trigger
"$1" | "$2"
EOT
cat > pipe-head <<'EOT'; chmod 755 pipe-head
while read next; do
read -p "HEAD's prompt>> " var </dev/tty || exit
echo "$var"
done
EOT
cat > pipe-tail <<'EOT'; chmod 755 pipe-tail
while read input; do
echo >&2 " TAIL'input: $input"
read -p " TAIL's prompt>> " var </dev/tty
echo >&2 " TAIL processing <$var>"
echo next # trigger the head of the pipeline
done
EOT
./circpipe ./pipe-head ./pipe-tail
HEAD's prompt>> foo
TAIL'input: foo
TAIL's prompt>> bar
TAIL processing <bar>
HEAD's prompt>> baz
TAIL'input: baz
TAIL's prompt>> quux
TAIL processing <quux>
HEAD's prompt>> ^D$
이것circpipe
스크립트는 일반 쉘 명령을 허용하고 루프에서 벗어나는 "꼬리"를 갖는 보다 일반적인 도구로 만들 수 있습니다.
위의 예와 달리 이는 기본적으로 루프를 "시작"하지 않으므로 -command
인수를 사용해야 합니다. 사용 예:
./circpipe -echo './pipe-head | stdbuf -oL sed s/e/o/g | ./pipe-tail'
HEAD's prompt>> pee
TAIL'input: poo
TAIL's prompt>> lol
TAIL processing <lol>
HEAD's prompt>>^D
링 튜브
#! /bin/sh
# usage: circpipe [-start_command] shell_command
# run 'shell_command' with its stdin connected to its stdout
# via a FIFO
# if 'start_command' is given, run it before `shell_command`
# with its stdout redirected to the same FIFO
[ "$#" -gt 0 ] || exit 0
fifo=$(mktemp -u)
mkfifo "$fifo" || exit 1
exec 3<>"$fifo" >"$fifo" 3<"$fifo"
rm "$fifo"
case $1 in -*) eval "${1#-}"; shift; esac
IFS='|'; eval "<&3 $* &"
exec >&-
wait