bash: 파이프라인 스크립트의 tty에서 읽는 중

bash: 파이프라인 스크립트의 tty에서 읽는 중

작업 중인 두 개의 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: 321pipe.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.shscript2.shscript1.sh

답변1

read -p호출과 읽기는 모두 현재 터미널 script.sh에서 pipe.sh수행되며 파이프라인의 명령은 병렬로 실행되므로 어느 명령이 사용자가 먼저 입력한 데이터를 가져올지 추측할 수 없습니다.

from은 프롬프트를 표시할 수 있지만 사용자가 입력한 문자열은 from에서 읽을 수 있으며 read -p그 반대의 경우도 마찬가지입니다.script.shread -ppipe.sh

이와 같은 파이프에서는 a | b계속 하기 전에 b입력을 기다리는 것이 쉽지만 a그 반대의 경우는 아닙니다. 파이프가 버퍼링 중이므로 a읽은 데이터가 없다는 것을 알기 전에 많은 데이터를 작성해야 합니다.b

이 문제를 해결하는 한 가지 방법은 일종의 "루프 파이프"로 bstdout과 stdin을 연결 하고 ( ) 처럼 stdin의 입력을 기다리도록 수정하는 것 입니다. aascript.shbpipe.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

관련 정보