| #!/usr/bin/env python |
| # |
| # This test covers what happens when a mirror block job is cancelled |
| # in various phases of its existence. |
| # |
| # Note that this test only checks the emitted events (i.e. |
| # BLOCK_JOB_COMPLETED vs. BLOCK_JOB_CANCELLED), it does not compare |
| # whether the target is in sync with the source when the |
| # BLOCK_JOB_COMPLETED event occurs. This is covered by other tests |
| # (such as 041). |
| # |
| # Copyright (C) 2018 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/>. |
| # |
| # Creator/Owner: Max Reitz <mreitz@redhat.com> |
| |
| import iotests |
| from iotests import log, qemu_img, qemu_io_silent |
| |
| iotests.verify_image_format(supported_fmts=['qcow2', 'raw']) |
| |
| |
| # Launches the VM, adds two null-co nodes (source and target), and |
| # starts a blockdev-mirror job on them. |
| # |
| # Either both or none of speed and buf_size must be given. |
| |
| def start_mirror(vm, speed=None, buf_size=None): |
| vm.launch() |
| |
| ret = vm.qmp('blockdev-add', |
| node_name='source', |
| driver='null-co', |
| size=1048576) |
| assert ret['return'] == {} |
| |
| ret = vm.qmp('blockdev-add', |
| node_name='target', |
| driver='null-co', |
| size=1048576) |
| assert ret['return'] == {} |
| |
| if speed is not None: |
| ret = vm.qmp('blockdev-mirror', |
| job_id='mirror', |
| device='source', |
| target='target', |
| sync='full', |
| speed=speed, |
| buf_size=buf_size) |
| else: |
| ret = vm.qmp('blockdev-mirror', |
| job_id='mirror', |
| device='source', |
| target='target', |
| sync='full') |
| |
| assert ret['return'] == {} |
| |
| |
| log('') |
| log('=== Cancel mirror job before convergence ===') |
| log('') |
| |
| log('--- force=false ---') |
| log('') |
| |
| with iotests.VM() as vm: |
| # Low speed so it does not converge |
| start_mirror(vm, 65536, 65536) |
| |
| log('Cancelling job') |
| log(vm.qmp('block-job-cancel', device='mirror', force=False)) |
| |
| log(vm.event_wait('BLOCK_JOB_CANCELLED'), |
| filters=[iotests.filter_qmp_event]) |
| |
| log('') |
| log('--- force=true ---') |
| log('') |
| |
| with iotests.VM() as vm: |
| # Low speed so it does not converge |
| start_mirror(vm, 65536, 65536) |
| |
| log('Cancelling job') |
| log(vm.qmp('block-job-cancel', device='mirror', force=True)) |
| |
| log(vm.event_wait('BLOCK_JOB_CANCELLED'), |
| filters=[iotests.filter_qmp_event]) |
| |
| |
| log('') |
| log('=== Cancel mirror job after convergence ===') |
| log('') |
| |
| log('--- force=false ---') |
| log('') |
| |
| with iotests.VM() as vm: |
| start_mirror(vm) |
| |
| log(vm.event_wait('BLOCK_JOB_READY'), |
| filters=[iotests.filter_qmp_event]) |
| |
| log('Cancelling job') |
| log(vm.qmp('block-job-cancel', device='mirror', force=False)) |
| |
| log(vm.event_wait('BLOCK_JOB_COMPLETED'), |
| filters=[iotests.filter_qmp_event]) |
| |
| log('') |
| log('--- force=true ---') |
| log('') |
| |
| with iotests.VM() as vm: |
| start_mirror(vm) |
| |
| log(vm.event_wait('BLOCK_JOB_READY'), |
| filters=[iotests.filter_qmp_event]) |
| |
| log('Cancelling job') |
| log(vm.qmp('block-job-cancel', device='mirror', force=True)) |
| |
| log(vm.event_wait('BLOCK_JOB_CANCELLED'), |
| filters=[iotests.filter_qmp_event]) |
| |
| log('') |
| log('=== Cancel mirror job from throttled node by quitting ===') |
| log('') |
| |
| with iotests.VM() as vm, \ |
| iotests.FilePath('src.img') as src_img_path: |
| |
| assert qemu_img('create', '-f', iotests.imgfmt, src_img_path, '64M') == 0 |
| assert qemu_io_silent('-f', iotests.imgfmt, src_img_path, |
| '-c', 'write -P 42 0M 64M') == 0 |
| |
| vm.launch() |
| |
| ret = vm.qmp('object-add', qom_type='throttle-group', id='tg', |
| props={'x-bps-read': 4096}) |
| assert ret['return'] == {} |
| |
| ret = vm.qmp('blockdev-add', |
| node_name='source', |
| driver=iotests.imgfmt, |
| file={ |
| 'driver': 'file', |
| 'filename': src_img_path |
| }) |
| assert ret['return'] == {} |
| |
| ret = vm.qmp('blockdev-add', |
| node_name='throttled-source', |
| driver='throttle', |
| throttle_group='tg', |
| file='source') |
| assert ret['return'] == {} |
| |
| ret = vm.qmp('blockdev-add', |
| node_name='target', |
| driver='null-co', |
| size=(64 * 1048576)) |
| assert ret['return'] == {} |
| |
| ret = vm.qmp('blockdev-mirror', |
| job_id='mirror', |
| device='throttled-source', |
| target='target', |
| sync='full') |
| assert ret['return'] == {} |
| |
| log(vm.qmp('quit')) |
| |
| with iotests.Timeout(5, 'Timeout waiting for VM to quit'): |
| vm.shutdown(has_quit=True) |