TISbackup/libtisbackup/backup_xva.py

272 lines
12 KiB
Python
Raw Normal View History

2013-05-23 10:19:43 +02:00
#!/usr/bin/python
# -*- coding: utf-8 -*-
# -----------------------------------------------------------------------
# This file is part of TISBackup
#
# TISBackup is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# TISBackup is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with TISBackup. If not, see <http://www.gnu.org/licenses/>.
#
# -----------------------------------------------------------------------
2014-07-25 15:04:10 +02:00
from __future__ import with_statement
2013-05-23 10:19:43 +02:00
import logging
import re
import os
import datetime
2014-07-25 15:04:10 +02:00
import urllib
2013-05-23 10:19:43 +02:00
import socket
2014-07-25 15:04:10 +02:00
import tarfile
import hashlib
2013-05-23 10:19:43 +02:00
from stat import *
import ssl
from common import *
import XenAPI
if hasattr(ssl, '_create_unverified_context'):
ssl._create_default_https_context = ssl._create_unverified_context
2013-05-23 10:19:43 +02:00
class backup_xva(backup_generic):
"""Backup a VM running on a XCP server as a XVA file (requires xe tools and XenAPI)"""
type = 'xen-xva'
2014-07-25 15:04:10 +02:00
2013-05-23 10:19:43 +02:00
required_params = backup_generic.required_params + ['xcphost','password_file','server_name']
2015-07-28 10:48:45 +02:00
optional_params = backup_generic.optional_params + ['enable_https', 'halt_vm', 'verify_export', 'reuse_snapshot', 'ignore_proxies', 'use_compression' ]
2014-07-25 15:04:10 +02:00
enable_https = "no"
halt_vm = "no"
verify_export = "no"
reuse_snapshot = "no"
2015-06-30 11:18:11 +02:00
ignore_proxies = "yes"
2015-07-28 10:48:45 +02:00
use_compression = "true"
if str2bool(ignore_proxies):
2015-06-30 11:18:11 +02:00
os.environ['http_proxy']=""
os.environ['https_proxy']=""
2014-07-25 15:04:10 +02:00
def verify_export_xva(self,filename):
self.logger.debug("[%s] Verify xva export integrity",self.server_name)
tar = tarfile.open(filename)
2014-07-25 15:04:10 +02:00
members = tar.getmembers()
for tarinfo in members:
if re.search('^[0-9]*$',os.path.basename(tarinfo.name)):
sha1sum = hashlib.sha1(tar.extractfile(tarinfo).read()).hexdigest()
sha1sum2 = tar.extractfile(tarinfo.name+'.checksum').read()
if not sha1sum == sha1sum2:
raise Exception("File corrupt")
tar.close()
def export_xva(self, vdi_name, filename, halt_vm,dry_run,enable_https=True, reuse_snapshot="no"):
2013-05-23 10:19:43 +02:00
user_xen, password_xen, null = open(self.password_file).read().split('\n')
session = XenAPI.Session('https://'+self.xcphost)
try:
session.login_with_password(user_xen,password_xen)
except XenAPI.Failure, error:
msg,ip = error.details
2014-07-25 15:04:10 +02:00
2013-05-23 10:19:43 +02:00
if msg == 'HOST_IS_SLAVE':
xcphost = ip
session = XenAPI.Session('https://'+xcphost)
session.login_with_password(user_xen,password_xen)
2014-07-25 15:04:10 +02:00
if not session.xenapi.VM.get_by_name_label(vdi_name):
return "bad VM name: %s" % vdi_name
2014-07-25 15:04:10 +02:00
2013-05-23 10:19:43 +02:00
vm = session.xenapi.VM.get_by_name_label(vdi_name)[0]
status_vm = session.xenapi.VM.get_power_state(vm)
2014-07-25 15:04:10 +02:00
self.logger.debug("[%s] Check if previous fail backups exist",vdi_name)
backups_fail = files = [f for f in os.listdir(self.backup_dir) if f.startswith(vdi_name) and f.endswith(".tmp")]
for backup_fail in backups_fail:
self.logger.debug('[%s] Delete backup "%s"', vdi_name, backup_fail)
os.unlink(os.path.join(self.backup_dir, backup_fail))
#add snapshot option
if not str2bool(halt_vm):
2014-07-25 15:04:10 +02:00
self.logger.debug("[%s] Check if previous tisbackups snapshots exist",vdi_name)
old_snapshots = session.xenapi.VM.get_by_name_label("tisbackup-%s"%(vdi_name))
self.logger.debug("[%s] Old snaps count %s", vdi_name, len(old_snapshots))
if len(old_snapshots) == 1 and str2bool(reuse_snapshot) == True:
snapshot = old_snapshots[0]
self.logger.debug("[%s] Reusing snap \"%s\"", vdi_name, session.xenapi.VM.get_name_description(snapshot))
vm = snapshot # vm = session.xenapi.VM.get_by_name_label("tisbackup-%s"%(vdi_name))[0]
else:
self.logger.debug("[%s] Deleting %s old snaps", vdi_name, len(old_snapshots))
for old_snapshot in old_snapshots:
self.logger.debug("[%s] Destroy snapshot %s",vdi_name,session.xenapi.VM.get_name_description(old_snapshot))
try:
for vbd in session.xenapi.VM.get_VBDs(old_snapshot):
if session.xenapi.VBD.get_type(vbd) == 'CD' and session.xenapi.VBD.get_record(vbd)['empty'] == False:
session.xenapi.VBD.eject(vbd)
else:
vdi = session.xenapi.VBD.get_VDI(vbd)
if not 'NULL' in vdi:
session.xenapi.VDI.destroy(vdi)
session.xenapi.VM.destroy(old_snapshot)
except XenAPI.Failure, error:
return("error when destroy snapshot %s"%(error))
now = datetime.datetime.now()
self.logger.debug("[%s] Snapshot in progress",vdi_name)
2014-07-25 15:04:10 +02:00
try:
snapshot = session.xenapi.VM.snapshot(vm,"tisbackup-%s"%(vdi_name))
self.logger.debug("[%s] got snapshot %s", vdi_name, snapshot)
2014-07-25 15:04:10 +02:00
except XenAPI.Failure, error:
return("error when snapshot %s"%(error))
#get snapshot opaqueRef
vm = session.xenapi.VM.get_by_name_label("tisbackup-%s"%(vdi_name))[0]
session.xenapi.VM.set_name_description(snapshot,"snapshot created by tisbackup on: %s"%(now.strftime("%Y-%m-%d %H:%M")))
else:
2014-07-25 15:04:10 +02:00
self.logger.debug("[%s] Status of VM: %s",self.backup_name,status_vm)
if status_vm == "Running":
self.logger.debug("[%s] Shudown in progress",self.backup_name)
if dry_run:
print "session.xenapi.VM.clean_shutdown(vm)"
2014-07-25 15:04:10 +02:00
else:
session.xenapi.VM.clean_shutdown(vm)
2013-05-23 10:19:43 +02:00
try:
try:
2014-07-25 15:04:10 +02:00
filename_temp = filename+".tmp"
2013-05-23 10:19:43 +02:00
self.logger.debug("[%s] Copy in progress",self.backup_name)
if not str2bool(self.use_compression):
2015-10-20 10:22:40 +02:00
socket.setdefaulttimeout(120)
2014-11-20 15:48:29 +01:00
scheme = "http://"
if str2bool(enable_https):
2014-11-20 15:48:29 +01:00
scheme = "https://"
2015-07-28 10:48:45 +02:00
url = scheme+user_xen+":"+password_xen+"@"+self.xcphost+"/export?use_compression="+self.use_compression+"&uuid="+session.xenapi.VM.get_uuid(vm)
2014-07-25 15:04:10 +02:00
urllib.urlretrieve(url, filename_temp)
urllib.urlcleanup()
2013-05-23 10:19:43 +02:00
2014-11-20 15:48:56 +01:00
except Exception as e:
self.logger.error("[%s] error when fetching snap: %s", "tisbackup-%s"%(vdi_name), e)
2014-07-25 15:04:10 +02:00
if os.path.exists(filename_temp):
os.unlink(filename_temp)
2014-11-20 15:48:56 +01:00
raise
2014-07-25 15:04:10 +02:00
2013-05-23 10:19:43 +02:00
finally:
if not str2bool(halt_vm):
2014-07-25 15:04:10 +02:00
self.logger.debug("[%s] Destroy snapshot",'tisbackup-%s'%(vdi_name))
try:
for vbd in session.xenapi.VM.get_VBDs(snapshot):
if session.xenapi.VBD.get_type(vbd) == 'CD' and session.xenapi.VBD.get_record(vbd)['empty'] == False:
session.xenapi.VBD.eject(vbd)
else:
vdi = session.xenapi.VBD.get_VDI(vbd)
if not 'NULL' in vdi:
session.xenapi.VDI.destroy(vdi)
session.xenapi.VM.destroy(snapshot)
2014-07-25 15:04:10 +02:00
except XenAPI.Failure, error:
return("error when destroy snapshot %s"%(error))
2014-07-25 15:04:10 +02:00
elif status_vm == "Running":
2013-05-23 10:19:43 +02:00
self.logger.debug("[%s] Starting in progress",self.backup_name)
if dry_run:
print "session.xenapi.Async.VM.start(vm,False,True)"
else:
session.xenapi.Async.VM.start(vm,False,True)
2014-07-25 15:04:10 +02:00
2013-05-23 10:19:43 +02:00
session.logout()
2014-07-25 15:04:10 +02:00
if os.path.exists(filename_temp):
2017-07-24 18:23:15 +02:00
tar = os.system('tar tf "%s" > /dev/null' % filename_temp)
if not tar == 0:
os.unlink(filename_temp)
2013-05-23 10:19:43 +02:00
return("Tar error")
if str2bool(self.verify_export):
2014-07-25 15:04:10 +02:00
self.verify_export_xva(filename_temp)
os.rename(filename_temp,filename)
2013-05-23 10:19:43 +02:00
return(0)
2014-07-25 15:04:10 +02:00
2013-05-23 10:19:43 +02:00
def do_backup(self,stats):
try:
dest_filename = os.path.join(self.backup_dir,"%s-%s.%s" % (self.backup_name,self.backup_start_date,'xva'))
2014-07-25 15:04:10 +02:00
options = []
2013-05-23 10:19:43 +02:00
options_params = " ".join(options)
cmd = self.export_xva( vdi_name= self.server_name,filename= dest_filename, halt_vm= self.halt_vm, enable_https=self.enable_https, dry_run= self.dry_run, reuse_snapshot=self.reuse_snapshot)
2013-05-23 10:19:43 +02:00
if os.path.exists(dest_filename):
stats['written_bytes'] = os.stat(dest_filename)[ST_SIZE]
stats['total_files_count'] = 1
stats['written_files_count'] = 1
stats['total_bytes'] = stats['written_bytes']
else:
stats['written_bytes'] = 0
stats['backup_location'] = dest_filename
if cmd == 0:
stats['log']='XVA backup from %s OK, %d bytes written' % (self.server_name,stats['written_bytes'])
stats['status']='OK'
2013-05-23 10:19:43 +02:00
else:
2014-07-25 15:04:10 +02:00
raise Exception(cmd)
2013-05-23 10:19:43 +02:00
except BaseException , e:
stats['status']='ERROR'
stats['log']=str(e)
raise
2015-04-01 16:37:47 +02:00
def register_existingbackups(self):
"""scan backup dir and insert stats in database"""
registered = [b['backup_location'] for b in self.dbstat.query('select distinct backup_location from stats where backup_name=?',(self.backup_name,))]
filelist = os.listdir(self.backup_dir)
filelist.sort()
for item in filelist:
if item.endswith('.xva'):
dir_name = os.path.join(self.backup_dir,item)
if not dir_name in registered:
start = (datetime.datetime.strptime(item,self.backup_name+'-%Y%m%d-%Hh%Mm%S.xva') + datetime.timedelta(0,30*60)).isoformat()
if fileisodate(dir_name)>start:
stop = fileisodate(dir_name)
else:
stop = start
self.logger.info('Registering %s started on %s',dir_name,start)
self.logger.debug(' Disk usage %s','du -sb "%s"' % dir_name)
if not self.dry_run:
size_bytes = int(os.popen('du -sb "%s"' % dir_name).read().split('\t')[0])
else:
size_bytes = 0
self.logger.debug(' Size in bytes : %i',size_bytes)
if not self.dry_run:
self.dbstat.add(self.backup_name,self.server_name,'',\
backup_start=start,backup_end = stop,status='OK',total_bytes=size_bytes,backup_location=dir_name,TYPE='BACKUP')
else:
self.logger.info('Skipping %s, already registered',dir_name)
2013-05-23 10:19:43 +02:00
register_driver(backup_xva)
if __name__=='__main__':
logger = logging.getLogger('tisbackup')
logger.setLevel(logging.DEBUG)
formatter = logging.Formatter('%(asctime)s %(levelname)s %(message)s')
handler = logging.StreamHandler()
handler.setFormatter(formatter)
logger.addHandler(handler)
cp = ConfigParser()
cp.read('/opt/tisbackup/configtest.ini')
b = backup_xva()
b.read_config(cp)