所以我得到了这个 bash 脚本,它从远程服务器执行备份命令,每个服务器主机名都在名为/tmp/quarantine.list. 内容是这样的:
server1
server2
Run Code Online (Sandbox Code Playgroud)
server1并且server2是我的远程服务器的 ssh 别名,因此如果我需要访问它们中的任何一个,我只需运行 例如ssh server1。
我需要备份 中的每个服务器/tmp/quarantine.list,因此我使用while循环循环每一行。这是我的代码。
    # Reading from /tmp/quarantine.list
while IFS= read -r list; do
  if [ "$list" == "server4" ]; then
    users="adminjoe"
  fi
  tempfile="${logdir}/rsync_$list.log"
  SECONDS=0
  set +x
  if rsync -ca --stats --info=progress2 --remove-source-files $list:/home/${users}/backup/20* $backup_location/$list >> $tempfile 2>&1; then
    ssh $list "find /home/${users}/backup/20* -type d -empty -delete"
  else
    echo "Error while running rsync"
    exit 1
  fi
  set -x
  elapsed_time=$SECONDS
  transferred_size=$(gawk 'BEGIN { FS=":" } {gsub(/^[ \t]+|[ \t]+$/, "", $2)} /Total transferred file size/ { print $2 }' $tempfile | tr -cd "[:digit:]")
  echo "Backup of $list done at $(/bin/date +'%Y-%m-%d_%H:%M:%S') --> $(bytesToHumanReadable $transferred_size)" | tee -a "${HOME}/backup.log"
  echo "<b>$list --> $(bytesToHumanReadable $transferred_size)</b>" >> $konten
  echo "$transferred_size $list" >> $summaryfile_size
  echo "$elapsed_time $list" >> $summaryfile_time
done < "/tmp/quarantine.list"
rm "/tmp/quarantine.list"
Run Code Online (Sandbox Code Playgroud)
问题是我的脚本只选择 的第一行/tmp/quarantine.list,例如,如果有两个远程隔离。
server2
server6
Run Code Online (Sandbox Code Playgroud)
该脚本仅处理server2,而server6保持不变。奇怪的是,当我将选项设置为 rsync 时,它将毫无问题地--dry-run处理所有行!/tmp/quarantine.list
--dry-run+ echo 'Backup of server2 done at 2021-10-28_16:55:26 --> 3.28 GB'
Backup of server2 done at 2021-10-28_16:55:26 --> 3.28 GB
++ bytesToHumanReadable 3280769140
++ S=("bytes" "kB" "MB" "GB" "TB" "PB" "EB" "YB" "ZB")
++ local i=3280769140 d= s=0 S
++ (( i > 1000 && s < 9-1 ))
++ printf -v d .%02d 14
++ i=3280769
++ s=1
++ (( i > 1000 && s < 9-1 ))
++ printf -v d .%02d 76
++ i=3280
++ s=2
++ (( i > 1000 && s < 9-1 ))
++ printf -v d .%02d 28
++ i=3
++ s=3
++ (( i > 1000 && s < 9-1 ))
++ echo '3.28 GB'
+ echo '<b>server2 --> 3.28 GB</b>'
+ echo '3280769140 server2'
+ echo '1883 server2'
+ IFS=
+ read -r list
+ rm /tmp/quarantine.list
Run Code Online (Sandbox Code Playgroud)
--dry-run+ echo 'Backup of server2 done at 2021-10-28_16:07:14 --> 3.28 GB'
Backup of server2 done at 2021-10-28_16:07:14 --> 3.28 GB
++ bytesToHumanReadable 3280769140
++ S=("bytes" "kB" "MB" "GB" "TB" "PB" "EB" "YB" "ZB")
++ local i=3280769140 d= s=0 S
++ (( i > 1000 && s < 9-1 ))
++ printf -v d .%02d 14
++ i=3280769
++ s=1
++ (( i > 1000 && s < 9-1 ))
++ printf -v d .%02d 76
++ i=3280
++ s=2
++ (( i > 1000 && s < 9-1 ))
++ printf -v d .%02d 28
++ i=3
++ s=3
++ (( i > 1000 && s < 9-1 ))
++ echo '3.28 GB'
+ echo '<b>server2 --> 3.28 GB</b>'
+ echo '3280769140 server2'
+ echo '32 server2'
+ IFS=
+ read -r list
+ '[' server6 == server4 ']'
Run Code Online (Sandbox Code Playgroud)
正如您所看到的,后者继续而server6不是前者。我只是想知道我在这里做错了什么?
如果文件的最后一行不以换行符结尾,则可能会发生这种情况。然后read将实际读取该行,但它将返回非零状态(参考:https: //www.gnu.org/software/bash/manual/bash.html#index-read)。该非零状态导致while循环结束。
处理这个问题的惯用方法是:
while IFS= read -r list || [[ -n $list ]]; do
Run Code Online (Sandbox Code Playgroud)