次のように、サーバーの負荷平均を決定するスクリプトを作成しました。
#!/bin/bash
loadavg=`top -b -n1 | grep -i load | awk -F, '{print$4}'| awk -F: '{print $2}'`
if [ "${loadavg}" -le 1 ]
then
echo "OK - Load Average = ${loadavg} | Load_Average=${loadavg}"
exit 0;
elif [ "${loadavg}" -gt 1 ] && [ "${loadavg}" -le 2 ]
then
echo "WARNING - Load Average = ${loadavg} | Load_Average=${loadavg}"
exit 1;
elif [ "${loadavg}" -gt 2 ]
then
echo "CRITICAL - Load Average = ${loadavg} | Load_Average=${loadavg}"
exit 2;
else
echo "UNKNOWN - Load Average = NaN | Load_Average=NaN"
fi
スクリプトを実行すると、次のエラーが表示されます。
./loadavg.sh
./loadavg.sh: line 5: [: 0.06: integer expression expected
./loadavg.sh: line 9: [: 0.06: integer expression expected
./loadavg.sh: line 13: [: 0.06: integer expression expected
UNKNOWN - Load Average = NaN | Load_Average=NaN
答え1
bash
ksh93
(またはzsh
1とは反対に) は浮動小数点演算を実行できません。awk
ただし、 は実行できるので、 ですべてを実行できますawk
。
top
また、負荷を取得するために (そして 1 秒待つ)を使用する必要はありません。負荷を取得する標準的な方法は を使用することですuptime
。
uptime | awk '{load=+$(NF-2)}
load > 2 {print "CRITICAL: " load; exit 2}
load > 1 {print "WARNING: " load; exit 1}
{print "OK: " load; exit 0}
END {if (!NR) {print "UNKNOWN"; exit 3}'
exit
1ただし、非整数を比較する場合は、構文の代わりにzsh
を使用する必要があります。(( loadavg > 2 ))
[ "$loadavg" -gt 2 ]
答え2
top -b -n1 | grep -i load | awk -F, '{print$4}'| awk -F: '{print $2}'
nada を返すので、エラーが発生します。
答え3
はloadavg
null です。次の理由で構文エラーが発生します[
:
$ top -b n1 | grep -i load | awk -F, '{print$4}'| awk -F: '{print $2}'
<blank line here>
次のように変更する必要があります:
$ top -b n1 | grep -i load | awk -F, '{print$4}'| awk -F: '{print $1}'
0.24
ただし、スクリプトでは新しいテストを使用する必要があります。新しいテストはこの問題に対処できます。
$ [[ "" -gt 1 ]] || echo "OK"
OK
年上の方と[
:
$ [ "" -gt 1 ] || echo "OK"
bash: [: : integer expression expected
OK
更新しました
bash
浮動小数点数を処理できないため、比較 (新しいテストを使用した場合でも[[..]]
) でエラーが表示されます。
このタスクを実行するには、次のような別のツールを使用することもできますbc
。awk
例:
$ [[ $(echo "3 > 1" | bc -l) == 1 ]] && echo OK
OK
$[[ $(echo "3 < 1" | bc -l) == 1 ]] || echo Not OK
Not OK
答え4
まず、top を実行するのは、単に「アップタイム」を取得するよりもコストがかかります。
$ uptime
16:15:38 up 6 days, 23:22, 23 users, load average: 0.99, 0.82, 0.70
2 番目に、@Stéphane Chazelas が述べたように、bash は浮動小数点演算を好みません。
$ [[ "0.1" -lt 1 ]]
bash: [[: 0.1: syntax error: invalid arithmetic operator (error token is ".1")
幸いなことに、bashの拡張でそれが可能になります。男の強打:
${parameter%word} ${parameter%%word} Remove matching suffix pattern. The word is expanded to produce a pattern just as in pathname expansion. If the pattern matches a trailing portion of the expanded value of parameter, then the result of the expansion is the expanded value of parameter with the shortest matching pattern (the ``%'' case) or the longest matching pattern (the ``%%'' case) deleted. If parameter is @ or *, the pattern removal operation is applied to each positional parameter in turn, and the expansion is the resultant list. If parameter is an array variable sub- scripted with @ or *, the pattern removal operation is applied to each member of the array in turn, and the expansion is the resultant list.
したがって、「%.*」を使用して小数部分を削除するのは非常に簡単になります。
$ la=`uptime | sed 's/.*average: \([0-9\.]*\).*/\1/'`
$ if [ ${la%.*} -lt 1 ]; then echo "OK - Load average is $la"; fi
OK - Load average is 0.42