blob: 2fcfa0b642491e558c684c069e96cd76db0fcb1a [file] [log] [blame]
#!/bin/bash
## Licensed to the Apache Software Foundation (ASF) under one
## or more contributor license agreements. See the NOTICE file
## distributed with this work for additional information
## regarding copyright ownership. The ASF licenses this file
## to you under the Apache License, Version 2.0 (the
## "License"); you may not use this file except in compliance
## with the License. You may obtain a copy of the License at
##
## http://www.apache.org/licenses/LICENSE-2.0
##
## Unless required by applicable law or agreed to in writing,
## software distributed under the License is distributed on an
## "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
## KIND, either express or implied. See the License for the
## specific language governing permissions and limitations
## under the License.
version=0.9.1
OPTIND=1
verb=0
logDir="/nsr/logs/cloudstack/"
snapPrefix="CSBKP_"$RANDOM"_"
clusterClient=""
networkerServer=""
hvVersion=""
libvVersion=""
apiVersion=""
kvmDName=""
kvmDUuid=""
logFile=""
mediaPool=""
retentionTime=""
log () {
[[ "$verb" -eq 1 ]] && builtin echo "$@"
if [[ "$1" == "-ne" || "$1" == "-e" || "$1" == "-n" ]]; then
builtin echo -e "$(date '+%Y-%m-%d %H-%M-%S>')" "${@: 2}" >> "$logFile"
else
builtin echo "$(date '+%Y-%m-%d %H-%M-%S>')" "$@" >> "$logFile"
fi
}
vercomp(){
local a b IFS=. -; set -f
printf -v a %08d $1; printf -v b %08d $3
test $a "$2" $b
}
usage() {
echo "
Usage:[-v] [-h] [-l log_dir] [-dr] [-s networker_server] [-c networker_cluster_client] [-t target_vm] [-u target_uuid] [-p snapprefix] [-P media_pool ] [-R retention_time ]
Options:
-h Help and usage
-v Enable verbose mode
-l log_dir. Specify log directory. Default is /nsr/logs/cloudstack
-s networker_server Specifiy the EMC Networker server we are going to use
-c networker_cluster_client Specify the EMC Networker client CLUSTER to use
-t target_vm KVM domain to backup
-u target_uuid KVM domain to backup
-p Snapshot Prefix for backups
-P mediaPool EMC Networker Media Pool
-R retention_time Backup retention time
Supplements Apache Cloudstack B&R Framework EMC Networker plugin and performs the backup of the Virtual Machines
"
}
sanity_checks() {
log "Performing environment sanity checks..."
log -ne "\t[1] Checking if Networker is installed\t"
if [[ $(systemctl list-unit-files | grep networker) = *networker* ]]; then
log "Success"
else
log "Failure"
log -e "\n\tNetworker Service NOT FOUND. Make sure that Networker client is properly installed"
exit 1
fi
log -ne "\t[2] Checking if Networker is running\t"
if [[ $(systemctl is-active networker) = *active* ]]; then
log "Success"
else
log "Failure"
log -e "\n\tNetworker Service is not running. Investigate Networker logs, startup server and try again"
exit 2
fi
log -ne "\t[3] Checking Networker DNS Resolution\t"
if [[ $(getent hosts "$networkerServer") = *$networkerServer* ]]; then
log "Success"
else
log "Failure"
log -e "\n\tNetworker Server cannot be resolved. Backups will most probably fail. Consider adding the ip/hostname to /etc/host or fix DNS resolution"
exit 3
fi
log -ne "\t[4] Checking QEMU / Libvirt Versions \t"
hvVersion=$(virsh version | grep hypervisor | awk '{print $(NF)}')
libvVersion=$(virsh version | grep libvirt | awk '{print $(NF)}' | tail -n 1)
apiVersion=$(virsh version | grep API | awk '{print $(NF)}')
if vercomp "$hvVersion" \> 2.1.2; then
log -n "Success"
log -ne "\t\t [ Libvirt: $libvVersion apiVersion: $apiVersion ]"
echo
else
log "Failure"
log -e "\n\tYour QEMU version $hvVersion is unsupported. Consider upgrading at least to latest QEMU at branch 2"
exit 4
fi
log -ne "\t[4] Checking Permissions \t\t"
if groups $USER | grep -q '\blibvirt\b'; then
log -n "Success"
log -ne "\t\t User $USER is part of libvirt group"
echo
else
log "Failure - User $USER is not part of libvirt group"
exit 6
fi
log "Environment Sanity Checks successfully passed"
}
echo "
Cloudstack B&R Framework - EMC Networker backup script
Version $version
"
backup_domain() {
name=$1
snapName=$2
log "Preparing snapshots and gathering information for backing up domain $name under snapshot name $snapName"
log "Retention time is $retentionTime"
declare -A TRGSRC
while IFS=',' read -r TARGET SOURCE
do
if [[ $SOURCE != "-" ]]; then
TRGSRC+=(["$TARGET"]="$SOURCE")
fi
done < <(virsh -c qemu:///system domblklist "$name" --details | grep file | grep -v 'cdrom' | grep -v 'floppy' | sed 's/ */,/g' | cut -d',' -f 4-)
diskspec=""
for target in "${!TRGSRC[@]}"; do
log -e "\tDisk for $target is at ${TRGSRC[${target}]}"
diskspec="$diskspec --diskspec $target,snapshot=external"
disks="$disks ${TRGSRC[${target}]} "
done
cmd="$(virsh -c qemu:///system snapshot-create-as --domain "$name" --name "$snapName" --no-metadata --atomic --quiesce --disk-only "$diskspec")"
retVal=$?
log "$cmd"
if [ "$retVal" -ne 0 ]; then
log "Agent not responding, trying to snapshot directly"
cmd="$(virsh -c qemu:///system snapshot-create-as --domain "$name" --name "$snapName" --no-metadata --atomic --disk-only "$diskspec")"
retVal=$?
if [ "$retVal" -ne 0 ]; then
log "Failed to create snapshot for $name"
exit 7
fi
log "Created snapshot(s) for $name"
fi
cmd="$(save -LL -q -e "${retentionTime}" -s "$networkerServer" -c "$clusterClient" -N "$name" -b "$mediaPool" $disks)"
retVal=$?
log "$cmd"
echo "$cmd" | grep -oE 'savetime=[0-9]{10}'
if [ $retVal -ne 0 ]; then
log "Unable to backup $disks for $name"
else
log "Backup $disks for $name completed!!"
fi
#Merge changes and conclude
SNAPSHOTS="$(virsh -c qemu:///system domblklist "$name" --details | grep file | grep -v 'cdrom' | grep -v 'floppy' | awk '{print $4}')"
for target in "${!TRGSRC[@]}"; do
log "Merging Snasphots for $target"
cmd="$(virsh -c qemu:///system blockcommit "$name" "$target" --active --pivot)"
retVal=$?
log "$cmd"
if [ $retVal -ne 0 ]; then
log "Unable to merge disk %target changes for domain $name"
exit 8
fi
done
#Clean snapshots
for snapshot in $SNAPSHOTS; do
log "Deleting Snapshot $snapshot"
cmd=$(rm -f "$snapshot")
retVal=$?
log "$cmd"
if [ $retVal -ne 0 ]; then
log "Unable to delete snapshot $snapshot"
exit 8
fi
log "Deleted Snapshot: $snapshot"
done
}
while getopts "h?vs:l:c:t:u:p:P:R:" opt; do
case "$opt" in
h|\?)
usage
exit 254
;;
c) clusterClient="$OPTARG"
;;
s) networkerServer="$OPTARG"
;;
l) logDir="$OPTARG"
;;
t) kvmDName="$OPTARG"
;;
u) kvmDUuid="$OPTARG"
;;
p) snapPrefix="$OPTARG"
;;
P) mediaPool="$OPTARG"
;;
R) retentionTime="$OPTARG"
;;
v) verb=1
;;
esac
done
shift $((OPTIND-1))
[ "${1:-}" = "--" ] && shift
if [[ -z "$networkerServer" || -z "$kvmDName" || -z "$clusterClient" || -z "$kvmDUuid" ]]; then
usage
exit 255
fi
if [ ! -d "$logDir" ]; then
mkdir -p "$logDir"
fi
logFile="$logDir/BACKUP-$kvmDName-$(date +'%Y_%m_%d_%I_%M_%p').log"
# Perform Initial sanity checks
sanity_checks
log -e "\nLooking for domain $kvmDName with UUID $kvmDUuid"
if [[ "$kvmDName" == $(virsh -c qemu:///system domname "$kvmDUuid" | head -1) && "$kvmDUuid" == $(virsh -c qemu:///system domuuid "$kvmDName" | head -1) ]]; then
log "Domain found...."
else
log "Domain not found on this host. Aborting....."
log "Check for the location of the Instance in the cloudstack management console"
exit 5
fi
backup_domain "$kvmDName" "$snapPrefix$kvmDName"
exit 0