blob: ddb035dc1f41a8926bc2158bb5cf061fcc4b445f [file] [log] [blame]
ZhiQiang Fan39f97222013-09-20 04:49:44 +08001# Copyright 2012 OpenStack Foundation
Jay Pipes13b479b2012-06-11 14:52:27 -04002# 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
David Kranzcf0040c2012-06-26 09:46:56 -040016import time
Jay Pipesf38eaac2012-06-21 13:37:35 -040017
Doug Hellmann583ce2c2015-03-11 14:55:46 +000018from oslo_log import log as logging
Matthew Treinish01472ff2015-02-20 17:26:52 -050019
Ghanshyam05049dd2016-02-12 17:44:48 +090020from tempest.api.compute import api_microversion_fixture
Joseph Lanouxb3e1f872015-01-30 11:13:07 +000021from tempest.common import compute
Ken'ichi Ohmichi8b9c7802015-07-08 05:57:37 +000022from tempest.common import waiters
Matthew Treinishb0a78fc2014-01-29 16:49:12 +000023from tempest import config
Sean Dague20e98612016-01-06 14:33:28 -050024from tempest import exceptions
Sergey Nikitin8654e5b2017-06-04 22:09:56 +040025from tempest.lib.common import api_version_request
Ghanshyam1f47cf92016-02-25 04:57:18 +090026from tempest.lib.common import api_version_utils
Ken'ichi Ohmichi757833a2017-03-10 10:30:30 -080027from tempest.lib.common.utils import data_utils
Jordan Pittier9e227c52016-02-09 14:35:18 +010028from tempest.lib.common.utils import test_utils
Andrea Frittoli (andreaf)db9672e2016-02-23 14:07:24 -050029from tempest.lib import exceptions as lib_exc
Attila Fazekasdc216422013-01-29 15:12:14 +010030import tempest.test
Jay Pipesf38eaac2012-06-21 13:37:35 -040031
Matthew Treinishb0a78fc2014-01-29 16:49:12 +000032CONF = config.CONF
Tiago Melloeda03b52012-08-22 23:47:29 -030033
Jay Pipesf38eaac2012-06-21 13:37:35 -040034LOG = logging.getLogger(__name__)
Daryl Walleckc7251962012-03-12 17:26:54 -050035
36
Ken'ichi Ohmichi4d237e72015-11-05 06:32:33 +000037class BaseV2ComputeTest(api_version_utils.BaseMicroversionTest,
38 tempest.test.BaseTestCase):
Sean Daguef237ccb2013-01-04 15:19:14 -050039 """Base test case class for all Compute API tests."""
Daryl Walleckc7251962012-03-12 17:26:54 -050040
Attila Fazekas430dae32013-10-17 15:19:32 +020041 force_tenant_isolation = False
Chris Yeoh8a79b9d2013-01-18 19:32:47 +103042
Andrea Frittolib21de6c2015-02-06 20:12:38 +000043 # TODO(andreaf) We should care also for the alt_manager here
44 # but only once client lazy load in the manager is done
45 credentials = ['primary']
46
Jay Pipesf38eaac2012-06-21 13:37:35 -040047 @classmethod
Emily Hugenbruche7991d92014-12-12 16:53:36 +000048 def skip_checks(cls):
Ken'ichi Ohmichi02a8ccd2015-11-05 06:05:29 +000049 super(BaseV2ComputeTest, cls).skip_checks()
Matthew Treinishf8ff3582015-08-25 12:41:56 -040050 if not CONF.service_available.nova:
51 raise cls.skipException("Nova is not available")
ghanshyam29591532016-03-11 17:12:43 +090052 cfg_min_version = CONF.compute.min_microversion
53 cfg_max_version = CONF.compute.max_microversion
Ken'ichi Ohmichi4d237e72015-11-05 06:32:33 +000054 api_version_utils.check_skip_with_microversion(cls.min_microversion,
55 cls.max_microversion,
56 cfg_min_version,
57 cfg_max_version)
Jay Pipesf38eaac2012-06-21 13:37:35 -040058
Emily Hugenbruche7991d92014-12-12 16:53:36 +000059 @classmethod
60 def setup_credentials(cls):
61 cls.set_network_resources()
Ken'ichi Ohmichi02a8ccd2015-11-05 06:05:29 +000062 super(BaseV2ComputeTest, cls).setup_credentials()
Daryl Walleckc7251962012-03-12 17:26:54 -050063
Emily Hugenbruche7991d92014-12-12 16:53:36 +000064 @classmethod
65 def setup_clients(cls):
Ken'ichi Ohmichi02a8ccd2015-11-05 06:05:29 +000066 super(BaseV2ComputeTest, cls).setup_clients()
Jordan Pittier8160d312017-04-18 11:52:23 +020067 cls.servers_client = cls.os_primary.servers_client
68 cls.server_groups_client = cls.os_primary.server_groups_client
69 cls.flavors_client = cls.os_primary.flavors_client
70 cls.compute_images_client = cls.os_primary.compute_images_client
71 cls.extensions_client = cls.os_primary.extensions_client
72 cls.floating_ip_pools_client = cls.os_primary.floating_ip_pools_client
73 cls.floating_ips_client = cls.os_primary.compute_floating_ips_client
74 cls.keypairs_client = cls.os_primary.keypairs_client
John Warren5cdbf422016-01-05 12:42:43 -050075 cls.security_group_rules_client = (
Jordan Pittier8160d312017-04-18 11:52:23 +020076 cls.os_primary.compute_security_group_rules_client)
77 cls.security_groups_client =\
78 cls.os_primary.compute_security_groups_client
79 cls.quotas_client = cls.os_primary.quotas_client
80 cls.compute_networks_client = cls.os_primary.compute_networks_client
81 cls.limits_client = cls.os_primary.limits_client
82 cls.volumes_extensions_client =\
83 cls.os_primary.volumes_extensions_client
84 cls.snapshots_extensions_client =\
85 cls.os_primary.snapshots_extensions_client
86 cls.interfaces_client = cls.os_primary.interfaces_client
87 cls.fixed_ips_client = cls.os_primary.fixed_ips_client
88 cls.availability_zone_client = cls.os_primary.availability_zone_client
89 cls.agents_client = cls.os_primary.agents_client
90 cls.aggregates_client = cls.os_primary.aggregates_client
91 cls.services_client = cls.os_primary.services_client
Emily Hugenbruche7991d92014-12-12 16:53:36 +000092 cls.instance_usages_audit_log_client = (
Jordan Pittier8160d312017-04-18 11:52:23 +020093 cls.os_primary.instance_usages_audit_log_client)
94 cls.hypervisor_client = cls.os_primary.hypervisor_client
95 cls.certificates_client = cls.os_primary.certificates_client
96 cls.migrations_client = cls.os_primary.migrations_client
Emily Hugenbruche7991d92014-12-12 16:53:36 +000097 cls.security_group_default_rules_client = (
Jordan Pittier8160d312017-04-18 11:52:23 +020098 cls.os_primary.security_group_default_rules_client)
99 cls.versions_client = cls.os_primary.compute_versions_client
Emily Hugenbruche7991d92014-12-12 16:53:36 +0000100
Jordan Pittier8160d312017-04-18 11:52:23 +0200101 cls.volumes_client = cls.os_primary.volumes_v2_client
Ivan Kolodyazhnybcfc32e2015-08-06 13:31:36 +0300102
Emily Hugenbruche7991d92014-12-12 16:53:36 +0000103 @classmethod
104 def resource_setup(cls):
Ken'ichi Ohmichi02a8ccd2015-11-05 06:05:29 +0000105 super(BaseV2ComputeTest, cls).resource_setup()
Ghanshyam05049dd2016-02-12 17:44:48 +0900106 cls.request_microversion = (
107 api_version_utils.select_request_microversion(
108 cls.min_microversion,
ghanshyam29591532016-03-11 17:12:43 +0900109 CONF.compute.min_microversion))
Matthew Treinishb0a78fc2014-01-29 16:49:12 +0000110 cls.build_interval = CONF.compute.build_interval
111 cls.build_timeout = CONF.compute.build_timeout
Matthew Treinishb0a78fc2014-01-29 16:49:12 +0000112 cls.image_ref = CONF.compute.image_ref
113 cls.image_ref_alt = CONF.compute.image_ref_alt
114 cls.flavor_ref = CONF.compute.flavor_ref
115 cls.flavor_ref_alt = CONF.compute.flavor_ref_alt
lanoux283273b2015-12-04 03:01:54 -0800116 cls.ssh_user = CONF.validation.image_ssh_user
117 cls.image_ssh_user = CONF.validation.image_ssh_user
118 cls.image_ssh_password = CONF.validation.image_ssh_password
Jay Pipesf38eaac2012-06-21 13:37:35 -0400119 cls.servers = []
Sean Dagued62bf1c2013-06-05 14:36:25 -0400120 cls.images = []
Zhi Kun Liu02e7a7b2014-01-08 16:08:32 +0800121 cls.security_groups = []
Abhijeet.Jain87dd4452014-04-23 15:51:23 +0530122 cls.server_groups = []
Matt Riedemann342b37c2016-09-21 15:38:12 -0400123 cls.volumes = []
Matthew Treinishf7fca6a2013-12-09 16:27:23 +0000124
Emily Hugenbruche7991d92014-12-12 16:53:36 +0000125 @classmethod
126 def resource_cleanup(cls):
Masayuki Igawa77d3f0f2017-03-10 18:57:00 +0900127 cls.clear_resources('images', cls.images,
128 cls.compute_images_client.delete_image)
Emily Hugenbruche7991d92014-12-12 16:53:36 +0000129 cls.clear_servers()
Masayuki Igawa77d3f0f2017-03-10 18:57:00 +0900130 cls.clear_resources('security groups', cls.security_groups,
131 cls.security_groups_client.delete_security_group)
132 cls.clear_resources('server groups', cls.server_groups,
133 cls.server_groups_client.delete_server_group)
Matt Riedemann342b37c2016-09-21 15:38:12 -0400134 cls.clear_volumes()
Ken'ichi Ohmichi02a8ccd2015-11-05 06:05:29 +0000135 super(BaseV2ComputeTest, cls).resource_cleanup()
Ken'ichi Ohmichi543a5d42014-05-02 08:44:15 +0900136
Matthew Treinishf7fca6a2013-12-09 16:27:23 +0000137 @classmethod
Jay Pipes444c3e62012-10-04 19:26:35 -0400138 def clear_servers(cls):
Salvatore1422a9a2014-10-08 15:53:25 +0200139 LOG.debug('Clearing servers: %s', ','.join(
140 server['id'] for server in cls.servers))
Jay Pipes444c3e62012-10-04 19:26:35 -0400141 for server in cls.servers:
Dan Smith74e7bcb2012-08-21 09:18:26 -0700142 try:
Jordan Pittier9e227c52016-02-09 14:35:18 +0100143 test_utils.call_and_ignore_notfound_exc(
144 cls.servers_client.delete_server, server['id'])
Joe Gordon0c335792014-09-23 12:36:11 -0700145 except Exception:
Jordan Pittier525ec712016-12-07 17:51:26 +0100146 LOG.exception('Deleting server %s failed', server['id'])
Dan Smith74e7bcb2012-08-21 09:18:26 -0700147
Jay Pipes444c3e62012-10-04 19:26:35 -0400148 for server in cls.servers:
Dan Smith74e7bcb2012-08-21 09:18:26 -0700149 try:
Ken'ichi Ohmichie91a0c62015-08-13 02:09:16 +0000150 waiters.wait_for_server_termination(cls.servers_client,
151 server['id'])
Dan Smith74e7bcb2012-08-21 09:18:26 -0700152 except Exception:
Jordan Pittier525ec712016-12-07 17:51:26 +0100153 LOG.exception('Waiting for deletion of server %s failed',
154 server['id'])
Dan Smith74e7bcb2012-08-21 09:18:26 -0700155
156 @classmethod
Attila Fazekas305e65b2013-10-29 13:23:07 +0100157 def server_check_teardown(cls):
158 """Checks is the shared server clean enough for subsequent test.
Ken'ichi Ohmichi88363cb2015-11-19 08:00:54 +0000159
Attila Fazekas305e65b2013-10-29 13:23:07 +0100160 Method will delete the server when it's dirty.
161 The setUp method is responsible for creating a new server.
162 Exceptions raised in tearDown class are fails the test case,
Marian Horban6afb0232015-11-10 22:47:12 -0500163 This method supposed to use only by tearDown methods, when
Attila Fazekas305e65b2013-10-29 13:23:07 +0100164 the shared server_id is stored in the server_id of the class.
165 """
166 if getattr(cls, 'server_id', None) is not None:
167 try:
Ken'ichi Ohmichi0eb153c2015-07-13 02:18:25 +0000168 waiters.wait_for_server_status(cls.servers_client,
169 cls.server_id, 'ACTIVE')
Attila Fazekas305e65b2013-10-29 13:23:07 +0100170 except Exception as exc:
171 LOG.exception(exc)
172 cls.servers_client.delete_server(cls.server_id)
Ken'ichi Ohmichie91a0c62015-08-13 02:09:16 +0000173 waiters.wait_for_server_termination(cls.servers_client,
174 cls.server_id)
Attila Fazekas305e65b2013-10-29 13:23:07 +0100175 cls.server_id = None
176 raise
177
178 @classmethod
Masayuki Igawa77d3f0f2017-03-10 18:57:00 +0900179 def clear_resources(cls, resource_name, resources, resource_del_func):
180 LOG.debug('Clearing %s: %s', resource_name,
181 ','.join(map(str, resources)))
182 for res_id in resources:
Sean Dagued62bf1c2013-06-05 14:36:25 -0400183 try:
Jordan Pittier9e227c52016-02-09 14:35:18 +0100184 test_utils.call_and_ignore_notfound_exc(
Masayuki Igawa77d3f0f2017-03-10 18:57:00 +0900185 resource_del_func, res_id)
Zhi Kun Liu02e7a7b2014-01-08 16:08:32 +0800186 except Exception as exc:
Masayuki Igawa77d3f0f2017-03-10 18:57:00 +0900187 LOG.exception('Exception raised deleting %s: %s',
188 resource_name, res_id)
Zhi Kun Liu02e7a7b2014-01-08 16:08:32 +0800189 LOG.exception(exc)
Zhi Kun Liu02e7a7b2014-01-08 16:08:32 +0800190
191 @classmethod
Joe Gordon8843f0f2015-03-17 15:07:34 -0700192 def create_test_server(cls, validatable=False, volume_backed=False,
193 **kwargs):
Joseph Lanouxb3e1f872015-01-30 11:13:07 +0000194 """Wrapper utility that returns a test server.
Rohit Karajgidc300b22012-05-04 08:11:00 -0700195
Joseph Lanouxb3e1f872015-01-30 11:13:07 +0000196 This wrapper utility calls the common create test server and
197 returns a test server. The purpose of this wrapper is to minimize
198 the impact on the code of the tests already using this
199 function.
Joe Gordon8843f0f2015-03-17 15:07:34 -0700200
201 :param validatable: Whether the server will be pingable or sshable.
202 :param volume_backed: Whether the instance is volume backed or not.
Joseph Lanouxb3e1f872015-01-30 11:13:07 +0000203 """
zhufl7ae22682016-09-18 15:22:33 +0800204 if 'name' not in kwargs:
205 kwargs['name'] = data_utils.rand_name(cls.__name__ + "-server")
Sergey Nikitin8654e5b2017-06-04 22:09:56 +0400206
207 request_version = api_version_request.APIVersionRequest(
208 cls.request_microversion)
209 v2_37_version = api_version_request.APIVersionRequest('2.37')
210
211 # NOTE(snikitin): since microversion v2.37 'networks' field is required
212 if request_version >= v2_37_version and 'networks' not in kwargs:
213 kwargs['networks'] = 'none'
214
Joseph Lanouxb3e1f872015-01-30 11:13:07 +0000215 tenant_network = cls.get_tenant_network()
216 body, servers = compute.create_test_server(
zhufl04190882017-05-23 10:21:48 +0800217 cls.os_primary,
Joseph Lanouxb3e1f872015-01-30 11:13:07 +0000218 validatable,
219 validation_resources=cls.validation_resources,
220 tenant_network=tenant_network,
Joe Gordon8843f0f2015-03-17 15:07:34 -0700221 volume_backed=volume_backed,
Joseph Lanouxb3e1f872015-01-30 11:13:07 +0000222 **kwargs)
Ken'ichi Ohmichi51c8c262013-12-21 03:30:37 +0900223
224 cls.servers.extend(servers)
Sean Dague9b669e32012-12-13 18:40:08 -0500225
David Kranz0fb14292015-02-11 15:55:20 -0500226 return body
Sean Dague9b669e32012-12-13 18:40:08 -0500227
Zhi Kun Liu02e7a7b2014-01-08 16:08:32 +0800228 @classmethod
229 def create_security_group(cls, name=None, description=None):
230 if name is None:
231 name = data_utils.rand_name(cls.__name__ + "-securitygroup")
232 if description is None:
Ken'ichi Ohmichi4937f562015-03-23 00:15:01 +0000233 description = data_utils.rand_name('description')
ghanshyamb610b772015-08-24 17:29:38 +0900234 body = cls.security_groups_client.create_security_group(
235 name=name, description=description)['security_group']
Masayuki Igawa77d3f0f2017-03-10 18:57:00 +0900236 cls.security_groups.append(body['id'])
Zhi Kun Liu02e7a7b2014-01-08 16:08:32 +0800237
David Kranz9964b4e2015-02-06 15:45:29 -0500238 return body
Zhi Kun Liu02e7a7b2014-01-08 16:08:32 +0800239
Abhijeet.Jain87dd4452014-04-23 15:51:23 +0530240 @classmethod
Ghanshyam2a180b82014-06-16 13:54:22 +0900241 def create_test_server_group(cls, name="", policy=None):
Abhijeet.Jain87dd4452014-04-23 15:51:23 +0530242 if not name:
243 name = data_utils.rand_name(cls.__name__ + "-Server-Group")
Ghanshyam2a180b82014-06-16 13:54:22 +0900244 if policy is None:
Abhijeet.Jain87dd4452014-04-23 15:51:23 +0530245 policy = ['affinity']
Ken'ichi Ohmichi1f36daa2015-09-30 01:41:34 +0000246 body = cls.server_groups_client.create_server_group(
247 name=name, policies=policy)['server_group']
David Kranzda5d4ec2014-06-24 16:04:57 -0400248 cls.server_groups.append(body['id'])
David Kranzae99b9a2015-02-16 13:37:01 -0500249 return body
Abhijeet.Jain87dd4452014-04-23 15:51:23 +0530250
David Kranzcf0040c2012-06-26 09:46:56 -0400251 def wait_for(self, condition):
Sean Daguef237ccb2013-01-04 15:19:14 -0500252 """Repeatedly calls condition() until a timeout."""
David Kranzcf0040c2012-06-26 09:46:56 -0400253 start_time = int(time.time())
254 while True:
255 try:
256 condition()
Matthew Treinish05d9fb92012-12-07 16:14:05 -0500257 except Exception:
David Kranzcf0040c2012-06-26 09:46:56 -0400258 pass
259 else:
260 return
261 if int(time.time()) - start_time >= self.build_timeout:
262 condition()
263 return
264 time.sleep(self.build_interval)
Jay Pipesf38eaac2012-06-21 13:37:35 -0400265
Matt Riedemann807d0562014-01-27 12:03:10 -0800266 @staticmethod
267 def _delete_volume(volumes_client, volume_id):
268 """Deletes the given volume and waits for it to be gone."""
269 try:
Joseph Lanoux6809bab2014-12-18 14:57:18 +0000270 volumes_client.delete_volume(volume_id)
Matt Riedemann807d0562014-01-27 12:03:10 -0800271 # TODO(mriedem): We should move the wait_for_resource_deletion
272 # into the delete_volume method as a convenience to the caller.
273 volumes_client.wait_for_resource_deletion(volume_id)
Masayuki Igawabfa07602015-01-20 18:47:17 +0900274 except lib_exc.NotFound:
zhangguoqing6c096642016-01-04 06:17:21 +0000275 LOG.warning("Unable to delete volume '%s' since it was not found. "
Jordan Pittier525ec712016-12-07 17:51:26 +0100276 "Maybe it was already deleted?", volume_id)
Matt Riedemann807d0562014-01-27 12:03:10 -0800277
Attila Fazekas423834d2014-03-14 17:33:13 +0100278 @classmethod
279 def prepare_instance_network(cls):
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000280 if (CONF.validation.auth_method != 'disabled' and
281 CONF.validation.connect_method == 'floating'):
Attila Fazekas423834d2014-03-14 17:33:13 +0100282 cls.set_network_resources(network=True, subnet=True, router=True,
283 dhcp=True)
284
ivan-zhu8f992be2013-07-31 14:56:58 +0800285 @classmethod
286 def create_image_from_server(cls, server_id, **kwargs):
287 """Wrapper utility that returns an image created from the server."""
zhufl35a694b2017-02-14 17:10:53 +0800288 name = kwargs.pop('name',
289 data_utils.rand_name(cls.__name__ + "-image"))
290 wait_until = kwargs.pop('wait_until', None)
291 wait_for_server = kwargs.pop('wait_for_server', True)
ivan-zhu8f992be2013-07-31 14:56:58 +0800292
zhufl35a694b2017-02-14 17:10:53 +0800293 image = cls.compute_images_client.create_image(server_id, name=name,
294 **kwargs)
David Kranza5299eb2015-01-15 17:24:05 -0500295 image_id = data_utils.parse_image_id(image.response['location'])
ivan-zhu8f992be2013-07-31 14:56:58 +0800296 cls.images.append(image_id)
297
zhufl35a694b2017-02-14 17:10:53 +0800298 if wait_until is not None:
Matt Riedemann13954352017-02-07 14:03:54 -0500299 try:
300 waiters.wait_for_image_status(cls.compute_images_client,
zhufl35a694b2017-02-14 17:10:53 +0800301 image_id, wait_until)
Matt Riedemann13954352017-02-07 14:03:54 -0500302 except lib_exc.NotFound:
zhufl35a694b2017-02-14 17:10:53 +0800303 if wait_until.upper() == 'ACTIVE':
Matt Riedemann13954352017-02-07 14:03:54 -0500304 # If the image is not found after create_image returned
305 # that means the snapshot failed in nova-compute and nova
306 # deleted the image. There should be a compute fault
307 # recorded with the server in that case, so get the server
308 # and dump some details.
309 server = (
310 cls.servers_client.show_server(server_id)['server'])
311 if 'fault' in server:
312 raise exceptions.SnapshotNotFoundException(
313 server['fault'], image_id=image_id)
314 else:
315 raise exceptions.SnapshotNotFoundException(
316 image_id=image_id)
317 else:
318 raise
Ghanshyamae76c122015-12-22 13:41:35 +0900319 image = cls.compute_images_client.show_image(image_id)['image']
ivan-zhu8f992be2013-07-31 14:56:58 +0800320
zhufl35a694b2017-02-14 17:10:53 +0800321 if wait_until.upper() == 'ACTIVE':
322 if wait_for_server:
Bob Ball5fe62392017-02-20 09:51:00 +0000323 waiters.wait_for_server_status(cls.servers_client,
324 server_id, 'ACTIVE')
David Kranza5299eb2015-01-15 17:24:05 -0500325 return image
ivan-zhu8f992be2013-07-31 14:56:58 +0800326
327 @classmethod
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000328 def rebuild_server(cls, server_id, validatable=False, **kwargs):
ivan-zhu8f992be2013-07-31 14:56:58 +0800329 # Destroy an existing server and creates a new one
Matthew Treinish2cd19a42013-12-02 21:54:42 +0000330 if server_id:
zhufl9b682902016-12-15 09:16:34 +0800331 cls.delete_server(server_id)
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000332
Ghanshyam3390d9f2015-12-25 12:48:02 +0900333 cls.password = data_utils.rand_password()
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000334 server = cls.create_test_server(
335 validatable,
336 wait_until='ACTIVE',
Ghanshyam3390d9f2015-12-25 12:48:02 +0900337 adminPass=cls.password,
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000338 **kwargs)
Matthew Treinish2cd19a42013-12-02 21:54:42 +0000339 return server['id']
ivan-zhu8f992be2013-07-31 14:56:58 +0800340
Matt Riedemann5dc594c2014-01-27 11:40:28 -0800341 @classmethod
Jesse Keating613b4982015-05-04 15:05:19 -0700342 def delete_server(cls, server_id):
343 """Deletes an existing server and waits for it to be gone."""
344 try:
345 cls.servers_client.delete_server(server_id)
Ken'ichi Ohmichie91a0c62015-08-13 02:09:16 +0000346 waiters.wait_for_server_termination(cls.servers_client,
347 server_id)
Jesse Keating613b4982015-05-04 15:05:19 -0700348 except Exception:
Jordan Pittier525ec712016-12-07 17:51:26 +0100349 LOG.exception('Failed to delete server %s', server_id)
Jesse Keating613b4982015-05-04 15:05:19 -0700350
351 @classmethod
zhufl3d018b02016-11-25 16:43:04 +0800352 def resize_server(cls, server_id, new_flavor_id, **kwargs):
353 """resize and confirm_resize an server, waits for it to be ACTIVE."""
354 cls.servers_client.resize_server(server_id, new_flavor_id, **kwargs)
355 waiters.wait_for_server_status(cls.servers_client, server_id,
356 'VERIFY_RESIZE')
357 cls.servers_client.confirm_resize_server(server_id)
358 waiters.wait_for_server_status(cls.servers_client, server_id, 'ACTIVE')
359
360 @classmethod
Matt Riedemann5dc594c2014-01-27 11:40:28 -0800361 def delete_volume(cls, volume_id):
362 """Deletes the given volume and waits for it to be gone."""
Felipe Monteiroa40e9b72017-05-05 17:47:10 +0100363 cls._delete_volume(cls.volumes_client, volume_id)
Ken'ichi Ohmichi543a5d42014-05-02 08:44:15 +0900364
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000365 @classmethod
366 def get_server_ip(cls, server):
367 """Get the server fixed or floating IP.
368
Sean Dague20e98612016-01-06 14:33:28 -0500369 Based on the configuration we're in, return a correct ip
370 address for validating that a guest is up.
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000371 """
372 if CONF.validation.connect_method == 'floating':
Sean Dague20e98612016-01-06 14:33:28 -0500373 return cls.validation_resources['floating_ip']['ip']
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000374 elif CONF.validation.connect_method == 'fixed':
Sean Dague20e98612016-01-06 14:33:28 -0500375 addresses = server['addresses'][CONF.validation.network_for_ssh]
376 for address in addresses:
377 if address['version'] == CONF.validation.ip_version_for_ssh:
378 return address['addr']
zhufl955f82b2016-07-22 11:14:34 +0800379 raise exceptions.ServerUnreachable(server_id=server['id'])
Sean Dague20e98612016-01-06 14:33:28 -0500380 else:
guo yunxianffc4fc02016-11-15 09:56:08 +0800381 raise lib_exc.InvalidConfiguration()
Joseph Lanouxffe09dd2015-03-18 16:45:33 +0000382
Ghanshyam05049dd2016-02-12 17:44:48 +0900383 def setUp(self):
384 super(BaseV2ComputeTest, self).setUp()
385 self.useFixture(api_microversion_fixture.APIMicroversionFixture(
386 self.request_microversion))
387
Matt Riedemann342b37c2016-09-21 15:38:12 -0400388 @classmethod
zhufl8d23f922016-12-12 17:29:42 +0800389 def create_volume(cls, image_ref=None, **kwargs):
Matt Riedemann342b37c2016-09-21 15:38:12 -0400390 """Create a volume and wait for it to become 'available'.
391
Artom Lifshitzfc8f8e62016-04-13 11:08:32 +0000392 :param image_ref: Specify an image id to create a bootable volume.
zhufl8d23f922016-12-12 17:29:42 +0800393 :**kwargs: other parameters to create volume.
Matt Riedemann342b37c2016-09-21 15:38:12 -0400394 :returns: The available volume.
395 """
zhufl8d23f922016-12-12 17:29:42 +0800396 if 'size' not in kwargs:
397 kwargs['size'] = CONF.volume.volume_size
398 if 'display_name' not in kwargs:
399 vol_name = data_utils.rand_name(cls.__name__ + '-volume')
400 kwargs['display_name'] = vol_name
Artom Lifshitzfc8f8e62016-04-13 11:08:32 +0000401 if image_ref is not None:
zhufl8d23f922016-12-12 17:29:42 +0800402 kwargs['imageRef'] = image_ref
403 volume = cls.volumes_client.create_volume(**kwargs)['volume']
Matt Riedemann342b37c2016-09-21 15:38:12 -0400404 cls.volumes.append(volume)
lkuchlan52d7b0d2016-11-07 20:53:19 +0200405 waiters.wait_for_volume_resource_status(cls.volumes_client,
406 volume['id'], 'available')
Matt Riedemann342b37c2016-09-21 15:38:12 -0400407 return volume
408
409 @classmethod
410 def clear_volumes(cls):
411 LOG.debug('Clearing volumes: %s', ','.join(
412 volume['id'] for volume in cls.volumes))
413 for volume in cls.volumes:
414 try:
415 test_utils.call_and_ignore_notfound_exc(
416 cls.volumes_client.delete_volume, volume['id'])
417 except Exception:
418 LOG.exception('Deleting volume %s failed', volume['id'])
419
420 for volume in cls.volumes:
421 try:
422 cls.volumes_client.wait_for_resource_deletion(volume['id'])
423 except Exception:
424 LOG.exception('Waiting for deletion of volume %s failed',
425 volume['id'])
426
John Griffith030dd172017-06-27 23:22:14 +0000427 def attach_volume(self, server, volume, device=None, check_reserved=False):
Matt Riedemanncb16a662016-10-01 18:30:05 -0400428 """Attaches volume to server and waits for 'in-use' volume status.
429
430 The volume will be detached when the test tears down.
431
432 :param server: The server to which the volume will be attached.
433 :param volume: The volume to attach.
434 :param device: Optional mountpoint for the attached volume. Note that
435 this is not guaranteed for all hypervisors and is not recommended.
John Griffith030dd172017-06-27 23:22:14 +0000436 :param check_reserved: Consider a status of reserved as valid for
437 completion. This is to handle new Cinder attach where we more
438 accurately use 'reserved' for things like attaching to a shelved
439 server.
Matt Riedemanncb16a662016-10-01 18:30:05 -0400440 """
441 attach_kwargs = dict(volumeId=volume['id'])
442 if device:
443 attach_kwargs['device'] = device
John Griffith030dd172017-06-27 23:22:14 +0000444
zhufl36f0a972017-02-28 15:43:33 +0800445 attachment = self.servers_client.attach_volume(
446 server['id'], **attach_kwargs)['volumeAttachment']
Matt Riedemann342b37c2016-09-21 15:38:12 -0400447 # On teardown detach the volume and wait for it to be available. This
448 # is so we don't error out when trying to delete the volume during
449 # teardown.
lkuchlan52d7b0d2016-11-07 20:53:19 +0200450 self.addCleanup(waiters.wait_for_volume_resource_status,
Matt Riedemann342b37c2016-09-21 15:38:12 -0400451 self.volumes_client, volume['id'], 'available')
452 # Ignore 404s on detach in case the server is deleted or the volume
453 # is already detached.
454 self.addCleanup(test_utils.call_and_ignore_notfound_exc,
455 self.servers_client.detach_volume,
456 server['id'], volume['id'])
John Griffith030dd172017-06-27 23:22:14 +0000457 statuses = ['in-use']
458 if check_reserved:
459 statuses.append('reserved')
lkuchlan52d7b0d2016-11-07 20:53:19 +0200460 waiters.wait_for_volume_resource_status(self.volumes_client,
John Griffith030dd172017-06-27 23:22:14 +0000461 volume['id'], statuses)
zhufl36f0a972017-02-28 15:43:33 +0800462 return attachment
Matt Riedemann342b37c2016-09-21 15:38:12 -0400463
Ken'ichi Ohmichi543a5d42014-05-02 08:44:15 +0900464
Ken'ichi Ohmichi02a8ccd2015-11-05 06:05:29 +0000465class BaseV2ComputeAdminTest(BaseV2ComputeTest):
Ken'ichi Ohmichibcefa3d2014-05-09 08:14:05 +0900466 """Base test case class for Compute Admin API tests."""
ivan-zhuf2b00502013-10-18 10:06:52 +0800467
Andrea Frittolib21de6c2015-02-06 20:12:38 +0000468 credentials = ['primary', 'admin']
Emily Hugenbruche7991d92014-12-12 16:53:36 +0000469
470 @classmethod
471 def setup_clients(cls):
Ken'ichi Ohmichi02a8ccd2015-11-05 06:05:29 +0000472 super(BaseV2ComputeAdminTest, cls).setup_clients()
Ken'ichi Ohmichi9f5adf82014-12-12 04:01:32 +0000473 cls.availability_zone_admin_client = (
Jordan Pittier8160d312017-04-18 11:52:23 +0200474 cls.os_admin.availability_zone_client)
475 cls.admin_flavors_client = cls.os_admin.flavors_client
476 cls.admin_servers_client = cls.os_admin.servers_client
zhufl36eeab02017-01-18 11:49:04 +0800477
478 def create_flavor(self, ram, vcpus, disk, name=None,
479 is_public='True', **kwargs):
480 if name is None:
481 name = data_utils.rand_name(self.__class__.__name__ + "-flavor")
482 id = kwargs.pop('id', data_utils.rand_int_id(start=1000))
483 client = self.admin_flavors_client
484 flavor = client.create_flavor(
485 ram=ram, vcpus=vcpus, disk=disk, name=name,
486 id=id, is_public=is_public, **kwargs)['flavor']
487 self.addCleanup(client.wait_for_resource_deletion, flavor['id'])
488 self.addCleanup(client.delete_flavor, flavor['id'])
489 return flavor