Currently compute hosts_client returns Response by removing top key from Response. For example- return service_client.ResponseBody(resp, body['host']) As service clients are in direction to move to Tempest-lib, all service clients should return Response without any truncation. One good example is Resource pagination links which are lost with current way of return value. Resource pagination links are present in parallel (not inside) to top key of Response. This patch makes compute hosts_client to return complete Response body. Implements: blueprint method-return-value-and-move-service-clients-to-lib Change-Id: Ife5bdbbcbdd856ecbce66e8b9c74c70dbb93f077
175 lines
7.2 KiB
Python
175 lines
7.2 KiB
Python
# Copyright 2012 OpenStack Foundation
|
|
# All Rights Reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
|
|
import testtools
|
|
|
|
from tempest.api.compute import base
|
|
from tempest.common import waiters
|
|
from tempest import config
|
|
from tempest import test
|
|
|
|
CONF = config.CONF
|
|
|
|
|
|
class LiveBlockMigrationTestJSON(base.BaseV2ComputeAdminTest):
|
|
_host_key = 'OS-EXT-SRV-ATTR:host'
|
|
|
|
@classmethod
|
|
def setup_clients(cls):
|
|
super(LiveBlockMigrationTestJSON, cls).setup_clients()
|
|
cls.admin_hosts_client = cls.os_adm.hosts_client
|
|
cls.admin_servers_client = cls.os_adm.servers_client
|
|
cls.admin_migration_client = cls.os_adm.migrations_client
|
|
|
|
@classmethod
|
|
def resource_setup(cls):
|
|
super(LiveBlockMigrationTestJSON, cls).resource_setup()
|
|
|
|
cls.created_server_ids = []
|
|
|
|
def _get_compute_hostnames(self):
|
|
body = self.admin_hosts_client.list_hosts()['hosts']
|
|
return [
|
|
host_record['host_name']
|
|
for host_record in body
|
|
if host_record['service'] == 'compute'
|
|
]
|
|
|
|
def _get_server_details(self, server_id):
|
|
body = self.admin_servers_client.show_server(server_id)
|
|
return body
|
|
|
|
def _get_host_for_server(self, server_id):
|
|
return self._get_server_details(server_id)[self._host_key]
|
|
|
|
def _migrate_server_to(self, server_id, dest_host):
|
|
bmflm = CONF.compute_feature_enabled.block_migration_for_live_migration
|
|
body = self.admin_servers_client.live_migrate_server(
|
|
server_id, host=dest_host, block_migration=bmflm,
|
|
disk_over_commit=False)
|
|
return body
|
|
|
|
def _get_host_other_than(self, host):
|
|
for target_host in self._get_compute_hostnames():
|
|
if host != target_host:
|
|
return target_host
|
|
|
|
def _get_server_status(self, server_id):
|
|
return self._get_server_details(server_id)['status']
|
|
|
|
def _get_an_active_server(self):
|
|
for server_id in self.created_server_ids:
|
|
if 'ACTIVE' == self._get_server_status(server_id):
|
|
return server_id
|
|
else:
|
|
server = self.create_test_server(wait_until="ACTIVE")
|
|
server_id = server['id']
|
|
self.created_server_ids.append(server_id)
|
|
return server_id
|
|
|
|
def _volume_clean_up(self, server_id, volume_id):
|
|
body = self.volumes_client.show_volume(volume_id)
|
|
if body['status'] == 'in-use':
|
|
self.servers_client.detach_volume(server_id, volume_id)
|
|
self.volumes_client.wait_for_volume_status(volume_id, 'available')
|
|
self.volumes_client.delete_volume(volume_id)
|
|
|
|
def _test_live_block_migration(self, state='ACTIVE'):
|
|
"""Tests live block migration between two hosts.
|
|
|
|
Requires CONF.compute_feature_enabled.live_migration to be True.
|
|
|
|
:param state: The vm_state the migrated server should be in before and
|
|
after the live migration. Supported values are 'ACTIVE'
|
|
and 'PAUSED'.
|
|
"""
|
|
# Live block migrate an instance to another host
|
|
if len(self._get_compute_hostnames()) < 2:
|
|
raise self.skipTest(
|
|
"Less than 2 compute nodes, skipping migration test.")
|
|
server_id = self._get_an_active_server()
|
|
actual_host = self._get_host_for_server(server_id)
|
|
target_host = self._get_host_other_than(actual_host)
|
|
|
|
if state == 'PAUSED':
|
|
self.admin_servers_client.pause_server(server_id)
|
|
waiters.wait_for_server_status(self.admin_servers_client,
|
|
server_id, state)
|
|
|
|
self._migrate_server_to(server_id, target_host)
|
|
waiters.wait_for_server_status(self.servers_client, server_id, state)
|
|
migration_list = self.admin_migration_client.list_migrations()
|
|
|
|
msg = ("Live Migration failed. Migrations list for Instance "
|
|
"%s: [" % server_id)
|
|
for live_migration in migration_list:
|
|
if (live_migration['instance_uuid'] == server_id):
|
|
msg += "\n%s" % live_migration
|
|
msg += "]"
|
|
self.assertEqual(target_host, self._get_host_for_server(server_id),
|
|
msg)
|
|
|
|
@test.idempotent_id('1dce86b8-eb04-4c03-a9d8-9c1dc3ee0c7b')
|
|
@testtools.skipUnless(CONF.compute_feature_enabled.live_migration,
|
|
'Live migration not available')
|
|
def test_live_block_migration(self):
|
|
self._test_live_block_migration()
|
|
|
|
@test.idempotent_id('1e107f21-61b2-4988-8f22-b196e938ab88')
|
|
@testtools.skipUnless(CONF.compute_feature_enabled.live_migration,
|
|
'Live migration not available')
|
|
@testtools.skipUnless(CONF.compute_feature_enabled.pause,
|
|
'Pause is not available.')
|
|
@testtools.skipUnless(CONF.compute_feature_enabled
|
|
.live_migrate_paused_instances,
|
|
'Live migration of paused instances is not '
|
|
'available.')
|
|
def test_live_block_migration_paused(self):
|
|
self._test_live_block_migration(state='PAUSED')
|
|
|
|
@test.idempotent_id('e19c0cc6-6720-4ed8-be83-b6603ed5c812')
|
|
@testtools.skipIf(not CONF.compute_feature_enabled.live_migration or not
|
|
CONF.compute_feature_enabled.
|
|
block_migration_for_live_migration,
|
|
'Block Live migration not available')
|
|
@testtools.skipIf(not CONF.compute_feature_enabled.
|
|
block_migrate_cinder_iscsi,
|
|
'Block Live migration not configured for iSCSI')
|
|
def test_iscsi_volume(self):
|
|
# Live block migrate an instance to another host
|
|
if len(self._get_compute_hostnames()) < 2:
|
|
raise self.skipTest(
|
|
"Less than 2 compute nodes, skipping migration test.")
|
|
server_id = self._get_an_active_server()
|
|
actual_host = self._get_host_for_server(server_id)
|
|
target_host = self._get_host_other_than(actual_host)
|
|
|
|
volume = self.volumes_client.create_volume(display_name='test')
|
|
|
|
self.volumes_client.wait_for_volume_status(volume['id'],
|
|
'available')
|
|
self.addCleanup(self._volume_clean_up, server_id, volume['id'])
|
|
|
|
# Attach the volume to the server
|
|
self.servers_client.attach_volume(server_id, volumeId=volume['id'],
|
|
device='/dev/xvdb')
|
|
self.volumes_client.wait_for_volume_status(volume['id'], 'in-use')
|
|
|
|
self._migrate_server_to(server_id, target_host)
|
|
waiters.wait_for_server_status(self.servers_client,
|
|
server_id, 'ACTIVE')
|
|
self.assertEqual(target_host, self._get_host_for_server(server_id))
|