|  | #!/usr/bin/env python3 | 
|  | # group: rw migration quick | 
|  | # | 
|  | # Copyright (C) 2017 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: Stefan Hajnoczi <stefanha@redhat.com> | 
|  | # | 
|  | # Non-shared storage migration test using NBD server and drive-mirror | 
|  |  | 
|  | import iotests | 
|  |  | 
|  | iotests.script_initialize(supported_fmts=['qcow2', 'qed', 'raw'], | 
|  | supported_platforms=['linux']) | 
|  |  | 
|  | with iotests.FilePath('source.img') as source_img_path, \ | 
|  | iotests.FilePath('dest.img') as dest_img_path, \ | 
|  | iotests.FilePath('migration.sock', 'nbd.sock', base_dir=iotests.sock_dir) \ | 
|  | as (migration_sock_path, nbd_sock_path), \ | 
|  | iotests.VM('source') as source_vm, \ | 
|  | iotests.VM('dest') as dest_vm: | 
|  |  | 
|  | img_size = '1G' | 
|  | iotests.qemu_img_create('-f', iotests.imgfmt, source_img_path, img_size) | 
|  | iotests.qemu_img_create('-f', iotests.imgfmt, dest_img_path, img_size) | 
|  |  | 
|  | iotests.log('Launching VMs...') | 
|  | (source_vm.add_drive(source_img_path) | 
|  | .launch()) | 
|  | (dest_vm.add_drive(dest_img_path) | 
|  | .add_incoming('unix:{0}'.format(migration_sock_path)) | 
|  | .launch()) | 
|  |  | 
|  | source_vm.qmp_log('block-dirty-bitmap-add', node='drive0', name='bitmap0') | 
|  |  | 
|  | iotests.log('Launching NBD server on destination...') | 
|  | iotests.log(dest_vm.qmp('nbd-server-start', addr={'type': 'unix', 'data': {'path': nbd_sock_path}})) | 
|  | iotests.log(dest_vm.qmp('nbd-server-add', device='drive0', writable=True)) | 
|  |  | 
|  | iotests.log('Starting `drive-mirror` on source...') | 
|  | iotests.log(source_vm.qmp( | 
|  | 'drive-mirror', | 
|  | device='drive0', | 
|  | target='nbd+unix:///drive0?socket={0}'.format(nbd_sock_path), | 
|  | sync='full', | 
|  | format='raw', # always raw, the server handles the format | 
|  | mode='existing', | 
|  | job_id='mirror-job0')) | 
|  |  | 
|  | iotests.log('Waiting for `drive-mirror` to complete...') | 
|  | iotests.log(source_vm.event_wait('BLOCK_JOB_READY'), | 
|  | filters=[iotests.filter_qmp_event]) | 
|  |  | 
|  | iotests.log('Starting migration...') | 
|  | capabilities = [{'capability': 'events', 'state': True}, | 
|  | {'capability': 'dirty-bitmaps', 'state': True}] | 
|  | source_vm.qmp('migrate-set-capabilities', capabilities=capabilities) | 
|  | dest_vm.qmp('migrate-set-capabilities', capabilities=capabilities) | 
|  | iotests.log(source_vm.qmp('migrate', uri='unix:{0}'.format(migration_sock_path))) | 
|  |  | 
|  | source_vm.qmp_log('migrate-start-postcopy') | 
|  |  | 
|  | while True: | 
|  | event1 = source_vm.event_wait('MIGRATION') | 
|  | if event1['data']['status'] == 'postcopy-active': | 
|  | # This event is racy, it depends do we really do postcopy or bitmap | 
|  | # was migrated during downtime (and no data to migrate in postcopy | 
|  | # phase). So, don't log it. | 
|  | continue | 
|  | iotests.log(event1, filters=[iotests.filter_qmp_event]) | 
|  | if event1['data']['status'] in ('completed', 'failed'): | 
|  | iotests.log('Gracefully ending the `drive-mirror` job on source...') | 
|  | iotests.log(source_vm.qmp('block-job-cancel', device='mirror-job0')) | 
|  | break | 
|  |  | 
|  | while True: | 
|  | event2 = source_vm.event_wait('BLOCK_JOB_COMPLETED') | 
|  | iotests.log(event2, filters=[iotests.filter_qmp_event]) | 
|  | if event2['event'] == 'BLOCK_JOB_COMPLETED': | 
|  | iotests.log('Stopping the NBD server on destination...') | 
|  | iotests.log(dest_vm.qmp('nbd-server-stop')) | 
|  | break | 
|  |  | 
|  | iotests.log('Wait for migration completion on target...') | 
|  | migr_events = (('MIGRATION', {'data': {'status': 'completed'}}), | 
|  | ('MIGRATION', {'data': {'status': 'failed'}})) | 
|  | event = dest_vm.events_wait(migr_events) | 
|  | iotests.log(event, filters=[iotests.filter_qmp_event]) | 
|  |  | 
|  | iotests.log('Check bitmaps on source:') | 
|  | iotests.log(source_vm.qmp('query-block')['return'][0]['inserted']['dirty-bitmaps']) | 
|  |  | 
|  | iotests.log('Check bitmaps on target:') | 
|  | iotests.log(dest_vm.qmp('query-block')['return'][0]['inserted']['dirty-bitmaps']) |