Falls man in der komfortablen Lage ist, zwei Virtualisierungsmaschinen parallel laufen zu haben, kann man recht komfortabel und im laufenden Betrieb ein Snapshot des LVs von Server 1 auf Server 2 übertragen.
Das Skript geht davon aus, dass auf beiden Servern identische VMs mit LVM eingerichtet sind. Außerdem kann man passwortlos per SSH auf die Backup-Maschine zugreifen.
Falls eine Email-Adresse angegeben wird, muss der Email-Versand eingerichtet sein.
Im wesentlichen wird sichergestellt, dass der aktuelle Server der ist, auf dem die VM läuft und diese auf dem Backup-Server nicht läuft. Die Größe wird verglichen und ggf. angepasst.
Es gibt zweifellos noch einige Verbesserungsmöglichkeiten - aber bei uns läuft das erstmal so und macht nachts ein schnelles (ca. 8 minuten) Backup unseres Webservers (~80GB).
Vielleicht ist es (ganz oder in Teilen) hilfreich für irgendjemanden.
#!/bin/bash # this script assumes, that the target server is passwordless ssh-able # and that the user has sudo-rights on the local and the target-system # for mailing functionality install and setup mailutils / ssmtp # run on one of the kvm hosts user="serveradmin" source="<IP ADDRESS OF RUNNING SERVER>" target="<IP ADDRESS OF BACKUP-MACHINE>" mailto="" snapshot="backup" log="vmbackup.log" echo "Running vmbackup on $(hostname) at $(date) - syncing $source to $target" | tee "$log" if [ "$#" -eq 0 ]; then echo "X no vm name passed - ABORT" | tee -a "$log" exit 1 fi vm=$1 # check for needed packages/install - local and remote echo "* apt install needed packages" sudo apt install -y pigz pv &> /dev/null ssh $user@$target sudo apt install -y pigz pv &> /dev/null # check if vm is running locally (it has to) if [ "$(virsh list | grep $vm | grep -c 'running')" == "1" ]; then echo "* vm $vm running on source - OK" | tee -a "$log" else echo "X vm $vm not running on source - ABORT" | tee -a "$log" exit 1 fi # check if vm is running on remote machine (its not allowed to) if [ "$(virsh -c qemu+ssh://$user@$target/system list | grep $vm | grep -c 'running')" == "1" ]; then echo "X vm $vm is running on backup target - ABORT" | tee -a "$log" exit 1 else echo "* vm $vm not running on remote machine - OK" | tee -a "$log" fi # get name and size - local and remote - of lv from vm name (hopefully) locallvname=$(virsh dumpxml $vm | grep -A 3 '<disk ' | grep source | cut -d "'" -f 2) locallvsize=$(sudo lvs $locallvname -o LV_SIZE --noheadings --units b --nosuffix | tr -d " " ) remotelvname=$(virsh -c qemu+ssh://$user@$target/system dumpxml $vm | grep -A 3 '<disk ' | grep source | cut -d "'" -f 2) remotelvsize=$(ssh $user@$target sudo lvs $remotelvname -o LV_SIZE --noheadings --units B --nosuffix | tr -d " " ) # compare sizes if [ "$locallvsize" != "$remotelvsize" ]; then echo "* local size of lv ($locallvsize) differs from remote ($remotelvsize) - resizing remote lv" | tee -a "$log" ssh $user@$target sudo lvextend -L ${locallvsize}B $remotelvname remotelvsize=$(ssh $user@$target sudo lvs $remotelvname -o LV_SIZE --noheadings --units B --nosuffix | tr -d " " ) if [ "$locallvsize" != "$remotelvsize" ]; then echo "X size still does not matter - GIVING UP" | tee -a "$log" break 1 fi else echo "* local and remote size match - OK" | tee -a "$log" fi # create snapshot if [ "$(sudo lvs | grep -c $snapshot)" != "0" ]; then echo "X the snapshot-lv exists - ABORT" | tee -a $log exit 1 else echo "* create snapshot lv" | tee -a $log sudo lvcreate -l100%FREE -s -n $snapshot $locallvname &> /dev/null lvsnapshot=$(sudo lvs -o lv_path | grep $snapshot | tr -d " ") fi # ssh dd snapshot to other kvm host echo "* copy snapshot lv to backup lv" | tee -a "$log" sudo dd if=$lvsnapshot bs=4096 | pv | pigz | ssh $user@$target "pigz -d | sudo dd of=$remotelvname bs=4096" # delete snapshot lv echo "* remove snapshot lv" sudo lvremove -y $lvsnapshot &> /dev/null echo "* finished backup" | tee -a "$log" # mail log if [ "$mailto" != "" ]; then echo "* send log by mail" mail -s "SCRIPT: vmbackup finished on $(hostname)" $mailto < $log fi