Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
slapos
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Léo-Paul Géneau
slapos
Commits
bf25f31d
Commit
bf25f31d
authored
Mar 13, 2024
by
Łukasz Nowak
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
software/kvm: Switch to incremental backups
parent
f77d0cc2
Changes
9
Show whitespace changes
Inline
Side-by-side
Showing
9 changed files
with
311 additions
and
68 deletions
+311
-68
software/kvm/buildout.hash.cfg
software/kvm/buildout.hash.cfg
+9
-5
software/kvm/instance-kvm-export.cfg.jinja2
software/kvm/instance-kvm-export.cfg.jinja2
+21
-9
software/kvm/instance-kvm-import.cfg.jinja2.in
software/kvm/instance-kvm-import.cfg.jinja2.in
+2
-4
software/kvm/instance.cfg.in
software/kvm/instance.cfg.in
+21
-3
software/kvm/software.cfg
software/kvm/software.cfg
+17
-1
software/kvm/template/check-backup-directory.sh
software/kvm/template/check-backup-directory.sh
+37
-0
software/kvm/template/kvm-export.sh.jinja2
software/kvm/template/kvm-export.sh.jinja2
+34
-17
software/kvm/template/kvm-import.sh.jinja2
software/kvm/template/kvm-import.sh.jinja2
+13
-6
software/kvm/test/test.py
software/kvm/test/test.py
+157
-23
No files found.
software/kvm/buildout.hash.cfg
View file @
bf25f31d
...
@@ -15,7 +15,7 @@
...
@@ -15,7 +15,7 @@
[template]
[template]
filename = instance.cfg.in
filename = instance.cfg.in
md5sum =
9ae66fb63a3cdd8072582622aa1bb36c
md5sum =
bf0a70140e4775efaacdd5fe03ac3b62
[template-kvm]
[template-kvm]
filename = instance-kvm.cfg.jinja2
filename = instance-kvm.cfg.jinja2
...
@@ -31,19 +31,19 @@ md5sum = 839fc16c112d3b87e2dbd2e382e326de
...
@@ -31,19 +31,19 @@ md5sum = 839fc16c112d3b87e2dbd2e382e326de
[template-kvm-import]
[template-kvm-import]
filename = instance-kvm-import.cfg.jinja2.in
filename = instance-kvm-import.cfg.jinja2.in
md5sum =
a463a5e3cd2287d275d6943c2a11b7e4
md5sum =
0829d08337a09b8ef017e71fbb9384fb
[template-kvm-import-script]
[template-kvm-import-script]
filename = template/kvm-import.sh.jinja2
filename = template/kvm-import.sh.jinja2
md5sum =
013725987114c82ca3fd11097d0a7f9f
md5sum =
a6e6abb3b17fae37b84233d63014b885
[template-kvm-export]
[template-kvm-export]
filename = instance-kvm-export.cfg.jinja2
filename = instance-kvm-export.cfg.jinja2
md5sum =
34d1b7cc8ca62bfdfce759a1dfbbaccd
md5sum =
a02f0694dcb944c18d99f7f79afa2384
[template-kvm-export-script]
[template-kvm-export-script]
filename = template/kvm-export.sh.jinja2
filename = template/kvm-export.sh.jinja2
md5sum =
64aa1ce8785f6b94aabd787fa3443082
md5sum =
a1da7809d547b4c61e7c6337bf9f8b8a
[template-nginx]
[template-nginx]
filename = template/nginx_conf.in
filename = template/nginx_conf.in
...
@@ -96,3 +96,7 @@ md5sum = e9d40162ba77472775256637a2617d14
...
@@ -96,3 +96,7 @@ md5sum = e9d40162ba77472775256637a2617d14
[boot-image-select-source-config]
[boot-image-select-source-config]
filename = template/boot-image-select-source-config.json.in
filename = template/boot-image-select-source-config.json.in
md5sum = d53afe719e2cbfc2480277af340f8429
md5sum = d53afe719e2cbfc2480277af340f8429
[check-backup-directory.sh]
filename = template/check-backup-directory.sh
md5sum = e569494a941e1d585c2e0bbf070cf1c9
software/kvm/instance-kvm-export.cfg.jinja2
View file @
bf25f31d
...
@@ -6,6 +6,7 @@ extends =
...
@@ -6,6 +6,7 @@ extends =
parts +=
parts +=
cron-entry-backup
cron-entry-backup
${instance-kvm-parts:parts}
${instance-kvm-parts:parts}
check-backup-directory-promise
[slap-parameter]
[slap-parameter]
{% for k, v in slapparameter_dict.items() -%}
{% for k, v in slapparameter_dict.items() -%}
...
@@ -16,13 +17,10 @@ parts +=
...
@@ -16,13 +17,10 @@ parts +=
{%- endif %}
{%- endif %}
{% endfor -%}
{% endfor -%}
# Create the exporter executable, which is a simple shell script
[directory]
[exporter]
tmp = ${buildout:directory}/tmp
recipe = slapos.recipe.template:jinja2
url = {{ template_kvm_export }}
[disk]
output = ${directory:bin}/${slap-parameter:namebase}-exporter
# Resilient stack wants a "wrapper" parameter
wrapper = ${:output}
{%- set disk_type = slapparameter_dict.get('disk-type', 'virtio') %}
{%- set disk_type = slapparameter_dict.get('disk-type', 'virtio') %}
{%- if disk_type == "virtio" %}
{%- if disk_type == "virtio" %}
device = virtio0
device = virtio0
...
@@ -32,12 +30,19 @@ device = ide0-hd0
...
@@ -32,12 +30,19 @@ device = ide0-hd0
{%- else %}
{%- else %}
# unsupported disk-type {{ disk_type }}
# unsupported disk-type {{ disk_type }}
{%- endif %}
{%- endif %}
[exporter]
recipe = slapos.recipe.template:jinja2
url = {{ template_kvm_export }}
output = ${directory:bin}/${slap-parameter:namebase}-exporter
# Resilient stack wants a "wrapper" parameter
wrapper = ${:output}
context =
context =
section directory directory
section directory directory
section buildout buildout
section buildout buildout
key socket_path kvm-instance:socket-path
key socket_path kvm-instance:socket-path
key device :device
section disk disk
raw
gzip_binary {{ gzip_binary
}}
raw
qmpbackup {{ qmpbackup
}}
# Extends publish section with resilient parameters
# Extends publish section with resilient parameters
[publish-connection-information]
[publish-connection-information]
...
@@ -56,3 +61,10 @@ username = {{ slapparameter_dict['monitor-username'] }}
...
@@ -56,3 +61,10 @@ username = {{ slapparameter_dict['monitor-username'] }}
password = {{ slapparameter_dict['monitor-password'] }}
password = {{ slapparameter_dict['monitor-password'] }}
{% endif -%}
{% endif -%}
[check-backup-directory-promise]
# Check that disk image is not corrupted
<= monitor-promise-base
promise = check_command_execute
name = check-backup-directory.py
config-command = {{ check_backup_directory }} ${directory:backup}/${disk:device} ${directory:tmp}
failure_amount = 5
software/kvm/instance-kvm-import.cfg.jinja2.in
View file @
bf25f31d
...
@@ -48,6 +48,7 @@ bin = ${buildout:directory}/bin
...
@@ -48,6 +48,7 @@ bin = ${buildout:directory}/bin
srv = ${buildout:directory}/srv
srv = ${buildout:directory}/srv
var = ${buildout:directory}/var
var = ${buildout:directory}/var
log = ${:var}/log
log = ${:var}/log
tmp = ${buildout:directory}/tmp
scripts = ${:etc}/run
scripts = ${:etc}/run
services = ${:etc}/service
services = ${:etc}/service
novnc-conf = ${:etc}/novnc
novnc-conf = ${:etc}/novnc
...
@@ -65,10 +66,7 @@ output = ${directory:bin}/${slap-parameter:namebase}-importer
...
@@ -65,10 +66,7 @@ output = ${directory:bin}/${slap-parameter:namebase}-importer
wrapper = ${:output}
wrapper = ${:output}
context =
context =
section directory directory
section directory directory
raw zcat_binary {{ zcat_binary }}
raw qmprestore {{ qmprestore }}
raw gzip_binary {{ gzip_binary }}
backup-disk-path = ${directory:backup}/virtual.qcow2
[kvm-disk-image-corruption-bin]
[kvm-disk-image-corruption-bin]
recipe = collective.recipe.template
recipe = collective.recipe.template
...
...
software/kvm/instance.cfg.in
View file @
bf25f31d
...
@@ -122,6 +122,15 @@ template-replicated-destination = ${template-replicated:target}
...
@@ -122,6 +122,15 @@ template-replicated-destination = ${template-replicated:target}
import-list = file parts :template-parts-destination
import-list = file parts :template-parts-destination
file replicated :template-replicated-destination
file replicated :template-replicated-destination
[qmpbackup-binary]
recipe = slapos.cookbook:wrapper
environment =
PATH=${qemu:location}/bin
wrapper-path =
$${buildout:bin-directory}/qmpbackup
command-line =
${buildout:bin-directory}/qmpbackup
[dynamic-template-kvm-export]
[dynamic-template-kvm-export]
recipe = slapos.recipe.template:jinja2
recipe = slapos.recipe.template:jinja2
url = ${template-kvm-export:location}/instance-kvm-export.cfg.jinja2
url = ${template-kvm-export:location}/instance-kvm-export.cfg.jinja2
...
@@ -133,8 +142,18 @@ context =
...
@@ -133,8 +142,18 @@ context =
raw kvm_template $${dynamic-template-kvm:output}
raw kvm_template $${dynamic-template-kvm:output}
raw template_kvm_export ${template-kvm-export-script:target}
raw template_kvm_export ${template-kvm-export-script:target}
key pbsready_export_template template-pbsready-export:output
key pbsready_export_template template-pbsready-export:output
raw gzip_binary ${gzip:location}/bin/gzip
key slapparameter_dict slap-configuration:configuration
key slapparameter_dict slap-configuration:configuration
key qmpbackup qmpbackup-binary:wrapper-path
raw check_backup_directory ${check-backup-directory.sh:output}
[qmprestore-binary]
recipe = slapos.cookbook:wrapper
environment =
PATH=${qemu:location}/bin
wrapper-path =
$${buildout:bin-directory}/qmprestore
command-line =
${buildout:bin-directory}/qmprestore
[dynamic-template-kvm-import]
[dynamic-template-kvm-import]
recipe = slapos.recipe.template:jinja2
recipe = slapos.recipe.template:jinja2
...
@@ -148,5 +167,4 @@ context =
...
@@ -148,5 +167,4 @@ context =
raw template_kvm_import ${template-kvm-import-script:target}
raw template_kvm_import ${template-kvm-import-script:target}
key pbsready_import_template template-pbsready-import:output
key pbsready_import_template template-pbsready-import:output
key slapparameter_dict slap-configuration:configuration
key slapparameter_dict slap-configuration:configuration
raw zcat_binary ${gzip:location}/bin/zcat
key qmprestore qmprestore-binary:wrapper-path
raw gzip_binary ${gzip:location}/bin/gzip
software/kvm/software.cfg
View file @
bf25f31d
...
@@ -10,7 +10,6 @@ extends =
...
@@ -10,7 +10,6 @@ extends =
../../component/netcat/buildout.cfg
../../component/netcat/buildout.cfg
../../component/nginx/buildout.cfg
../../component/nginx/buildout.cfg
../../component/pycurl/buildout.cfg
../../component/pycurl/buildout.cfg
../../component/gzip/buildout.cfg
../../stack/slapos.cfg
../../stack/slapos.cfg
../../stack/resilient/buildout.cfg
../../stack/resilient/buildout.cfg
buildout.hash.cfg
buildout.hash.cfg
...
@@ -19,6 +18,7 @@ extends =
...
@@ -19,6 +18,7 @@ extends =
# to avoid versioning issues
# to avoid versioning issues
common-parts =
common-parts =
template
template
qmpbackup
# XXX: we have to manually add this for resilience
# XXX: we have to manually add this for resilience
pbs-recipe-egg
pbs-recipe-egg
...
@@ -32,6 +32,18 @@ parts = ${:common-parts}
...
@@ -32,6 +32,18 @@ parts = ${:common-parts}
# In qemu builtin vnc server, and make it available only for localhost
# In qemu builtin vnc server, and make it available only for localhost
# so that only novnc can listen to it.
# so that only novnc can listen to it.
[qmpbackup]
recipe = zc.recipe.egg
eggs =
qemu.qmp
qmpbackup
find-links +=
https://github.com/abbbi/qmpbackup/releases/download/v0.37/qmpbackup-0.37.tar.gz
[versions]
qemu.qmp = 0.0.3:whl
qmpbackup = 0.37
[python-with-eggs]
[python-with-eggs]
recipe = zc.recipe.egg
recipe = zc.recipe.egg
interpreter = ${:_buildout_section_name_}
interpreter = ${:_buildout_section_name_}
...
@@ -129,3 +141,7 @@ context =
...
@@ -129,3 +141,7 @@ context =
[boot-image-select-source-config]
[boot-image-select-source-config]
<= download-base
<= download-base
[check-backup-directory.sh]
<= template-base
output = ${buildout:parts-directory}/${:_buildout_section_name_}/check-backup-directory.sh
software/kvm/template/check-backup-directory.sh
0 → 100644
View file @
bf25f31d
#!/bin/bash
directory
=
$1
tmp
=
$2
# support a case of not ready yet directory
if
[
!
-d
$directory
]
;
then
exit
0
fi
tmpfile
=
$(
mktemp
-p
$tmp
)
trap
"rm -fr
$tmpfile
"
EXIT TERM INT
find
$directory
-type
f
-name
'FULL*qcow2'
-printf
'%f\n'
>
$tmpfile
full_amount
=
$(
wc
-l
$tmpfile
|
cut
-d
' '
-f
1
)
if
[
$full_amount
-gt
1
]
;
then
echo
"Too many FULL backups"
cat
$tmpfile
exit
1
fi
find
$directory
-type
f
-name
'INC*qcow2'
-printf
'%f\n'
>
$tmpfile
if
[
$(
wc
-l
$tmpfile
|
cut
-d
' '
-f
1
)
-gt
0
]
&&
[
$full_amount
-eq
0
]
;
then
echo
"INC present but no FULL backup"
cat
$tmpfile
exit
1
fi
find
$directory
-type
f
-name
'*.partial'
-printf
'%f\n'
>
$tmpfile
if
[
$(
wc
-l
$tmpfile
|
cut
-d
' '
-f
1
)
-ne
0
]
;
then
echo
"Partial file present"
cat
$tmpfile
exit
1
fi
exit
0
software/kvm/template/kvm-export.sh.jinja2
View file @
bf25f31d
...
@@ -6,25 +6,42 @@ set -e
...
@@ -6,25 +6,42 @@ set -e
LC_ALL
=
C
LC_ALL
=
C
export
LC_ALL
export
LC_ALL
BACKUP_DIR
={{
directory[
'backup'
]
}}
BACKUP_DIR
={{
directory[
'backup'
]
}}
BACKUP_FILE
=
virtual.qcow2
QMP_CLIENT
={{
buildout[
'directory'
]
}}
/software_release/bin/qemu-qmp-client
log
=
$(
mktemp
--tmpdir
={{
directory[
'tmp'
]
}}
)
trap
"rm -f
$log
"
EXIT TERM INT
$QMP_CLIENT
--socket
{{
socket_path
}}
--drive-backup
$BACKUP_DIR
/
$BACKUP_FILE
{{
device
}}
set
+e
# Due to the way qmp works, the VM file cannot be compressed on the fly.
qmpbackup
=
"{{ qmpbackup }} --socket {{ socket_path }} backup --compress --target
$BACKUP_DIR
--include {{ disk['device'] }}"
# Although the compression step is optional, the importer uses the .gz file
$qmpbackup
--level
auto
>
$log
# if present. So, remove it if you are disabling the compression.
RESULT
=
$?
cat
$log
# The downside of compression, here, is the temporary usage of more disk space
if
[
$RESULT
-ne
0
]
;
then
# in the exporter node. The goal is to minimize disk usage on the PBS server.
# recover from unfinished previous backup
if
egrep
-q
'Partial backup found in.*{{ disk['
device
']}}.*possible broken backup chain. Execute new full backup'
$log
;
then
find
$BACKUP_DIR
/
{{
disk[
'device'
]
}}
-name
'*.partial'
-delete
$qmpbackup
--level
auto
||
exit
$?
echo
"Recovered from partial backup by removing partial"
elif
egrep
-q
'Incremental backup requested but no active bitmap has been found'
$log
;
then
find
$BACKUP_DIR
/
{{
disk[
'device'
]
}}
-name
'*.qcow2'
-delete
$qmpbackup
--level
full
||
exit
$?
echo
"Post take-over cleanup"
else
exit
$RESULT
fi
fi
set
-e
# If you want to compress the file in-place:
# as new style backup went fine delete potential old style backup
# truncate -s $(gzip -c $BACKUP_FILE | dd of=$BACKUP_FILE conv=notrunc 2>&1 | sed -n '$ s/ .*$// p') $BACKUP_FILE
rm
-f
$BACKUP_DIR
/virtual.qcow2
{
,.gz
}
# but the importer script would have to be adapted.
echo
"Compressing backup..."
# cleanup the backup directory from too old backups, especially important after take-over
{{
gzip_binary
}}
--force
--rsyncable
$BACKUP_DIR
/
$BACKUP_FILE
recent_full
=
$(
find
$BACKUP_DIR
-type
f
-name
'FULL-*.qcow2'
-exec
ls
-t1
{}
+ |
head
-n1
)
if
[
x
"
$recent_full
"
!=
x
""
]
;
then
for
f
in
$(
find
$BACKUP_DIR
-type
f
-name
'*qcow2'
\!
-newer
$recent_full
)
;
do
if
[
"
$f
"
!=
"
$recent_full
"
]
;
then
rm
-vf
$f
fi
done
fi
cd
$BACKUP_DIR
&&
find
-type
f
!
-name
backup.signature
-print0
| xargs
-0
sha256sum
|
LC_ALL
=
C
sort
-k
66
>
backup.signature
cd
$BACKUP_DIR
&&
find
-type
f
!
-name
backup.signature
-print0
| xargs
-0
sha256sum
|
LC_ALL
=
C
sort
-k
66
>
backup.signature
software/kvm/template/kvm-import.sh.jinja2
View file @
bf25f31d
...
@@ -4,7 +4,6 @@ set -e
...
@@ -4,7 +4,6 @@ set -e
VM_DIR
={{
directory[
'srv'
]
}}
VM_DIR
={{
directory[
'srv'
]
}}
BACKUP_DIR
={{
directory[
'backup'
]
}}
BACKUP_DIR
={{
directory[
'backup'
]
}}
VM_FILE
=
virtual.qcow2
LC_ALL
=
C
LC_ALL
=
C
export
LC_ALL
export
LC_ALL
umask
077
umask
077
...
@@ -18,11 +17,19 @@ write_backup_proof () {
...
@@ -18,11 +17,19 @@ write_backup_proof () {
# For now we just make the diff before
# For now we just make the diff before
write_backup_proof
write_backup_proof
if
[
-f
"
$BACKUP_DIR
/
${
VM_FILE
}
.gz"
]
;
then
tmpfile
=
$(
mktemp
--tmpdir
={{
directory[
'tmp'
]
}}
)
{{
gzip_binary
}}
-t
"
$BACKUP_DIR
/
${
VM_FILE
}
.gz"
||
exit
10
# use temporary space inside of the partition, as it can be quite big
{{
zcat_binary
}}
"
$BACKUP_DIR
/
${
VM_FILE
}
.gz"
>
$VM_DIR
/
$VM_FILE
# assure the temporary directory is cleaned up
trap
"rm -rf
$tmpdir
$tmpfile
"
EXIT TERM INT
if
[
-d
$BACKUP_DIR
/ide0-hd0
]
;
then
disk_type
=
"ide0-hd0"
elif
[
-d
$BACKUP_DIR
/virtio0
]
;
then
disk_type
=
virtio0
else
else
rm
$VM_DIR
/
$VM_FILE
echo
"Unsupported disk type"
cp
$BACKUP_DIR
/
$VM_FILE
$VM_DIR
/
$VM_FILE
exit
1
fi
fi
rm
-f
$VM_DIR
/virtual.qcow2
{{
qmprestore
}}
merge
--dir
$BACKUP_DIR
/
$disk_type
--target
$VM_DIR
/virtual.qcow2
software/kvm/test/test.py
View file @
bf25f31d
...
@@ -239,6 +239,12 @@ class KvmMixin:
...
@@ -239,6 +239,12 @@ class KvmMixin:
return os.path.join(
return os.path.join(
cls.slap._instance_root, cls.getPartitionId(instance_type), *paths)
cls.slap._instance_root, cls.getPartitionId(instance_type), *paths)
@classmethod
def getBackupPartitionPath(cls, *paths):
return cls.getPartitionPath(
'
kvm
-
export
', '
srv
', '
backup
', '
kvm
',
cls.disk_type_backup_mapping[cls.disk_type], *paths)
def getConnectionParameterDictJson(self):
def getConnectionParameterDictJson(self):
return json.loads(
return json.loads(
self.computer_partition.getConnectionParameterDict()['
_
'])
self.computer_partition.getConnectionParameterDict()['
_
'])
...
@@ -805,6 +811,151 @@ class CronMixin(object):
...
@@ -805,6 +811,151 @@ class CronMixin(object):
return job_list_output
return job_list_output
class TestInstanceResilientBackupMixin(CronMixin, KvmMixin):
__partition_reference__ = '
irb
'
instance_max_retry = 20
disk_type = '
virtio
'
disk_type_backup_mapping = {
'
virtio
': '
virtio0
',
'
ide
': '
ide0
-
hd0
',
}
@classmethod
def getInstanceParameterDict(cls):
parameter_dict = {}
if cls.disk_type != '
virtio
':
parameter_dict['
disk
-
type
'] = cls.disk_type
return parameter_dict
@classmethod
def getInstanceSoftwareType(cls):
return '
kvm
-
resilient
'
def setUp(self):
super().setUp()
importer_partition = glob.glob(os.path.join(
self.slap.instance_directory, '
*
', '
template
-
kvm
-
import
.
cfg
'))
self.assertEqual(1, len(importer_partition))
self.importer_partition = os.path.dirname(importer_partition[0])
def call_exporter(self):
result = self.executeCronDMockJob('
kvm
-
export
', '
backup
')
self.assertEqual(len(result), 1)
self.assertEqual(
0,
result[0].returncode,
result[0].stdout.decode('
utf
-
8
'))
return result[0].stdout.decode('
utf
-
8
')
@skipUnlessKvm
class TestInstanceResilientBackupImporter(
TestInstanceResilientBackupMixin, KVMTestCase):
def test(self):
equeue_file = os.path.join(
self.importer_partition, '
var
', '
log
', '
equeue
.
log
')
destination_qcow2 = os.path.join(
self.importer_partition, '
srv
', '
virtual
.
qcow2
')
destination_backup = os.path.join(
self.importer_partition, '
srv
', '
backup
', '
kvm
',
self.disk_type_backup_mapping[self.disk_type])
# sanity check - no export/import happened yet
self.assertFalse(os.path.exists(self.getBackupPartitionPath()))
self.call_exporter()
def awaitBackup(equeue_file):
for f in range(30):
with open(equeue_file, 'r') as fh:
equeue_log = fh.read()
if '
finished
successfully
' in equeue_log:
break
time.sleep(1)
else:
self.fail('
Backup
not
finished
:
%
s
' % (equeue_log))
return equeue_log
equeue_log = awaitBackup(equeue_file)
self.assertNotIn('
qemu
-
img
rebase
', equeue_log)
self.assertEqual(
os.listdir(self.getBackupPartitionPath()),
os.listdir(destination_backup)
)
self.assertTrue(os.path.exists(destination_qcow2))
# clean up equeue file for precise assertion
with open(equeue_file, '
w
') as fh:
fh.write('')
# drop backup destination to assert its recreation
os.unlink(destination_qcow2)
self.call_exporter()
equeue_log = awaitBackup(equeue_file)
self.assertIn('
qemu
-
img
rebase
', equeue_log)
self.assertEqual(
os.listdir(self.getBackupPartitionPath()),
os.listdir(destination_backup)
)
self.assertTrue(os.path.exists(destination_qcow2))
# takeover
connection_parameter = self.computer_partition.getConnectionParameterDict()
takeover_result = requests.post(
connection_parameter['
takeover
-
kvm
-
1
-
url
'],
data={
'
password
': connection_parameter['
takeover
-
kvm
-
1
-
password
']})
self.assertEqual(httplib.OK, takeover_result.status_code)
self.assertTrue(takeover_result.text.startswith('
Success
.
'))
# the real assertions comes from re-stabilizing the instance tree
self.slap.waitForInstance(max_retry=10)
# check that all stabilizes after backup after takeover
self.call_exporter()
self.slap.waitForInstance(max_retry=10)
@skipUnlessKvm
class TestInstanceResilientBackupImporterIde(
TestInstanceResilientBackupImporter):
disk_type = '
ide
'
@skipUnlessKvm
class TestInstanceResilientBackupExporter(
TestInstanceResilientBackupMixin, KVMTestCase):
def test(self):
status_text = self.call_exporter()
self.assertEqual(
len(glob.glob(self.getBackupPartitionPath('
FULL
-*
.
qcow2
'))),
1)
self.assertEqual(
len(glob.glob(self.getBackupPartitionPath('
INC
-*
.
qcow2
'))),
0)
self.assertNotIn(
'
Recovered
from
partial
backup
by
removing
partial
',
status_text
)
# cover .partial file in the backup directory with fallback to full
current_backup = glob.glob(self.getBackupPartitionPath('
FULL
-*
'))[0]
with open(current_backup + '
.
partial
', '
w
') as fh:
fh.write('')
status_text = self.call_exporter()
self.assertEqual(
len(glob.glob(self.getBackupPartitionPath('
FULL
-*
.
qcow2
'))),
1)
self.assertEqual(
len(glob.glob(self.getBackupPartitionPath('
INC
-*
.
qcow2
'))),
1)
self.assertIn(
'
Recovered
from
partial
backup
by
removing
partial
',
status_text
)
self.assertTrue(os.path.exists(os.path.join(
self.getPartitionPath(
'
kvm
-
export
', '
etc
', '
plugin
', '
check
-
backup
-
directory
.
py
'))))
@skipUnlessKvm
class TestInstanceResilientBackupExporterIde(
TestInstanceResilientBackupExporter):
disk_type = '
ide
'
@skipUnlessKvm
@skipUnlessKvm
class TestInstanceResilient(KVMTestCase, KvmMixin):
class TestInstanceResilient(KVMTestCase, KvmMixin):
__partition_reference__ = '
ir
'
__partition_reference__ = '
ir
'
...
@@ -823,25 +974,6 @@ class TestInstanceResilient(KVMTestCase, KvmMixin):
...
@@ -823,25 +974,6 @@ class TestInstanceResilient(KVMTestCase, KvmMixin):
cls.kvm0_ipv6 = cls.getPartitionIPv6(cls.kvm_instance_partition_reference)
cls.kvm0_ipv6 = cls.getPartitionIPv6(cls.kvm_instance_partition_reference)
cls.kvm1_ipv6 = cls.getPartitionIPv6(cls.getPartitionId('
kvm1
'))
cls.kvm1_ipv6 = cls.getPartitionIPv6(cls.getPartitionId('
kvm1
'))
def test_kvm_exporter(self):
exporter_partition = os.path.join(
self.slap.instance_directory,
self.__partition_reference__ + '
2
')
backup_path = os.path.join(
exporter_partition, '
srv
', '
backup
', '
kvm
', '
virtual
.
qcow2
.
gz
')
exporter = os.path.join(exporter_partition, '
bin
', '
exporter
')
if os.path.exists(backup_path):
os.unlink(backup_path)
def call_exporter():
try:
return (0, subprocess.check_output(
[exporter], stderr=subprocess.STDOUT).decode('
utf
-
8
'))
except subprocess.CalledProcessError as e:
return (e.returncode, e.output.decode('
utf
-
8
'))
status_code, status_text = call_exporter()
self.assertEqual(0, status_code, status_text)
def test(self):
def test(self):
connection_parameter_dict = self
\
connection_parameter_dict = self
\
.computer_partition.getConnectionParameterDict()
.computer_partition.getConnectionParameterDict()
...
@@ -2380,10 +2512,11 @@ class ExternalDiskModernMixin(object):
...
@@ -2380,10 +2512,11 @@ class ExternalDiskModernMixin(object):
# find qemu_img from the tested SR via it's partition parameter, as
# find qemu_img from the tested SR via it's partition parameter, as
# otherwise qemu-kvm would be dependency of test suite
# otherwise qemu-kvm would be dependency of test suite
with
open
(
with
open
(
os
.
path
.
join
(
self
.
computer_partition_root_path
,
'buildout.cfg'
))
as
fh
:
glob
.
glob
(
os
.
path
.
join
(
self
.
slap
.
_instance_root
,
'*'
,
'bin'
,
'kvm_raw'
))[
0
])
as
fh
:
self
.
qemu_img
=
[
self
.
qemu_img
=
[
q
for
q
in
fh
.
readlines
()
q
for
q
in
fh
.
readlines
()
if
'
raw qemu_img_executable_location'
in
q
][
0
].
split
()[
-
1
]
if
'
qemu_img_path = '
in
q
][
0
].
split
()[
-
1
].
replace
(
"'"
,
""
)
self
.
first_disk
=
os
.
path
.
join
(
self
.
working_directory
,
'first_disk'
)
self
.
first_disk
=
os
.
path
.
join
(
self
.
working_directory
,
'first_disk'
)
subprocess
.
check_call
([
subprocess
.
check_call
([
self
.
qemu_img
,
"create"
,
"-f"
,
"qcow"
,
self
.
first_disk
,
"1M"
])
self
.
qemu_img
,
"create"
,
"-f"
,
"qcow"
,
self
.
first_disk
,
"1M"
])
...
@@ -2530,10 +2663,11 @@ class TestExternalDiskModernIndexRequired(KVMTestCase, ExternalDiskMixin):
...
@@ -2530,10 +2663,11 @@ class TestExternalDiskModernIndexRequired(KVMTestCase, ExternalDiskMixin):
# find qemu_img from the tested SR via it's partition parameter, as
# find qemu_img from the tested SR via it's partition parameter, as
# otherwise qemu-kvm would be dependency of test suite
# otherwise qemu-kvm would be dependency of test suite
with
open
(
with
open
(
os
.
path
.
join
(
self
.
computer_partition_root_path
,
'buildout.cfg'
))
as
fh
:
glob
.
glob
(
os
.
path
.
join
(
self
.
slap
.
_instance_root
,
'*'
,
'bin'
,
'kvm_raw'
))[
0
])
as
fh
:
qemu_img
=
[
qemu_img
=
[
q
for
q
in
fh
.
readlines
()
q
for
q
in
fh
.
readlines
()
if
'
raw qemu_img_executable_location'
in
q
][
0
].
split
()[
-
1
]
if
'
qemu_img_path = '
in
q
][
0
].
split
()[
-
1
].
replace
(
"'"
,
""
)
self
.
first_disk
=
os
.
path
.
join
(
self
.
working_directory
,
'first_disk'
)
self
.
first_disk
=
os
.
path
.
join
(
self
.
working_directory
,
'first_disk'
)
subprocess
.
check_call
([
subprocess
.
check_call
([
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment