Debian 7 には次のシェルスクリプトがあります。
#!/bin/sh
case "$IFACE" in
lo)
# The loopback interface does not count.
# only run when some other interface comes up
exit 0
;;
*)
;;
esac
(
flock -e 200
FLAGFILE=/var/run/cyvo-auto-started
if [ -e $FLAGFILE ]; then
exit 0
else
touch $FLAGFILE
fi
# Launch CyVo without the lock and as cyvo user
flock -u 200
sudo -u cyvo /usr/local/bin/cyvo-launch.sh
) 200>/var/lock/cyvo-autostart-lock # word unexpected here? (line 29)
それが私に与えるエラーは次のとおりです。
./cyvo: 29: ./cyvo: Syntax error: word unexpected
これは提案された構文なので本当に混乱しています。グループ注文する。/var/lock/cyvo-autostart-lock
を指すファイル記述子200を渡します()
。
私は何が間違っていましたか?
答え1
シェルスクリプトで移植可能に使用できる唯一のファイル記述子番号は、0から9まで(つまり、単一の数値)です。引用するPOSIX:
開いたファイルは、0 から始まる 10 進数で表されます。可能な最大値は実装によって定義されます。ただし、すべての実装はアプリケーションを使用するために少なくとも0〜9(含む)をサポートする必要があります。この数字を「ファイル記述子」と呼びます。 (…)ファイル記述子番号を指定するために、リダイレクト演算子の前に1つ以上の数字(<スペース>文字は許可されていません)が続くことがあります。
これ文法明らかに、リダイレクト演算子の前に複数の数字がある可能性があります。ただし、ほとんどのシェルは、演算子の前の数字が単一の数字で構成されている場合、ファイル記述子としてのみ解析します。たとえば、POSIX仕様によれば、ファイルecho 10>a
ディスクリプタ10をリダイレクトして空のa
ファイルを作成する必要があります。そうしないと、シェルがファイルディスクリプタ番号として10をサポートしていないため、エラーが発生します。このエラーは、プロセスで開かれたファイル数の制限を超えるファイル記述子の数によっても発生する可能性があります。
しかし、実際にはほとんどのシェルはそれをecho 10 >a
write 10
inに解析しますa
。 ATT ksh93 および posh も dash、mksh、zsh と同様に仕様に準拠していません。
$ ksh -c 'echo 10>a; wc -c a'
3 a
$ posh -c 'echo 10>a; wc -c a'
3 a
Linuxでは、POSIX準拠のbash、yash、およびBusyBoxのashのみが見つかりました。
$ bash -c 'echo 10>a; wc -c a'
0 a
同様に(…) 200>somefile
、リダイレクトに解析する必要があり、シェルがこれらの高いファイル記述子をサポートしているかどうかに応じて動作する場合と動作しない場合があります。ただし、ほとんどのシェルは200
この部分をファイル記述子番号ではなく通常のトークンに解析するため、構文エラーを報告します。
関連する注意では、空白行はecho 01>&1
ファイル01
記述子番号で解析する必要があるため、印刷する必要がありますが、再度 bash と BusyBox ash のみを実行できます。
したがって、実際には1桁の数字のファイル記述子のみを使用できます。シェルスクリプトから。記述子0、1、2は標準的な意味を有し、記述子3~9は自由に使用できる。シェルは内部目的のためにfd 3–9を使用しません。どのファイルディスクリプタを選択しても、シェルの起動時にファイルディスクリプタがすでに開いている可能性がありますが、これは問題ではありません。スクリプト内のすべてのファイル記述子の再割り当てはスクリプト内に保持され、スクリプトの親プロセスには影響しません。
答え2
もう一度考えてみると、私の元の回答よりも単純なアプローチ(元の許可された回答だったので以下にコピーします)は、ファイル記述子を200
< 9から変更することです。たとえば、5. したがって、200
スクリプトの3つのインスタンスをすべて次に変更します5
。
#!/bin/sh
case "$IFACE" in
lo)
# The loopback interface does not count.
# only run when some other interface comes up
exit 0
;;
*)
;;
esac
(
flock -e 5
FLAGFILE=/var/run/cyvo-auto-started
if [ -e $FLAGFILE ]; then
exit 0
else
touch $FLAGFILE
fi
# Launch CyVo without the lock and as cyvo user
flock -u 5
sudo -u cyvo /usr/local/bin/cyvo-launch.sh
) 5>/var/lock/cyvo-autostart-lock
元の答え:
@GlennJackmanが提案したように、sh
ファイル記述子> 9を処理できないか、スクリプトが機能しない他の機能があるためですsh
(dash
Debianベースのディストリビューション)。試してみましたがbash
効果が良かったので、ただ使ってみてくださいbash
。最初の行を次に変更します。
#!/bin/sh
到着
#!/bin/bash
これにより、スクリプトが期待どおりに機能するはずです。