3 # Test whether the backing BDSs are correct after completion of a
4 # mirror block job; in "existing" modes (drive-mirror with
5 # mode=existing and blockdev-mirror) the backing chain should not be
8 # Copyright (C) 2016 Red Hat, Inc.
10 # This program is free software; you can redistribute it and/or modify
11 # it under the terms of the GNU General Public License as published by
12 # the Free Software Foundation; either version 2 of the License, or
13 # (at your option) any later version.
15 # This program is distributed in the hope that it will be useful,
16 # but WITHOUT ANY WARRANTY; without even the implied warranty of
17 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 # GNU General Public License for more details.
20 # You should have received a copy of the GNU General Public License
21 # along with this program. If not, see <http://www.gnu.org/licenses/>.
26 from iotests import qemu_img
28 back0_img = os.path.join(iotests.test_dir, 'back0.' + iotests.imgfmt)
29 back1_img = os.path.join(iotests.test_dir, 'back1.' + iotests.imgfmt)
30 back2_img = os.path.join(iotests.test_dir, 'back2.' + iotests.imgfmt)
31 source_img = os.path.join(iotests.test_dir, 'source.' + iotests.imgfmt)
32 target_img = os.path.join(iotests.test_dir, 'target.' + iotests.imgfmt)
35 # Class variables for controlling its behavior:
37 # existing: If True, explicitly create the target image and blockdev-add it
38 # target_backing: If existing is True: Use this filename as the backing file
40 # (None: no backing file)
41 # target_blockdev_backing: If existing is True: Pass this dict as "backing"
42 # for the blockdev-add command
43 # (None: do not pass "backing")
44 # target_real_backing: If existing is True: The real filename of the backing
45 # image during runtime, only makes sense if
46 # target_blockdev_backing is not None
47 # (None: same as target_backing)
49 class BaseClass(iotests.QMPTestCase):
50 target_blockdev_backing = None
51 target_real_backing = None
54 qemu_img('create', '-f', iotests.imgfmt, back0_img, '1440K')
55 qemu_img('create', '-f', iotests.imgfmt, '-b', back0_img, back1_img)
56 qemu_img('create', '-f', iotests.imgfmt, '-b', back1_img, back2_img)
57 qemu_img('create', '-f', iotests.imgfmt, '-b', back2_img, source_img)
59 self.vm = iotests.VM()
60 # Add the BDS via blockdev-add so it stays around after the mirror block
61 # job has been completed
62 blockdev = {'node-name': 'source',
63 'driver': iotests.imgfmt,
64 'file': {'driver': 'file',
65 'filename': source_img}}
66 self.vm.add_blockdev(self.qmp_to_opts(blockdev))
67 self.vm.add_device('floppy,id=qdev0,drive=source')
70 self.assertIntactSourceBackingChain()
73 if self.target_backing:
74 qemu_img('create', '-f', iotests.imgfmt,
75 '-b', self.target_backing, target_img, '1440K')
77 qemu_img('create', '-f', iotests.imgfmt, target_img, '1440K')
79 if self.cmd == 'blockdev-mirror':
80 options = { 'node-name': 'target',
81 'driver': iotests.imgfmt,
82 'file': { 'driver': 'file',
83 'filename': target_img } }
84 if self.target_blockdev_backing:
85 options['backing'] = self.target_blockdev_backing
87 result = self.vm.qmp('blockdev-add', **options)
88 self.assert_qmp(result, 'return', {})
101 def findBlockNode(self, node_name, qdev=None):
103 result = self.vm.qmp('query-block')
104 for device in result['return']:
105 if device['qdev'] == qdev:
107 self.assert_qmp(device, 'inserted/node-name', node_name)
108 return device['inserted']
110 result = self.vm.qmp('query-named-block-nodes')
111 for node in result['return']:
112 if node['node-name'] == node_name:
115 self.fail('Cannot find node %s/%s' % (qdev, node_name))
117 def assertIntactSourceBackingChain(self):
118 node = self.findBlockNode('source')
120 self.assert_qmp(node, 'image' + '/backing-image' * 0 + '/filename',
122 self.assert_qmp(node, 'image' + '/backing-image' * 1 + '/filename',
124 self.assert_qmp(node, 'image' + '/backing-image' * 2 + '/filename',
126 self.assert_qmp(node, 'image' + '/backing-image' * 3 + '/filename',
128 self.assert_qmp_absent(node, 'image' + '/backing-image' * 4)
130 def assertCorrectBackingImage(self, node, default_image):
132 if self.target_real_backing:
133 image = self.target_real_backing
135 image = self.target_backing
137 image = default_image
140 self.assert_qmp(node, 'image/backing-image/filename', image)
142 self.assert_qmp_absent(node, 'image/backing-image')
145 # Class variables for controlling its behavior:
147 # cmd: Mirroring command to execute, either drive-mirror or blockdev-mirror
149 class MirrorBaseClass(BaseClass):
150 def runMirror(self, sync):
151 if self.cmd == 'blockdev-mirror':
152 result = self.vm.qmp(self.cmd, job_id='mirror-job', device='source',
153 sync=sync, target='target')
158 mode = 'absolute-paths'
159 result = self.vm.qmp(self.cmd, job_id='mirror-job', device='source',
160 sync=sync, target=target_img,
161 format=iotests.imgfmt, mode=mode,
164 self.assert_qmp(result, 'return', {})
166 self.vm.event_wait('BLOCK_JOB_READY')
168 result = self.vm.qmp('block-job-complete', device='mirror-job')
169 self.assert_qmp(result, 'return', {})
171 self.vm.event_wait('BLOCK_JOB_COMPLETED')
174 self.runMirror('full')
176 node = self.findBlockNode('target', 'qdev0')
177 self.assertCorrectBackingImage(node, None)
178 self.assertIntactSourceBackingChain()
181 self.runMirror('top')
183 node = self.findBlockNode('target', 'qdev0')
184 self.assertCorrectBackingImage(node, back2_img)
185 self.assertIntactSourceBackingChain()
188 self.runMirror('none')
190 node = self.findBlockNode('target', 'qdev0')
191 self.assertCorrectBackingImage(node, source_img)
192 self.assertIntactSourceBackingChain()
195 class TestDriveMirrorAbsolutePaths(MirrorBaseClass):
199 class TestDriveMirrorExistingNoBacking(MirrorBaseClass):
202 target_backing = None
204 class TestDriveMirrorExistingBacking(MirrorBaseClass):
207 target_backing = 'null-co://'
209 class TestBlockdevMirrorNoBacking(MirrorBaseClass):
210 cmd = 'blockdev-mirror'
212 target_backing = None
214 class TestBlockdevMirrorBacking(MirrorBaseClass):
215 cmd = 'blockdev-mirror'
217 target_backing = 'null-co://'
219 class TestBlockdevMirrorForcedBacking(MirrorBaseClass):
220 cmd = 'blockdev-mirror'
222 target_backing = None
223 target_blockdev_backing = { 'driver': 'null-co' }
224 target_real_backing = 'null-co://'
227 class TestCommit(BaseClass):
230 def testCommit(self):
231 result = self.vm.qmp('block-commit', job_id='commit-job',
232 device='source', base=back1_img)
233 self.assert_qmp(result, 'return', {})
235 self.vm.event_wait('BLOCK_JOB_READY')
237 result = self.vm.qmp('block-job-complete', device='commit-job')
238 self.assert_qmp(result, 'return', {})
240 self.vm.event_wait('BLOCK_JOB_COMPLETED')
242 node = self.findBlockNode(None, 'qdev0')
243 self.assert_qmp(node, 'image' + '/backing-image' * 0 + '/filename',
245 self.assert_qmp(node, 'image' + '/backing-image' * 1 + '/filename',
247 self.assert_qmp_absent(node, 'image' + '/backing-image' * 2 +
250 self.assertIntactSourceBackingChain()
254 MirrorBaseClass = None
256 if __name__ == '__main__':
257 iotests.main(supported_fmts=['qcow2'])