ZhiQiang Fan | 39f9722 | 2013-09-20 04:49:44 +0800 | [diff] [blame] | 1 | # Copyright 2012 OpenStack Foundation |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 2 | # All Rights Reserved. |
| 3 | # |
| 4 | # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| 5 | # not use this file except in compliance with the License. You may obtain |
| 6 | # a copy of the License at |
| 7 | # |
| 8 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | # |
| 10 | # Unless required by applicable law or agreed to in writing, software |
| 11 | # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| 12 | # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| 13 | # License for the specific language governing permissions and limitations |
| 14 | # under the License. |
| 15 | |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 16 | import random |
| 17 | import string |
| 18 | |
ivan-zhu | 1feeb38 | 2013-01-24 10:14:39 +0800 | [diff] [blame] | 19 | import testtools |
Matthew Treinish | a83a16e | 2012-12-07 13:44:02 -0500 | [diff] [blame] | 20 | |
Sean Dague | 1937d09 | 2013-05-17 16:36:38 -0400 | [diff] [blame] | 21 | from tempest.api.compute import base |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 22 | from tempest import config |
| 23 | from tempest import exceptions |
Chris Yeoh | 9465b0b | 2013-02-09 22:19:15 +1030 | [diff] [blame] | 24 | from tempest.test import attr |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 25 | |
Matthew Treinish | b0a78fc | 2014-01-29 16:49:12 +0000 | [diff] [blame^] | 26 | CONF = config.CONF |
| 27 | |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 28 | |
ivan-zhu | f2b0050 | 2013-10-18 10:06:52 +0800 | [diff] [blame] | 29 | class LiveBlockMigrationTestJSON(base.BaseV2ComputeAdminTest): |
Mate Lakat | 0ccf0eb | 2013-03-28 12:00:52 +0000 | [diff] [blame] | 30 | _host_key = 'OS-EXT-SRV-ATTR:host' |
Attila Fazekas | 19044d5 | 2013-02-16 07:35:06 +0100 | [diff] [blame] | 31 | _interface = 'json' |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 32 | |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 33 | @classmethod |
| 34 | def setUpClass(cls): |
ravikumar-venkatesan | 753262b | 2013-03-21 12:43:57 +0000 | [diff] [blame] | 35 | super(LiveBlockMigrationTestJSON, cls).setUpClass() |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 36 | |
Attila Fazekas | 8e99b99 | 2013-02-24 09:53:23 +0100 | [diff] [blame] | 37 | cls.admin_hosts_client = cls.os_adm.hosts_client |
| 38 | cls.admin_servers_client = cls.os_adm.servers_client |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 39 | |
| 40 | cls.created_server_ids = [] |
| 41 | |
| 42 | def _get_compute_hostnames(self): |
| 43 | _resp, body = self.admin_hosts_client.list_hosts() |
| 44 | return [ |
| 45 | host_record['host_name'] |
| 46 | for host_record in body |
| 47 | if host_record['service'] == 'compute' |
| 48 | ] |
| 49 | |
| 50 | def _get_server_details(self, server_id): |
| 51 | _resp, body = self.admin_servers_client.get_server(server_id) |
| 52 | return body |
| 53 | |
| 54 | def _get_host_for_server(self, server_id): |
Mate Lakat | 0ccf0eb | 2013-03-28 12:00:52 +0000 | [diff] [blame] | 55 | return self._get_server_details(server_id)[self._host_key] |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 56 | |
| 57 | def _migrate_server_to(self, server_id, dest_host): |
| 58 | _resp, body = self.admin_servers_client.live_migrate_server( |
Bob Ball | 0dca897 | 2013-04-09 16:23:31 +0100 | [diff] [blame] | 59 | server_id, dest_host, |
Matthew Treinish | b0a78fc | 2014-01-29 16:49:12 +0000 | [diff] [blame^] | 60 | CONF.compute_feature_enabled.block_migration_for_live_migration) |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 61 | return body |
| 62 | |
| 63 | def _get_host_other_than(self, host): |
| 64 | for target_host in self._get_compute_hostnames(): |
| 65 | if host != target_host: |
| 66 | return target_host |
| 67 | |
| 68 | def _get_non_existing_host_name(self): |
| 69 | random_name = ''.join( |
| 70 | random.choice(string.ascii_uppercase) for x in range(20)) |
| 71 | |
Jaroslav Henner | 3c3d179 | 2012-11-16 10:28:47 +0100 | [diff] [blame] | 72 | self.assertNotIn(random_name, self._get_compute_hostnames()) |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 73 | |
| 74 | return random_name |
| 75 | |
| 76 | def _get_server_status(self, server_id): |
| 77 | return self._get_server_details(server_id)['status'] |
| 78 | |
| 79 | def _get_an_active_server(self): |
| 80 | for server_id in self.created_server_ids: |
| 81 | if 'ACTIVE' == self._get_server_status(server_id): |
| 82 | return server_id |
| 83 | else: |
Ken'ichi Ohmichi | cfc052e | 2013-10-23 11:50:04 +0900 | [diff] [blame] | 84 | _, server = self.create_test_server(wait_until="ACTIVE") |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 85 | server_id = server['id'] |
| 86 | self.password = server['adminPass'] |
| 87 | self.password = 'password' |
| 88 | self.created_server_ids.append(server_id) |
| 89 | return server_id |
| 90 | |
Bob Ball | c078be9 | 2013-04-09 14:25:00 +0100 | [diff] [blame] | 91 | def _volume_clean_up(self, server_id, volume_id): |
| 92 | resp, body = self.volumes_client.get_volume(volume_id) |
| 93 | if body['status'] == 'in-use': |
| 94 | self.servers_client.detach_volume(server_id, volume_id) |
| 95 | self.volumes_client.wait_for_volume_status(volume_id, 'available') |
| 96 | self.volumes_client.delete_volume(volume_id) |
| 97 | |
Matthew Treinish | d5c9602 | 2013-10-17 21:51:23 +0000 | [diff] [blame] | 98 | @testtools.skipIf(not CONF.compute_feature_enabled.live_migration, |
Bob Ball | 0dca897 | 2013-04-09 16:23:31 +0100 | [diff] [blame] | 99 | 'Live migration not available') |
Giampaolo Lauria | e9c7702 | 2013-05-22 01:23:58 -0400 | [diff] [blame] | 100 | @attr(type='gate') |
Attila Fazekas | 8e99b99 | 2013-02-24 09:53:23 +0100 | [diff] [blame] | 101 | def test_live_block_migration(self): |
Sean Dague | 4dd2c0b | 2013-01-03 17:50:28 -0500 | [diff] [blame] | 102 | # Live block migrate an instance to another host |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 103 | if len(self._get_compute_hostnames()) < 2: |
ivan-zhu | 1feeb38 | 2013-01-24 10:14:39 +0800 | [diff] [blame] | 104 | raise self.skipTest( |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 105 | "Less than 2 compute nodes, skipping migration test.") |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 106 | server_id = self._get_an_active_server() |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 107 | actual_host = self._get_host_for_server(server_id) |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 108 | target_host = self._get_host_other_than(actual_host) |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 109 | self._migrate_server_to(server_id, target_host) |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 110 | self.servers_client.wait_for_server_status(server_id, 'ACTIVE') |
Chang Bo Guo | fc77e93 | 2013-09-16 17:38:26 -0700 | [diff] [blame] | 111 | self.assertEqual(target_host, self._get_host_for_server(server_id)) |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 112 | |
Matthew Treinish | d5c9602 | 2013-10-17 21:51:23 +0000 | [diff] [blame] | 113 | @testtools.skipIf(not CONF.compute_feature_enabled.live_migration, |
Bob Ball | 0dca897 | 2013-04-09 16:23:31 +0100 | [diff] [blame] | 114 | 'Live migration not available') |
Giampaolo Lauria | e9c7702 | 2013-05-22 01:23:58 -0400 | [diff] [blame] | 115 | @attr(type='gate') |
Attila Fazekas | 8e99b99 | 2013-02-24 09:53:23 +0100 | [diff] [blame] | 116 | def test_invalid_host_for_migration(self): |
Sean Dague | 4dd2c0b | 2013-01-03 17:50:28 -0500 | [diff] [blame] | 117 | # Migrating to an invalid host should not change the status |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 118 | server_id = self._get_an_active_server() |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 119 | target_host = self._get_non_existing_host_name() |
| 120 | |
Attila Fazekas | fa756cb | 2013-02-12 10:52:42 +0100 | [diff] [blame] | 121 | self.assertRaises(exceptions.BadRequest, self._migrate_server_to, |
| 122 | server_id, target_host) |
Chang Bo Guo | fc77e93 | 2013-09-16 17:38:26 -0700 | [diff] [blame] | 123 | self.assertEqual('ACTIVE', self._get_server_status(server_id)) |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 124 | |
Matthew Treinish | d5c9602 | 2013-10-17 21:51:23 +0000 | [diff] [blame] | 125 | @testtools.skipIf(not CONF.compute_feature_enabled.live_migration or not |
| 126 | CONF.compute_feature_enabled. |
| 127 | block_migration_for_live_migration, |
Bob Ball | c078be9 | 2013-04-09 14:25:00 +0100 | [diff] [blame] | 128 | 'Block Live migration not available') |
Matthew Treinish | d5c9602 | 2013-10-17 21:51:23 +0000 | [diff] [blame] | 129 | @testtools.skipIf(not CONF.compute_feature_enabled. |
| 130 | block_migrate_cinder_iscsi, |
Bob Ball | c078be9 | 2013-04-09 14:25:00 +0100 | [diff] [blame] | 131 | 'Block Live migration not configured for iSCSI') |
Giampaolo Lauria | e9c7702 | 2013-05-22 01:23:58 -0400 | [diff] [blame] | 132 | @attr(type='gate') |
Bob Ball | c078be9 | 2013-04-09 14:25:00 +0100 | [diff] [blame] | 133 | def test_iscsi_volume(self): |
| 134 | # Live block migrate an instance to another host |
| 135 | if len(self._get_compute_hostnames()) < 2: |
| 136 | raise self.skipTest( |
| 137 | "Less than 2 compute nodes, skipping migration test.") |
| 138 | server_id = self._get_an_active_server() |
| 139 | actual_host = self._get_host_for_server(server_id) |
| 140 | target_host = self._get_host_other_than(actual_host) |
| 141 | |
| 142 | resp, volume = self.volumes_client.create_volume(1, |
| 143 | display_name='test') |
| 144 | |
| 145 | self.volumes_client.wait_for_volume_status(volume['id'], |
| 146 | 'available') |
| 147 | self.addCleanup(self._volume_clean_up, server_id, volume['id']) |
| 148 | |
| 149 | # Attach the volume to the server |
| 150 | self.servers_client.attach_volume(server_id, volume['id'], |
| 151 | device='/dev/xvdb') |
| 152 | self.volumes_client.wait_for_volume_status(volume['id'], 'in-use') |
| 153 | |
| 154 | self._migrate_server_to(server_id, target_host) |
| 155 | self.servers_client.wait_for_server_status(server_id, 'ACTIVE') |
Chang Bo Guo | fc77e93 | 2013-09-16 17:38:26 -0700 | [diff] [blame] | 156 | self.assertEqual(target_host, self._get_host_for_server(server_id)) |
Bob Ball | c078be9 | 2013-04-09 14:25:00 +0100 | [diff] [blame] | 157 | |
Mate Lakat | 99ee914 | 2012-09-14 12:34:46 +0100 | [diff] [blame] | 158 | @classmethod |
| 159 | def tearDownClass(cls): |
| 160 | for server_id in cls.created_server_ids: |
| 161 | cls.servers_client.delete_server(server_id) |
| 162 | |
ravikumar-venkatesan | 753262b | 2013-03-21 12:43:57 +0000 | [diff] [blame] | 163 | super(LiveBlockMigrationTestJSON, cls).tearDownClass() |
| 164 | |
| 165 | |
| 166 | class LiveBlockMigrationTestXML(LiveBlockMigrationTestJSON): |
Mate Lakat | 0ccf0eb | 2013-03-28 12:00:52 +0000 | [diff] [blame] | 167 | _host_key = ( |
| 168 | '{http://docs.openstack.org/compute/ext/extended_status/api/v1.1}host') |
ravikumar-venkatesan | 753262b | 2013-03-21 12:43:57 +0000 | [diff] [blame] | 169 | _interface = 'xml' |