1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
|
#!/usr/bin/env python3
#
# Test incremental/backup across iothread contexts
#
# Copyright (c) 2019 John Snow for Red Hat, Inc.
#
# This program 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 2 of the License, or
# (at your option) any later version.
#
# This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
#
# owner=jsnow@redhat.com
import os
import iotests
from iotests import log
iotests.verify_image_format(supported_fmts=['qcow2'])
size = 64 * 1024 * 1024
with iotests.FilePath('img0') as img0_path, \
iotests.FilePath('img1') as img1_path, \
iotests.FilePath('img0-full') as img0_full_path, \
iotests.FilePath('img1-full') as img1_full_path, \
iotests.FilePath('img0-incr') as img0_incr_path, \
iotests.FilePath('img1-incr') as img1_incr_path, \
iotests.VM() as vm:
def create_target(filepath, name, size):
basename = os.path.basename(filepath)
nodename = "file_{}".format(basename)
log(vm.command('blockdev-create', job_id='job1',
options={
'driver': 'file',
'filename': filepath,
'size': 0,
}))
vm.run_job('job1')
log(vm.command('blockdev-add', driver='file',
node_name=nodename, filename=filepath))
log(vm.command('blockdev-create', job_id='job2',
options={
'driver': iotests.imgfmt,
'file': nodename,
'size': size,
}))
vm.run_job('job2')
log(vm.command('blockdev-add', driver=iotests.imgfmt,
node_name=name,
file=nodename))
log('--- Preparing images & VM ---\n')
vm.add_object('iothread,id=iothread0')
vm.add_object('iothread,id=iothread1')
vm.add_device('virtio-scsi-pci,id=scsi0,iothread=iothread0')
vm.add_device('virtio-scsi-pci,id=scsi1,iothread=iothread1')
iotests.qemu_img_create('-f', iotests.imgfmt, img0_path, str(size))
iotests.qemu_img_create('-f', iotests.imgfmt, img1_path, str(size))
vm.add_drive(img0_path, interface='none')
vm.add_device('scsi-hd,id=device0,drive=drive0,bus=scsi0.0')
vm.add_drive(img1_path, interface='none')
vm.add_device('scsi-hd,id=device1,drive=drive1,bus=scsi1.0')
log('--- Starting VM ---\n')
vm.launch()
log('--- Create Targets & Full Backups ---\n')
create_target(img0_full_path, 'img0-full', size)
create_target(img1_full_path, 'img1-full', size)
ret = vm.qmp_log('transaction', indent=2, actions=[
{ 'type': 'block-dirty-bitmap-add',
'data': { 'node': 'drive0', 'name': 'bitmap0' }},
{ 'type': 'block-dirty-bitmap-add',
'data': { 'node': 'drive1', 'name': 'bitmap1' }},
{ 'type': 'blockdev-backup',
'data': { 'device': 'drive0',
'target': 'img0-full',
'sync': 'full',
'job-id': 'j0' }},
{ 'type': 'blockdev-backup',
'data': { 'device': 'drive1',
'target': 'img1-full',
'sync': 'full',
'job-id': 'j1' }}
])
if "error" in ret:
raise Exception(ret['error']['desc'])
vm.run_job('j0', auto_dismiss=True)
vm.run_job('j1', auto_dismiss=True)
log('\n--- Create Targets & Incremental Backups ---\n')
create_target(img0_incr_path, 'img0-incr', size)
create_target(img1_incr_path, 'img1-incr', size)
ret = vm.qmp_log('transaction', indent=2, actions=[
{ 'type': 'blockdev-backup',
'data': { 'device': 'drive0',
'target': 'img0-incr',
'sync': 'incremental',
'bitmap': 'bitmap0',
'job-id': 'j2' }},
{ 'type': 'blockdev-backup',
'data': { 'device': 'drive1',
'target': 'img1-incr',
'sync': 'incremental',
'bitmap': 'bitmap1',
'job-id': 'j3' }}
])
if "error" in ret:
raise Exception(ret['error']['desc'])
vm.run_job('j2', auto_dismiss=True)
vm.run_job('j3', auto_dismiss=True)
log('\n--- Done ---')
vm.shutdown()
|