単純なbashスクリプトがあります。
#!/bin/bash
enp="$(ip route | grep "192.168.1.1" | cut -d' ' -f 3)"
while true; do
vnstat -l -i "$enp" --style 4 --json | awk '{ print strftime("[%d/%b/%Y:%H:%M:%S]"), $0 }' >> raw/caching_server_network.json
printf -- '-%.0s' {1..100}; echo "" >> raw/caching_server_network.json
sleep 10
done
現在、このスクリプトは各vnstat出力の先頭にstrftimeを追加し、呼び出すたびにstrftimeが更新されます。出力:
[04/Jan/2022:13:31:17] {"index":1012,"seconds":2024,"rx":{"ratestring":"13.59 Mbit/s","bytespersecond":1698943,"packetspersecond":15220,"bytes":3397887,"packets":30441,"totalbytes":29895276777,"totalpackets":43496469},"tx":{"ratestring":"2.81 Gbit/s","bytespersecond":350918926,"packetspersecond":233842,"bytes":701837853,"packets":467685,"totalbytes":518923724379,"totalpackets":354639132}}
[04/Jan/2022:13:31:41] {"index":1013,"seconds":2026,"rx":{"ratestring":"206.68 Mbit/s","bytespersecond":25834765,"packetspersecond":31840,"bytes":51669531,"packets":63681,"totalbytes":29946946308,"totalpackets":43560150},"tx":{"ratestring":"3.15 Gbit/s","bytespersecond":394123782,"packetspersecond":269030,"bytes":788247565,"packets":538060,"totalbytes":519711971944,"totalpackets":355177192}}
私がしたいのは、日付/時刻文字列にミリ秒を含めることです。しかし、問題が発生しました。日付出力は更新されなくなりました。各呼び出しは文字列に同じ時間を追加します。
私は以下を試しました:
#!/bin/bash
enp="$(ip route | grep "192.168.1.1" | cut -d' ' -f 3)"
while true; do
vnstat -l -i "$enp" --style 4 --json | awk '{ print "'"$(date +%F:%T.%3N)"'", $0 }' >> raw/caching_server_network.json
printf -- '-%.0s' {1..100}; echo "" >> raw/caching_server_network.json
sleep 10
done
出力:
04/Jan/2022:13:30:11.743 {"index":975,"seconds":1950,"rx":{"ratestring":"73.04 Mbit/s","bytespersecond":9130121,"packetspersecond":25590,"bytes":18260242,"packets":51181,"totalbytes":29368713172,"totalpackets":42315654},"tx":{"ratestring":"3.90 Gbit/s","bytespersecond":486961964,"packetspersecond":326200,"bytes":973923928,"packets":652400,"totalbytes":498158299423,"totalpackets":340676257}}
04/Jan/2022:13:30:11.743 {"index":976,"seconds":1952,"rx":{"ratestring":"12.23 Mbit/s","bytespersecond":1529119,"packetspersecond":17962,"bytes":3058239,"packets":35924,"totalbytes":29371771411,"totalpackets":42351578},"tx":{"ratestring":"4.18 Gbit/s","bytespersecond":522336815,"packetspersecond":346742,"bytes":1044673631,"packets":693484,"totalbytes":499202973054,"totalpackets":341369741}}
上記の出力で、2行の日付/時刻がどのように等しいかを確認してください。日付呼び出しを使用するときに、vnstat呼び出しごとに更新された日付をどのように取得しますか?
答え1
各呼び出しは時間を変更しますvnstat
が、vnstat
複数の行が出力される場合、すべての行は同じ時間を受け取ります。これはシェルが実行され、date
出力が引数に挿入されるためですawk
。
各行の新しいタイムスタンプを取得するには、私のコンピュータにdate
タイムスタンプがawk
ないvnstat
ため、より簡単な例を作成しました。
#! /bin/bash
while true; do
for i in {1..3} ; do
echo $i
sleep .1
done | awk '{ system("date +%F:%T.%3N\\ | tr -d \\\\n"); print $0 }'
printf -- '-%.0s' {1..80}; echo ""
sleep 1
done
tr
タイムスタンプの後の改行文字を削除します。
答え2
シェルループを使用して各反復ごとにawk
ユーティリティdate
を呼び出す代わりにsleep
(より悪くはコマンド出力の各行を呼び出す)代わりに、内部的にすべての操作を実行できるようにすべての操作を実行しますperl
。
perl -MPOSIX -MTime::HiRes=gettimeofday -e '
while (1) {
open CMD, "-|", @ARGV;
while (<CMD>) {
my ($s,$us) = gettimeofday;
printf "%s.%03d %s", strftime("%F:%T", localtime$s), $us/1000, $_
}
print "-" x 100 . "\n";
sleep 10;
}' vnstat -l -i "$enp" --style 4 --json >> file.log
またはzsh
:
zmodload zsh/zselect
ts_format='%D{%F:%T.%3.}'
sep=${(l[80][-])}
while true; do
vnstat -l -i "$enp" --style 4 --json |
while IFS= read -r line; do
print -r ${(%)ts_format} $line
done
echo $sep
zselect -t 1000 # cs
done >> file.log