blob: 024fe43facd37cd30932e5d000eb0a43fa2bf996 [file] [log] [blame]
Daniel Mellado3c0aeab2016-01-29 11:30:25 +00001# Copyright 2012 OpenStack Foundation
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
Ihar Hrachyshka59382252016-04-05 15:54:33 +020016import functools
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +020017import math
Lajos Katona2f904652018-08-23 14:04:56 +020018import time
Ihar Hrachyshka59382252016-04-05 15:54:33 +020019
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000020import netaddr
Chandan Kumarc125fd12017-11-15 19:41:01 +053021from neutron_lib import constants as const
Lajos Katona2f904652018-08-23 14:04:56 +020022from oslo_log import log
Chandan Kumarc125fd12017-11-15 19:41:01 +053023from tempest.common import utils as tutils
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000024from tempest.lib.common.utils import data_utils
25from tempest.lib import exceptions as lib_exc
26from tempest import test
27
Chandan Kumar667d3d32017-09-22 12:24:06 +053028from neutron_tempest_plugin.api import clients
29from neutron_tempest_plugin.common import constants
30from neutron_tempest_plugin.common import utils
31from neutron_tempest_plugin import config
32from neutron_tempest_plugin import exceptions
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000033
34CONF = config.CONF
35
Lajos Katona2f904652018-08-23 14:04:56 +020036LOG = log.getLogger(__name__)
37
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000038
39class BaseNetworkTest(test.BaseTestCase):
40
Brian Haleyae328b92018-10-09 19:51:54 -040041 """Base class for Neutron tests that use the Tempest Neutron REST client
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000042
43 Per the Neutron API Guide, API v1.x was removed from the source code tree
44 (docs.openstack.org/api/openstack-network/2.0/content/Overview-d1e71.html)
45 Therefore, v2.x of the Neutron API is assumed. It is also assumed that the
46 following options are defined in the [network] section of etc/tempest.conf:
47
48 project_network_cidr with a block of cidr's from which smaller blocks
49 can be allocated for tenant networks
50
51 project_network_mask_bits with the mask bits to be used to partition
52 the block defined by tenant-network_cidr
53
54 Finally, it is assumed that the following option is defined in the
55 [service_available] section of etc/tempest.conf
56
57 neutron as True
58 """
59
60 force_tenant_isolation = False
61 credentials = ['primary']
62
63 # Default to ipv4.
Federico Ressi0ddc93b2018-04-09 12:01:48 +020064 _ip_version = const.IP_VERSION_4
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000065
Federico Ressi61b564e2018-07-06 08:10:31 +020066 # Derive from BaseAdminNetworkTest class to have this initialized
67 admin_client = None
68
Federico Ressia69dcd52018-07-06 09:45:34 +020069 external_network_id = CONF.network.public_network_id
70
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000071 @classmethod
72 def get_client_manager(cls, credential_type=None, roles=None,
73 force_new=None):
Genadi Chereshnyacc395c02016-07-25 12:17:37 +030074 manager = super(BaseNetworkTest, cls).get_client_manager(
75 credential_type=credential_type,
76 roles=roles,
77 force_new=force_new
78 )
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000079 # Neutron uses a different clients manager than the one in the Tempest
Jens Harbott860b46a2017-11-15 21:23:15 +000080 # save the original in case mixed tests need it
81 if credential_type == 'primary':
82 cls.os_tempest = manager
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000083 return clients.Manager(manager.credentials)
84
85 @classmethod
86 def skip_checks(cls):
87 super(BaseNetworkTest, cls).skip_checks()
88 if not CONF.service_available.neutron:
89 raise cls.skipException("Neutron support is required")
Federico Ressi0ddc93b2018-04-09 12:01:48 +020090 if (cls._ip_version == const.IP_VERSION_6 and
91 not CONF.network_feature_enabled.ipv6):
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000092 raise cls.skipException("IPv6 Tests are disabled.")
Jakub Libosvar1982aa12017-05-30 11:15:33 +000093 for req_ext in getattr(cls, 'required_extensions', []):
Chandan Kumarc125fd12017-11-15 19:41:01 +053094 if not tutils.is_extension_enabled(req_ext, 'network'):
Jakub Libosvar1982aa12017-05-30 11:15:33 +000095 msg = "%s extension not enabled." % req_ext
96 raise cls.skipException(msg)
Daniel Mellado3c0aeab2016-01-29 11:30:25 +000097
98 @classmethod
99 def setup_credentials(cls):
100 # Create no network resources for these test.
101 cls.set_network_resources()
102 super(BaseNetworkTest, cls).setup_credentials()
103
104 @classmethod
105 def setup_clients(cls):
106 super(BaseNetworkTest, cls).setup_clients()
fumihiko kakumaa216fc12017-07-14 10:43:29 +0900107 cls.client = cls.os_primary.network_client
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000108
109 @classmethod
110 def resource_setup(cls):
111 super(BaseNetworkTest, cls).resource_setup()
112
113 cls.networks = []
Miguel Lavalle124378b2016-09-21 16:41:47 -0500114 cls.admin_networks = []
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000115 cls.subnets = []
Kevin Bentonba3651c2017-09-01 17:13:01 -0700116 cls.admin_subnets = []
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000117 cls.ports = []
118 cls.routers = []
119 cls.floating_ips = []
Slawek Kaplonski003fcae2019-05-26 22:38:35 +0200120 cls.port_forwardings = []
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000121 cls.metering_labels = []
122 cls.service_profiles = []
123 cls.flavors = []
124 cls.metering_label_rules = []
125 cls.qos_rules = []
126 cls.qos_policies = []
127 cls.ethertype = "IPv" + str(cls._ip_version)
Miguel Lavalleb1c7a3d2021-01-31 19:05:22 -0600128 cls.address_groups = []
129 cls.admin_address_groups = []
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000130 cls.address_scopes = []
131 cls.admin_address_scopes = []
132 cls.subnetpools = []
133 cls.admin_subnetpools = []
Itzik Brownbac51dc2016-10-31 12:25:04 +0000134 cls.security_groups = []
Dongcan Ye2de722e2018-07-04 11:01:37 +0000135 cls.admin_security_groups = []
Chandan Kumarc125fd12017-11-15 19:41:01 +0530136 cls.projects = []
Nguyen Phuong An67993fc2017-11-24 11:30:25 +0700137 cls.log_objects = []
Federico Ressi0ddc93b2018-04-09 12:01:48 +0200138 cls.reserved_subnet_cidrs = set()
Federico Ressiab286e42018-06-19 09:52:10 +0200139 cls.keypairs = []
Federico Ressi82e83e32018-07-03 14:19:55 +0200140 cls.trunks = []
Kailun Qineaaf9782018-12-20 04:45:01 +0800141 cls.network_segment_ranges = []
Harald Jensåsc9782fa2019-06-03 22:35:41 +0200142 cls.conntrack_helpers = []
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000143
144 @classmethod
yangjianfeng23e40c22020-11-22 08:42:18 +0000145 def reserve_external_subnet_cidrs(cls):
146 client = cls.os_admin.network_client
147 ext_nets = client.list_networks(
148 **{"router:external": True})['networks']
149 for ext_net in ext_nets:
150 ext_subnets = client.list_subnets(
151 network_id=ext_net['id'])['subnets']
152 for ext_subnet in ext_subnets:
153 cls.reserve_subnet_cidr(ext_subnet['cidr'])
154
155 @classmethod
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000156 def resource_cleanup(cls):
157 if CONF.service_available.neutron:
Federico Ressi82e83e32018-07-03 14:19:55 +0200158 # Clean up trunks
159 for trunk in cls.trunks:
160 cls._try_delete_resource(cls.delete_trunk, trunk)
161
Slawek Kaplonski003fcae2019-05-26 22:38:35 +0200162 # Clean up port forwardings
163 for pf in cls.port_forwardings:
164 cls._try_delete_resource(cls.delete_port_forwarding, pf)
165
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000166 # Clean up floating IPs
167 for floating_ip in cls.floating_ips:
Federico Ressia69dcd52018-07-06 09:45:34 +0200168 cls._try_delete_resource(cls.delete_floatingip, floating_ip)
169
Harald Jensåsc9782fa2019-06-03 22:35:41 +0200170 # Clean up conntrack helpers
171 for cth in cls.conntrack_helpers:
172 cls._try_delete_resource(cls.delete_conntrack_helper, cth)
173
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000174 # Clean up routers
175 for router in cls.routers:
176 cls._try_delete_resource(cls.delete_router,
177 router)
178 # Clean up metering label rules
179 for metering_label_rule in cls.metering_label_rules:
180 cls._try_delete_resource(
181 cls.admin_client.delete_metering_label_rule,
182 metering_label_rule['id'])
183 # Clean up metering labels
184 for metering_label in cls.metering_labels:
185 cls._try_delete_resource(
186 cls.admin_client.delete_metering_label,
187 metering_label['id'])
188 # Clean up flavors
189 for flavor in cls.flavors:
190 cls._try_delete_resource(
191 cls.admin_client.delete_flavor,
192 flavor['id'])
193 # Clean up service profiles
194 for service_profile in cls.service_profiles:
195 cls._try_delete_resource(
196 cls.admin_client.delete_service_profile,
197 service_profile['id'])
198 # Clean up ports
199 for port in cls.ports:
200 cls._try_delete_resource(cls.client.delete_port,
201 port['id'])
202 # Clean up subnets
203 for subnet in cls.subnets:
204 cls._try_delete_resource(cls.client.delete_subnet,
205 subnet['id'])
Kevin Bentonba3651c2017-09-01 17:13:01 -0700206 # Clean up admin subnets
207 for subnet in cls.admin_subnets:
208 cls._try_delete_resource(cls.admin_client.delete_subnet,
209 subnet['id'])
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000210 # Clean up networks
211 for network in cls.networks:
Federico Ressi61b564e2018-07-06 08:10:31 +0200212 cls._try_delete_resource(cls.delete_network, network)
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000213
Miguel Lavalle124378b2016-09-21 16:41:47 -0500214 # Clean up admin networks
215 for network in cls.admin_networks:
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000216 cls._try_delete_resource(cls.admin_client.delete_network,
217 network['id'])
218
Itzik Brownbac51dc2016-10-31 12:25:04 +0000219 # Clean up security groups
Federico Ressi4c590d72018-10-10 14:01:08 +0200220 for security_group in cls.security_groups:
221 cls._try_delete_resource(cls.delete_security_group,
222 security_group)
Itzik Brownbac51dc2016-10-31 12:25:04 +0000223
Dongcan Ye2de722e2018-07-04 11:01:37 +0000224 # Clean up admin security groups
Federico Ressi4c590d72018-10-10 14:01:08 +0200225 for security_group in cls.admin_security_groups:
226 cls._try_delete_resource(cls.delete_security_group,
227 security_group,
228 client=cls.admin_client)
Dongcan Ye2de722e2018-07-04 11:01:37 +0000229
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000230 for subnetpool in cls.subnetpools:
231 cls._try_delete_resource(cls.client.delete_subnetpool,
232 subnetpool['id'])
233
234 for subnetpool in cls.admin_subnetpools:
235 cls._try_delete_resource(cls.admin_client.delete_subnetpool,
236 subnetpool['id'])
237
238 for address_scope in cls.address_scopes:
239 cls._try_delete_resource(cls.client.delete_address_scope,
240 address_scope['id'])
241
242 for address_scope in cls.admin_address_scopes:
243 cls._try_delete_resource(
244 cls.admin_client.delete_address_scope,
245 address_scope['id'])
246
Chandan Kumarc125fd12017-11-15 19:41:01 +0530247 for project in cls.projects:
248 cls._try_delete_resource(
249 cls.identity_admin_client.delete_project,
250 project['id'])
251
Sławek Kapłońskie100c4d2017-08-23 21:18:34 +0000252 # Clean up QoS rules
253 for qos_rule in cls.qos_rules:
254 cls._try_delete_resource(cls.admin_client.delete_qos_rule,
255 qos_rule['id'])
256 # Clean up QoS policies
257 # as all networks and ports are already removed, QoS policies
258 # shouldn't be "in use"
259 for qos_policy in cls.qos_policies:
260 cls._try_delete_resource(cls.admin_client.delete_qos_policy,
261 qos_policy['id'])
262
Nguyen Phuong An67993fc2017-11-24 11:30:25 +0700263 # Clean up log_objects
264 for log_object in cls.log_objects:
265 cls._try_delete_resource(cls.admin_client.delete_log,
266 log_object['id'])
267
Federico Ressiab286e42018-06-19 09:52:10 +0200268 for keypair in cls.keypairs:
269 cls._try_delete_resource(cls.delete_keypair, keypair)
270
Kailun Qineaaf9782018-12-20 04:45:01 +0800271 # Clean up network_segment_ranges
272 for network_segment_range in cls.network_segment_ranges:
273 cls._try_delete_resource(
274 cls.admin_client.delete_network_segment_range,
275 network_segment_range['id'])
276
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000277 super(BaseNetworkTest, cls).resource_cleanup()
278
279 @classmethod
280 def _try_delete_resource(cls, delete_callable, *args, **kwargs):
281 """Cleanup resources in case of test-failure
282
283 Some resources are explicitly deleted by the test.
284 If the test failed to delete a resource, this method will execute
285 the appropriate delete methods. Otherwise, the method ignores NotFound
286 exceptions thrown for resources that were correctly deleted by the
287 test.
288
289 :param delete_callable: delete method
290 :param args: arguments for delete method
291 :param kwargs: keyword arguments for delete method
292 """
293 try:
294 delete_callable(*args, **kwargs)
295 # if resource is not found, this means it was deleted in the test
296 except lib_exc.NotFound:
297 pass
298
299 @classmethod
Federico Ressi61b564e2018-07-06 08:10:31 +0200300 def create_network(cls, network_name=None, client=None, external=None,
301 shared=None, provider_network_type=None,
302 provider_physical_network=None,
303 provider_segmentation_id=None, **kwargs):
304 """Create a network.
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000305
Federico Ressi61b564e2018-07-06 08:10:31 +0200306 When client is not provider and admin_client is attribute is not None
307 (for example when using BaseAdminNetworkTest base class) and using any
308 of the convenience parameters (external, shared, provider_network_type,
309 provider_physical_network and provider_segmentation_id) it silently
310 uses admin_client. If the network is not shared then it uses the same
311 project_id as regular client.
312
313 :param network_name: Human-readable name of the network
314
315 :param client: client to be used for connecting to network service
316
317 :param external: indicates whether the network has an external routing
318 facility that's not managed by the networking service.
319
320 :param shared: indicates whether this resource is shared across all
321 projects. By default, only administrative users can change this value.
322 If True and admin_client attribute is not None, then the network is
323 created under administrative project.
324
325 :param provider_network_type: the type of physical network that this
326 network should be mapped to. For example, 'flat', 'vlan', 'vxlan', or
327 'gre'. Valid values depend on a networking back-end.
328
329 :param provider_physical_network: the physical network where this
330 network should be implemented. The Networking API v2.0 does not provide
331 a way to list available physical networks. For example, the Open
332 vSwitch plug-in configuration file defines a symbolic name that maps to
333 specific bridges on each compute host.
334
335 :param provider_segmentation_id: The ID of the isolated segment on the
336 physical network. The network_type attribute defines the segmentation
337 model. For example, if the network_type value is 'vlan', this ID is a
338 vlan identifier. If the network_type value is 'gre', this ID is a gre
339 key.
340
341 :param **kwargs: extra parameters to be forwarded to network service
342 """
343
344 name = (network_name or kwargs.pop('name', None) or
345 data_utils.rand_name('test-network-'))
346
347 # translate convenience parameters
348 admin_client_required = False
349 if provider_network_type:
350 admin_client_required = True
351 kwargs['provider:network_type'] = provider_network_type
352 if provider_physical_network:
353 admin_client_required = True
354 kwargs['provider:physical_network'] = provider_physical_network
355 if provider_segmentation_id:
356 admin_client_required = True
357 kwargs['provider:segmentation_id'] = provider_segmentation_id
358 if external is not None:
359 admin_client_required = True
360 kwargs['router:external'] = bool(external)
361 if shared is not None:
362 admin_client_required = True
363 kwargs['shared'] = bool(shared)
364
365 if not client:
366 if admin_client_required and cls.admin_client:
367 # For convenience silently switch to admin client
368 client = cls.admin_client
369 if not shared:
370 # Keep this network visible from current project
371 project_id = (kwargs.get('project_id') or
372 kwargs.get('tenant_id') or
373 cls.client.tenant_id)
374 kwargs.update(project_id=project_id, tenant_id=project_id)
375 else:
376 # Use default client
377 client = cls.client
378
379 network = client.create_network(name=name, **kwargs)['network']
380 network['client'] = client
381 cls.networks.append(network)
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000382 return network
383
384 @classmethod
Federico Ressi61b564e2018-07-06 08:10:31 +0200385 def delete_network(cls, network, client=None):
386 client = client or network.get('client') or cls.client
387 client.delete_network(network['id'])
388
389 @classmethod
390 def create_shared_network(cls, network_name=None, **kwargs):
391 return cls.create_network(name=network_name, shared=True, **kwargs)
Miguel Lavalle124378b2016-09-21 16:41:47 -0500392
393 @classmethod
Sławek Kapłońskid98e27d2018-05-07 16:16:28 +0200394 def create_subnet(cls, network, gateway='', cidr=None, mask_bits=None,
Federico Ressi98f20ec2018-05-11 06:09:49 +0200395 ip_version=None, client=None, reserve_cidr=True,
396 **kwargs):
Federico Ressi0ddc93b2018-04-09 12:01:48 +0200397 """Wrapper utility that returns a test subnet.
398
399 Convenient wrapper for client.create_subnet method. It reserves and
400 allocates CIDRs to avoid creating overlapping subnets.
401
402 :param network: network where to create the subnet
403 network['id'] must contain the ID of the network
404
405 :param gateway: gateway IP address
406 It can be a str or a netaddr.IPAddress
407 If gateway is not given, then it will use default address for
408 given subnet CIDR, like "192.168.0.1" for "192.168.0.0/24" CIDR
Sławek Kapłońskid98e27d2018-05-07 16:16:28 +0200409 if gateway is given as None then no gateway will be assigned
Federico Ressi0ddc93b2018-04-09 12:01:48 +0200410
411 :param cidr: CIDR of the subnet to create
412 It can be either None, a str or a netaddr.IPNetwork instance
413
414 :param mask_bits: CIDR prefix length
415 It can be either None or a numeric value.
416 If cidr parameter is given then mask_bits is used to determinate a
417 sequence of valid CIDR to use as generated.
418 Please see netaddr.IPNetwork.subnet method documentation[1]
419
420 :param ip_version: ip version of generated subnet CIDRs
421 It can be None, IP_VERSION_4 or IP_VERSION_6
422 It has to match given either given CIDR and gateway
423
424 :param ip_version: numeric value (either IP_VERSION_4 or IP_VERSION_6)
425 this value must match CIDR and gateway IP versions if any of them is
426 given
427
428 :param client: client to be used to connect to network service
429
Federico Ressi98f20ec2018-05-11 06:09:49 +0200430 :param reserve_cidr: if True then it reserves assigned CIDR to avoid
431 using the same CIDR for further subnets in the scope of the same
432 test case class
433
Federico Ressi0ddc93b2018-04-09 12:01:48 +0200434 :param **kwargs: optional parameters to be forwarded to wrapped method
435
436 [1] http://netaddr.readthedocs.io/en/latest/tutorial_01.html#supernets-and-subnets # noqa
437 """
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000438
439 # allow tests to use admin client
440 if not client:
441 client = cls.client
442
Federico Ressi0ddc93b2018-04-09 12:01:48 +0200443 if gateway:
444 gateway_ip = netaddr.IPAddress(gateway)
445 if ip_version:
446 if ip_version != gateway_ip.version:
447 raise ValueError(
448 "Gateway IP version doesn't match IP version")
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000449 else:
Federico Ressi0ddc93b2018-04-09 12:01:48 +0200450 ip_version = gateway_ip.version
Sławek Kapłońskid98e27d2018-05-07 16:16:28 +0200451 else:
452 ip_version = ip_version or cls._ip_version
Federico Ressi0ddc93b2018-04-09 12:01:48 +0200453
454 for subnet_cidr in cls.get_subnet_cidrs(
455 ip_version=ip_version, cidr=cidr, mask_bits=mask_bits):
Federico Ressi98f20ec2018-05-11 06:09:49 +0200456 if gateway is not None:
457 kwargs['gateway_ip'] = str(gateway or (subnet_cidr.ip + 1))
Slawek Kaplonski21f53422018-11-02 16:02:09 +0100458 else:
459 kwargs['gateway_ip'] = None
Federico Ressi98f20ec2018-05-11 06:09:49 +0200460 try:
461 body = client.create_subnet(
462 network_id=network['id'],
463 cidr=str(subnet_cidr),
464 ip_version=subnet_cidr.version,
465 **kwargs)
466 break
467 except lib_exc.BadRequest as e:
468 if 'overlaps with another subnet' not in str(e):
469 raise
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000470 else:
471 message = 'Available CIDR for subnet creation could not be found'
472 raise ValueError(message)
473 subnet = body['subnet']
Kevin Bentonba3651c2017-09-01 17:13:01 -0700474 if client is cls.client:
475 cls.subnets.append(subnet)
476 else:
477 cls.admin_subnets.append(subnet)
Federico Ressi98f20ec2018-05-11 06:09:49 +0200478 if reserve_cidr:
479 cls.reserve_subnet_cidr(subnet_cidr)
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000480 return subnet
481
482 @classmethod
Federico Ressi0ddc93b2018-04-09 12:01:48 +0200483 def reserve_subnet_cidr(cls, addr, **ipnetwork_kwargs):
484 """Reserve given subnet CIDR making sure it is not used by create_subnet
485
486 :param addr: the CIDR address to be reserved
487 It can be a str or netaddr.IPNetwork instance
488
489 :param **ipnetwork_kwargs: optional netaddr.IPNetwork constructor
490 parameters
491 """
492
493 if not cls.try_reserve_subnet_cidr(addr, **ipnetwork_kwargs):
Bernard Cafarellic3bec862020-09-10 13:59:49 +0200494 raise ValueError('Subnet CIDR already reserved: {0!r}'.format(
Federico Ressi0ddc93b2018-04-09 12:01:48 +0200495 addr))
496
497 @classmethod
498 def try_reserve_subnet_cidr(cls, addr, **ipnetwork_kwargs):
499 """Reserve given subnet CIDR if it hasn't been reserved before
500
501 :param addr: the CIDR address to be reserved
502 It can be a str or netaddr.IPNetwork instance
503
504 :param **ipnetwork_kwargs: optional netaddr.IPNetwork constructor
505 parameters
506
507 :return: True if it wasn't reserved before, False elsewhere.
508 """
509
510 subnet_cidr = netaddr.IPNetwork(addr, **ipnetwork_kwargs)
511 if subnet_cidr in cls.reserved_subnet_cidrs:
512 return False
513 else:
514 cls.reserved_subnet_cidrs.add(subnet_cidr)
515 return True
516
517 @classmethod
518 def get_subnet_cidrs(
519 cls, cidr=None, mask_bits=None, ip_version=None):
520 """Iterate over a sequence of unused subnet CIDR for IP version
521
522 :param cidr: CIDR of the subnet to create
523 It can be either None, a str or a netaddr.IPNetwork instance
524
525 :param mask_bits: CIDR prefix length
526 It can be either None or a numeric value.
527 If cidr parameter is given then mask_bits is used to determinate a
528 sequence of valid CIDR to use as generated.
529 Please see netaddr.IPNetwork.subnet method documentation[1]
530
531 :param ip_version: ip version of generated subnet CIDRs
532 It can be None, IP_VERSION_4 or IP_VERSION_6
533 It has to match given CIDR if given
534
535 :return: iterator over reserved CIDRs of type netaddr.IPNetwork
536
537 [1] http://netaddr.readthedocs.io/en/latest/tutorial_01.html#supernets-and-subnets # noqa
538 """
539
540 if cidr:
541 # Generate subnet CIDRs starting from given CIDR
542 # checking it is of requested IP version
543 cidr = netaddr.IPNetwork(cidr, version=ip_version)
544 else:
545 # Generate subnet CIDRs starting from configured values
546 ip_version = ip_version or cls._ip_version
547 if ip_version == const.IP_VERSION_4:
548 mask_bits = mask_bits or config.safe_get_config_value(
549 'network', 'project_network_mask_bits')
550 cidr = netaddr.IPNetwork(config.safe_get_config_value(
551 'network', 'project_network_cidr'))
552 elif ip_version == const.IP_VERSION_6:
553 mask_bits = config.safe_get_config_value(
554 'network', 'project_network_v6_mask_bits')
555 cidr = netaddr.IPNetwork(config.safe_get_config_value(
556 'network', 'project_network_v6_cidr'))
557 else:
558 raise ValueError('Invalid IP version: {!r}'.format(ip_version))
559
560 if mask_bits:
561 subnet_cidrs = cidr.subnet(mask_bits)
562 else:
563 subnet_cidrs = iter([cidr])
564
565 for subnet_cidr in subnet_cidrs:
566 if subnet_cidr not in cls.reserved_subnet_cidrs:
567 yield subnet_cidr
568
569 @classmethod
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000570 def create_port(cls, network, **kwargs):
571 """Wrapper utility that returns a test port."""
Edan Davidd75e48e2018-01-03 02:49:52 -0500572 if CONF.network.port_vnic_type and 'binding:vnic_type' not in kwargs:
573 kwargs['binding:vnic_type'] = CONF.network.port_vnic_type
Glenn Van de Water5d9b1402020-09-16 15:14:14 +0200574 if CONF.network.port_profile and 'binding:profile' not in kwargs:
575 kwargs['binding:profile'] = CONF.network.port_profile
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000576 body = cls.client.create_port(network_id=network['id'],
577 **kwargs)
578 port = body['port']
579 cls.ports.append(port)
580 return port
581
582 @classmethod
583 def update_port(cls, port, **kwargs):
584 """Wrapper utility that updates a test port."""
585 body = cls.client.update_port(port['id'],
586 **kwargs)
587 return body['port']
588
589 @classmethod
Genadi Chereshnyac0411e92016-07-11 16:59:42 +0300590 def _create_router_with_client(
591 cls, client, router_name=None, admin_state_up=False,
592 external_network_id=None, enable_snat=None, **kwargs
593 ):
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000594 ext_gw_info = {}
595 if external_network_id:
596 ext_gw_info['network_id'] = external_network_id
YAMAMOTO Takashi9bd4f972017-06-20 12:49:30 +0900597 if enable_snat is not None:
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000598 ext_gw_info['enable_snat'] = enable_snat
Genadi Chereshnyac0411e92016-07-11 16:59:42 +0300599 body = client.create_router(
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000600 router_name, external_gateway_info=ext_gw_info,
601 admin_state_up=admin_state_up, **kwargs)
602 router = body['router']
603 cls.routers.append(router)
604 return router
605
606 @classmethod
Genadi Chereshnyac0411e92016-07-11 16:59:42 +0300607 def create_router(cls, *args, **kwargs):
608 return cls._create_router_with_client(cls.client, *args, **kwargs)
609
610 @classmethod
611 def create_admin_router(cls, *args, **kwargs):
rajat294495c042017-06-28 15:37:16 +0530612 return cls._create_router_with_client(cls.os_admin.network_client,
Genadi Chereshnyac0411e92016-07-11 16:59:42 +0300613 *args, **kwargs)
614
615 @classmethod
Federico Ressia69dcd52018-07-06 09:45:34 +0200616 def create_floatingip(cls, external_network_id=None, port=None,
617 client=None, **kwargs):
618 """Creates a floating IP.
619
620 Create a floating IP and schedule it for later deletion.
621 If a client is passed, then it is used for deleting the IP too.
622
623 :param external_network_id: network ID where to create
624 By default this is 'CONF.network.public_network_id'.
625
626 :param port: port to bind floating IP to
627 This is translated to 'port_id=port['id']'
628 By default it is None.
629
630 :param client: network client to be used for creating and cleaning up
631 the floating IP.
632
633 :param **kwargs: additional creation parameters to be forwarded to
634 networking server.
635 """
636
637 client = client or cls.client
638 external_network_id = (external_network_id or
639 cls.external_network_id)
640
641 if port:
Federico Ressi47f6ae42018-09-24 16:19:14 +0200642 port_id = kwargs.setdefault('port_id', port['id'])
643 if port_id != port['id']:
644 message = "Port ID specified twice: {!s} != {!s}".format(
645 port_id, port['id'])
646 raise ValueError(message)
Federico Ressia69dcd52018-07-06 09:45:34 +0200647
648 fip = client.create_floatingip(external_network_id,
649 **kwargs)['floatingip']
650
651 # save client to be used later in cls.delete_floatingip
652 # for final cleanup
653 fip['client'] = client
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000654 cls.floating_ips.append(fip)
655 return fip
656
657 @classmethod
Federico Ressia69dcd52018-07-06 09:45:34 +0200658 def delete_floatingip(cls, floating_ip, client=None):
659 """Delete floating IP
660
661 :param client: Client to be used
662 If client is not given it will use the client used to create
663 the floating IP, or cls.client if unknown.
664 """
665
666 client = client or floating_ip.get('client') or cls.client
667 client.delete_floatingip(floating_ip['id'])
668
669 @classmethod
Slawek Kaplonski003fcae2019-05-26 22:38:35 +0200670 def create_port_forwarding(cls, fip_id, internal_port_id,
671 internal_port, external_port,
672 internal_ip_address=None, protocol="tcp",
673 client=None):
674 """Creates a port forwarding.
675
676 Create a port forwarding and schedule it for later deletion.
677 If a client is passed, then it is used for deleting the PF too.
678
679 :param fip_id: The ID of the floating IP address.
680
681 :param internal_port_id: The ID of the Neutron port associated to
682 the floating IP port forwarding.
683
684 :param internal_port: The TCP/UDP/other protocol port number of the
685 Neutron port fixed IP address associated to the floating ip
686 port forwarding.
687
688 :param external_port: The TCP/UDP/other protocol port number of
689 the port forwarding floating IP address.
690
691 :param internal_ip_address: The fixed IPv4 address of the Neutron
692 port associated to the floating IP port forwarding.
693
694 :param protocol: The IP protocol used in the floating IP port
695 forwarding.
696
697 :param client: network client to be used for creating and cleaning up
698 the floating IP port forwarding.
699 """
700
701 client = client or cls.client
702
703 pf = client.create_port_forwarding(
704 fip_id, internal_port_id, internal_port, external_port,
705 internal_ip_address, protocol)['port_forwarding']
706
707 # save ID of floating IP associated with port forwarding for final
708 # cleanup
709 pf['floatingip_id'] = fip_id
710
711 # save client to be used later in cls.delete_port_forwarding
712 # for final cleanup
713 pf['client'] = client
714 cls.port_forwardings.append(pf)
715 return pf
716
717 @classmethod
Flavio Fernandesa1952c62020-10-02 06:39:08 -0400718 def update_port_forwarding(cls, fip_id, pf_id, client=None, **kwargs):
719 """Wrapper utility for update_port_forwarding."""
720 client = client or cls.client
721 return client.update_port_forwarding(fip_id, pf_id, **kwargs)
722
723 @classmethod
Slawek Kaplonski003fcae2019-05-26 22:38:35 +0200724 def delete_port_forwarding(cls, pf, client=None):
725 """Delete port forwarding
726
727 :param client: Client to be used
728 If client is not given it will use the client used to create
729 the port forwarding, or cls.client if unknown.
730 """
731
732 client = client or pf.get('client') or cls.client
733 client.delete_port_forwarding(pf['floatingip_id'], pf['id'])
734
735 @classmethod
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000736 def create_router_interface(cls, router_id, subnet_id):
737 """Wrapper utility that returns a router interface."""
738 interface = cls.client.add_router_interface_with_subnet_id(
739 router_id, subnet_id)
740 return interface
741
742 @classmethod
Bence Romsics46bd3af2019-09-13 10:52:41 +0200743 def add_extra_routes_atomic(cls, *args, **kwargs):
744 return cls.client.add_extra_routes_atomic(*args, **kwargs)
745
746 @classmethod
747 def remove_extra_routes_atomic(cls, *args, **kwargs):
748 return cls.client.remove_extra_routes_atomic(*args, **kwargs)
749
750 @classmethod
Sławek Kapłońskiff294062016-12-04 15:00:54 +0000751 def get_supported_qos_rule_types(cls):
752 body = cls.client.list_qos_rule_types()
753 return [rule_type['type'] for rule_type in body['rule_types']]
754
755 @classmethod
Ihar Hrachyshkab7940d92016-06-10 13:44:22 +0200756 def create_qos_policy(cls, name, description=None, shared=False,
Rodolfo Alonso Hernandeze2d062f2020-01-14 17:11:42 +0000757 project_id=None, is_default=False):
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000758 """Wrapper utility that returns a test QoS policy."""
759 body = cls.admin_client.create_qos_policy(
Rodolfo Alonso Hernandeze2d062f2020-01-14 17:11:42 +0000760 name, description, shared, project_id, is_default)
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000761 qos_policy = body['policy']
762 cls.qos_policies.append(qos_policy)
763 return qos_policy
764
765 @classmethod
Sławek Kapłoński153f3452017-03-24 22:04:53 +0000766 def create_qos_bandwidth_limit_rule(cls, policy_id, max_kbps,
767 max_burst_kbps,
Chandan Kumarc125fd12017-11-15 19:41:01 +0530768 direction=const.EGRESS_DIRECTION):
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000769 """Wrapper utility that returns a test QoS bandwidth limit rule."""
770 body = cls.admin_client.create_bandwidth_limit_rule(
Sławek Kapłoński153f3452017-03-24 22:04:53 +0000771 policy_id, max_kbps, max_burst_kbps, direction)
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000772 qos_rule = body['bandwidth_limit_rule']
773 cls.qos_rules.append(qos_rule)
774 return qos_rule
775
776 @classmethod
Lajos Katona2f904652018-08-23 14:04:56 +0200777 def create_qos_minimum_bandwidth_rule(cls, policy_id, min_kbps,
778 direction=const.EGRESS_DIRECTION):
779 """Wrapper utility that creates and returns a QoS min bw rule."""
780 body = cls.admin_client.create_minimum_bandwidth_rule(
781 policy_id, direction, min_kbps)
782 qos_rule = body['minimum_bandwidth_rule']
783 cls.qos_rules.append(qos_rule)
784 return qos_rule
785
786 @classmethod
elajkatdbb0b482021-05-04 17:20:07 +0200787 def create_qos_dscp_marking_rule(cls, policy_id, dscp_mark):
788 """Wrapper utility that creates and returns a QoS dscp rule."""
789 body = cls.admin_client.create_dscp_marking_rule(
790 policy_id, dscp_mark)
791 qos_rule = body['dscp_marking_rule']
792 cls.qos_rules.append(qos_rule)
793 return qos_rule
794
795 @classmethod
Jakub Libosvar83704832017-12-06 16:02:28 +0000796 def delete_router(cls, router, client=None):
797 client = client or cls.client
Aditya Vaja49819a72018-11-26 14:20:10 -0800798 if 'routes' in router:
799 client.remove_router_extra_routes(router['id'])
Jakub Libosvar83704832017-12-06 16:02:28 +0000800 body = client.list_router_interfaces(router['id'])
Chandan Kumarc125fd12017-11-15 19:41:01 +0530801 interfaces = [port for port in body['ports']
802 if port['device_owner'] in const.ROUTER_INTERFACE_OWNERS]
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000803 for i in interfaces:
804 try:
Jakub Libosvar83704832017-12-06 16:02:28 +0000805 client.remove_router_interface_with_subnet_id(
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000806 router['id'], i['fixed_ips'][0]['subnet_id'])
807 except lib_exc.NotFound:
808 pass
Jakub Libosvar83704832017-12-06 16:02:28 +0000809 client.delete_router(router['id'])
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000810
811 @classmethod
812 def create_address_scope(cls, name, is_admin=False, **kwargs):
813 if is_admin:
814 body = cls.admin_client.create_address_scope(name=name, **kwargs)
815 cls.admin_address_scopes.append(body['address_scope'])
816 else:
817 body = cls.client.create_address_scope(name=name, **kwargs)
818 cls.address_scopes.append(body['address_scope'])
819 return body['address_scope']
820
821 @classmethod
Igor Malinovskiyb80f1d02020-03-06 13:39:52 +0200822 def create_subnetpool(cls, name, is_admin=False, client=None, **kwargs):
823 if client is None:
824 client = cls.admin_client if is_admin else cls.client
825
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000826 if is_admin:
Igor Malinovskiyb80f1d02020-03-06 13:39:52 +0200827 body = client.create_subnetpool(name, **kwargs)
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000828 cls.admin_subnetpools.append(body['subnetpool'])
829 else:
Igor Malinovskiyb80f1d02020-03-06 13:39:52 +0200830 body = client.create_subnetpool(name, **kwargs)
Daniel Mellado3c0aeab2016-01-29 11:30:25 +0000831 cls.subnetpools.append(body['subnetpool'])
832 return body['subnetpool']
833
Chandan Kumarc125fd12017-11-15 19:41:01 +0530834 @classmethod
Miguel Lavalleb1c7a3d2021-01-31 19:05:22 -0600835 def create_address_group(cls, name, is_admin=False, **kwargs):
836 if is_admin:
837 body = cls.admin_client.create_address_group(name=name, **kwargs)
838 cls.admin_address_groups.append(body['address_group'])
839 else:
840 body = cls.client.create_address_group(name=name, **kwargs)
841 cls.address_groups.append(body['address_group'])
842 return body['address_group']
843
844 @classmethod
Chandan Kumarc125fd12017-11-15 19:41:01 +0530845 def create_project(cls, name=None, description=None):
846 test_project = name or data_utils.rand_name('test_project_')
847 test_description = description or data_utils.rand_name('desc_')
848 project = cls.identity_admin_client.create_project(
849 name=test_project,
850 description=test_description)['project']
851 cls.projects.append(project)
Dongcan Ye2de722e2018-07-04 11:01:37 +0000852 # Create a project will create a default security group.
Dongcan Ye2de722e2018-07-04 11:01:37 +0000853 sgs_list = cls.admin_client.list_security_groups(
854 tenant_id=project['id'])['security_groups']
Federico Ressi4c590d72018-10-10 14:01:08 +0200855 for security_group in sgs_list:
856 # Make sure delete_security_group method will use
857 # the admin client for this group
858 security_group['client'] = cls.admin_client
859 cls.security_groups.append(security_group)
Chandan Kumarc125fd12017-11-15 19:41:01 +0530860 return project
861
862 @classmethod
Federico Ressi4c590d72018-10-10 14:01:08 +0200863 def create_security_group(cls, name=None, project=None, client=None,
864 **kwargs):
865 if project:
866 client = client or cls.admin_client
867 project_id = kwargs.setdefault('project_id', project['id'])
868 tenant_id = kwargs.setdefault('tenant_id', project['id'])
869 if project_id != project['id'] or tenant_id != project['id']:
870 raise ValueError('Project ID specified multiple times')
871 else:
872 client = client or cls.client
873
874 name = name or data_utils.rand_name(cls.__name__)
875 security_group = client.create_security_group(name=name, **kwargs)[
876 'security_group']
877 security_group['client'] = client
878 cls.security_groups.append(security_group)
879 return security_group
880
881 @classmethod
882 def delete_security_group(cls, security_group, client=None):
883 client = client or security_group.get('client') or cls.client
884 client.delete_security_group(security_group['id'])
885
886 @classmethod
887 def create_security_group_rule(cls, security_group=None, project=None,
888 client=None, ip_version=None, **kwargs):
889 if project:
890 client = client or cls.admin_client
891 project_id = kwargs.setdefault('project_id', project['id'])
892 tenant_id = kwargs.setdefault('tenant_id', project['id'])
893 if project_id != project['id'] or tenant_id != project['id']:
894 raise ValueError('Project ID specified multiple times')
895
896 if 'security_group_id' not in kwargs:
897 security_group = (security_group or
898 cls.get_security_group(client=client))
899
900 if security_group:
901 client = client or security_group.get('client')
902 security_group_id = kwargs.setdefault('security_group_id',
903 security_group['id'])
904 if security_group_id != security_group['id']:
905 raise ValueError('Security group ID specified multiple times.')
906
907 ip_version = ip_version or cls._ip_version
908 default_params = (
909 constants.DEFAULT_SECURITY_GROUP_RULE_PARAMS[ip_version])
Miguel Lavalleb1c7a3d2021-01-31 19:05:22 -0600910 if ('remote_address_group_id' in kwargs and 'remote_ip_prefix' in
911 default_params):
912 default_params.pop('remote_ip_prefix')
Federico Ressi4c590d72018-10-10 14:01:08 +0200913 for key, value in default_params.items():
914 kwargs.setdefault(key, value)
915
916 client = client or cls.client
917 return client.create_security_group_rule(**kwargs)[
918 'security_group_rule']
919
920 @classmethod
921 def get_security_group(cls, name='default', client=None):
922 client = client or cls.client
923 security_groups = client.list_security_groups()['security_groups']
924 for security_group in security_groups:
925 if security_group['name'] == name:
926 return security_group
927 raise ValueError("No such security group named {!r}".format(name))
Chandan Kumarc125fd12017-11-15 19:41:01 +0530928
Federico Ressiab286e42018-06-19 09:52:10 +0200929 @classmethod
930 def create_keypair(cls, client=None, name=None, **kwargs):
931 client = client or cls.os_primary.keypairs_client
932 name = name or data_utils.rand_name('keypair-test')
933 keypair = client.create_keypair(name=name, **kwargs)['keypair']
934
935 # save client for later cleanup
936 keypair['client'] = client
937 cls.keypairs.append(keypair)
938 return keypair
939
940 @classmethod
941 def delete_keypair(cls, keypair, client=None):
942 client = (client or keypair.get('client') or
943 cls.os_primary.keypairs_client)
944 client.delete_keypair(keypair_name=keypair['name'])
945
Federico Ressi82e83e32018-07-03 14:19:55 +0200946 @classmethod
947 def create_trunk(cls, port=None, subports=None, client=None, **kwargs):
948 """Create network trunk
949
950 :param port: dictionary containing parent port ID (port['id'])
951 :param client: client to be used for connecting to networking service
952 :param **kwargs: extra parameters to be forwarded to network service
953
954 :returns: dictionary containing created trunk details
955 """
956 client = client or cls.client
957
958 if port:
959 kwargs['port_id'] = port['id']
960
961 trunk = client.create_trunk(subports=subports, **kwargs)['trunk']
962 # Save client reference for later deletion
963 trunk['client'] = client
964 cls.trunks.append(trunk)
965 return trunk
966
967 @classmethod
Huifeng Le1c9f40b2018-11-07 01:14:21 +0800968 def delete_trunk(cls, trunk, client=None, detach_parent_port=True):
Federico Ressi82e83e32018-07-03 14:19:55 +0200969 """Delete network trunk
970
971 :param trunk: dictionary containing trunk ID (trunk['id'])
972
973 :param client: client to be used for connecting to networking service
974 """
975 client = client or trunk.get('client') or cls.client
976 trunk.update(client.show_trunk(trunk['id'])['trunk'])
977
978 if not trunk['admin_state_up']:
979 # Cannot touch trunk before admin_state_up is True
980 client.update_trunk(trunk['id'], admin_state_up=True)
981 if trunk['sub_ports']:
982 # Removes trunk ports before deleting it
983 cls._try_delete_resource(client.remove_subports, trunk['id'],
984 trunk['sub_ports'])
985
986 # we have to detach the interface from the server before
987 # the trunk can be deleted.
988 parent_port = {'id': trunk['port_id']}
989
990 def is_parent_port_detached():
991 parent_port.update(client.show_port(parent_port['id'])['port'])
992 return not parent_port['device_id']
993
Huifeng Le1c9f40b2018-11-07 01:14:21 +0800994 if detach_parent_port and not is_parent_port_detached():
Federico Ressi82e83e32018-07-03 14:19:55 +0200995 # this could probably happen when trunk is deleted and parent port
996 # has been assigned to a VM that is still running. Here we are
997 # assuming that device_id points to such VM.
998 cls.os_primary.compute.InterfacesClient().delete_interface(
999 parent_port['device_id'], parent_port['id'])
1000 utils.wait_until_true(is_parent_port_detached)
1001
1002 client.delete_trunk(trunk['id'])
1003
Harald Jensåsc9782fa2019-06-03 22:35:41 +02001004 @classmethod
1005 def create_conntrack_helper(cls, router_id, helper, protocol, port,
1006 client=None):
1007 """Create a conntrack helper
1008
1009 Create a conntrack helper and schedule it for later deletion. If a
1010 client is passed, then it is used for deleteing the CTH too.
1011
1012 :param router_id: The ID of the Neutron router associated to the
1013 conntrack helper.
1014
1015 :param helper: The conntrack helper module alias
1016
1017 :param protocol: The conntrack helper IP protocol used in the conntrack
1018 helper.
1019
1020 :param port: The conntrack helper IP protocol port number for the
1021 conntrack helper.
1022
1023 :param client: network client to be used for creating and cleaning up
1024 the conntrack helper.
1025 """
1026
1027 client = client or cls.client
1028
1029 cth = client.create_conntrack_helper(router_id, helper, protocol,
1030 port)['conntrack_helper']
1031
1032 # save ID of router associated with conntrack helper for final cleanup
1033 cth['router_id'] = router_id
1034
1035 # save client to be used later in cls.delete_conntrack_helper for final
1036 # cleanup
1037 cth['client'] = client
1038 cls.conntrack_helpers.append(cth)
1039 return cth
1040
1041 @classmethod
1042 def delete_conntrack_helper(cls, cth, client=None):
1043 """Delete conntrack helper
1044
1045 :param client: Client to be used
1046 If client is not given it will use the client used to create the
1047 conntrack helper, or cls.client if unknown.
1048 """
1049
1050 client = client or cth.get('client') or cls.client
1051 client.delete_conntrack_helper(cth['router_id'], cth['id'])
1052
Daniel Mellado3c0aeab2016-01-29 11:30:25 +00001053
1054class BaseAdminNetworkTest(BaseNetworkTest):
1055
1056 credentials = ['primary', 'admin']
1057
1058 @classmethod
1059 def setup_clients(cls):
1060 super(BaseAdminNetworkTest, cls).setup_clients()
fumihiko kakumaa216fc12017-07-14 10:43:29 +09001061 cls.admin_client = cls.os_admin.network_client
Jakub Libosvarf5758012017-08-15 13:45:30 +00001062 cls.identity_admin_client = cls.os_admin.projects_client
Daniel Mellado3c0aeab2016-01-29 11:30:25 +00001063
1064 @classmethod
1065 def create_metering_label(cls, name, description):
1066 """Wrapper utility that returns a test metering label."""
1067 body = cls.admin_client.create_metering_label(
1068 description=description,
1069 name=data_utils.rand_name("metering-label"))
1070 metering_label = body['metering_label']
1071 cls.metering_labels.append(metering_label)
1072 return metering_label
1073
1074 @classmethod
1075 def create_metering_label_rule(cls, remote_ip_prefix, direction,
1076 metering_label_id):
1077 """Wrapper utility that returns a test metering label rule."""
1078 body = cls.admin_client.create_metering_label_rule(
1079 remote_ip_prefix=remote_ip_prefix, direction=direction,
1080 metering_label_id=metering_label_id)
1081 metering_label_rule = body['metering_label_rule']
1082 cls.metering_label_rules.append(metering_label_rule)
1083 return metering_label_rule
1084
1085 @classmethod
Kailun Qineaaf9782018-12-20 04:45:01 +08001086 def create_network_segment_range(cls, name, shared,
1087 project_id, network_type,
1088 physical_network, minimum,
1089 maximum):
1090 """Wrapper utility that returns a test network segment range."""
1091 network_segment_range_args = {'name': name,
1092 'shared': shared,
1093 'project_id': project_id,
1094 'network_type': network_type,
1095 'physical_network': physical_network,
1096 'minimum': minimum,
1097 'maximum': maximum}
1098 body = cls.admin_client.create_network_segment_range(
1099 **network_segment_range_args)
1100 network_segment_range = body['network_segment_range']
1101 cls.network_segment_ranges.append(network_segment_range)
1102 return network_segment_range
1103
1104 @classmethod
Daniel Mellado3c0aeab2016-01-29 11:30:25 +00001105 def create_flavor(cls, name, description, service_type):
1106 """Wrapper utility that returns a test flavor."""
1107 body = cls.admin_client.create_flavor(
1108 description=description, service_type=service_type,
1109 name=name)
1110 flavor = body['flavor']
1111 cls.flavors.append(flavor)
1112 return flavor
1113
1114 @classmethod
1115 def create_service_profile(cls, description, metainfo, driver):
1116 """Wrapper utility that returns a test service profile."""
1117 body = cls.admin_client.create_service_profile(
1118 driver=driver, metainfo=metainfo, description=description)
1119 service_profile = body['service_profile']
1120 cls.service_profiles.append(service_profile)
1121 return service_profile
1122
1123 @classmethod
Nguyen Phuong An67993fc2017-11-24 11:30:25 +07001124 def create_log(cls, name, description=None,
1125 resource_type='security_group', resource_id=None,
1126 target_id=None, event='ALL', enabled=True):
1127 """Wrapper utility that returns a test log object."""
1128 log_args = {'name': name,
1129 'description': description,
1130 'resource_type': resource_type,
1131 'resource_id': resource_id,
1132 'target_id': target_id,
1133 'event': event,
1134 'enabled': enabled}
1135 body = cls.admin_client.create_log(**log_args)
1136 log_object = body['log']
1137 cls.log_objects.append(log_object)
1138 return log_object
1139
1140 @classmethod
Daniel Mellado3c0aeab2016-01-29 11:30:25 +00001141 def get_unused_ip(cls, net_id, ip_version=None):
Gary Kotton011345f2016-06-15 08:04:31 -07001142 """Get an unused ip address in a allocation pool of net"""
Daniel Mellado3c0aeab2016-01-29 11:30:25 +00001143 body = cls.admin_client.list_ports(network_id=net_id)
1144 ports = body['ports']
1145 used_ips = []
1146 for port in ports:
1147 used_ips.extend(
1148 [fixed_ip['ip_address'] for fixed_ip in port['fixed_ips']])
1149 body = cls.admin_client.list_subnets(network_id=net_id)
1150 subnets = body['subnets']
1151
1152 for subnet in subnets:
1153 if ip_version and subnet['ip_version'] != ip_version:
1154 continue
1155 cidr = subnet['cidr']
1156 allocation_pools = subnet['allocation_pools']
1157 iterators = []
1158 if allocation_pools:
1159 for allocation_pool in allocation_pools:
1160 iterators.append(netaddr.iter_iprange(
1161 allocation_pool['start'], allocation_pool['end']))
1162 else:
1163 net = netaddr.IPNetwork(cidr)
1164
1165 def _iterip():
1166 for ip in net:
1167 if ip not in (net.network, net.broadcast):
1168 yield ip
1169 iterators.append(iter(_iterip()))
1170
1171 for iterator in iterators:
1172 for ip in iterator:
1173 if str(ip) not in used_ips:
1174 return str(ip)
1175
1176 message = (
1177 "net(%s) has no usable IP address in allocation pools" % net_id)
1178 raise exceptions.InvalidConfiguration(message)
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001179
Lajos Katona2f904652018-08-23 14:04:56 +02001180 @classmethod
1181 def create_provider_network(cls, physnet_name, start_segmentation_id,
1182 max_attempts=30):
1183 segmentation_id = start_segmentation_id
Lajos Katona7eb67252019-01-14 12:55:35 +01001184 for attempts in range(max_attempts):
Lajos Katona2f904652018-08-23 14:04:56 +02001185 try:
Lajos Katona7eb67252019-01-14 12:55:35 +01001186 return cls.create_network(
Lajos Katona2f904652018-08-23 14:04:56 +02001187 name=data_utils.rand_name('test_net'),
1188 shared=True,
1189 provider_network_type='vlan',
1190 provider_physical_network=physnet_name,
1191 provider_segmentation_id=segmentation_id)
Lajos Katona2f904652018-08-23 14:04:56 +02001192 except lib_exc.Conflict:
Lajos Katona2f904652018-08-23 14:04:56 +02001193 segmentation_id += 1
1194 if segmentation_id > 4095:
1195 raise lib_exc.TempestException(
1196 "No free segmentation id was found for provider "
1197 "network creation!")
1198 time.sleep(CONF.network.build_interval)
Lajos Katona7eb67252019-01-14 12:55:35 +01001199 LOG.exception("Failed to create provider network after "
1200 "%d attempts", max_attempts)
1201 raise lib_exc.TimeoutException
Lajos Katona2f904652018-08-23 14:04:56 +02001202
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001203
Sławek Kapłońskiff294062016-12-04 15:00:54 +00001204def require_qos_rule_type(rule_type):
1205 def decorator(f):
1206 @functools.wraps(f)
1207 def wrapper(self, *func_args, **func_kwargs):
1208 if rule_type not in self.get_supported_qos_rule_types():
1209 raise self.skipException(
1210 "%s rule type is required." % rule_type)
1211 return f(self, *func_args, **func_kwargs)
1212 return wrapper
1213 return decorator
1214
1215
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001216def _require_sorting(f):
1217 @functools.wraps(f)
1218 def inner(self, *args, **kwargs):
Chandan Kumarc125fd12017-11-15 19:41:01 +05301219 if not tutils.is_extension_enabled("sorting", "network"):
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001220 self.skipTest('Sorting feature is required')
1221 return f(self, *args, **kwargs)
1222 return inner
1223
1224
1225def _require_pagination(f):
1226 @functools.wraps(f)
1227 def inner(self, *args, **kwargs):
Chandan Kumarc125fd12017-11-15 19:41:01 +05301228 if not tutils.is_extension_enabled("pagination", "network"):
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001229 self.skipTest('Pagination feature is required')
1230 return f(self, *args, **kwargs)
1231 return inner
1232
1233
1234class BaseSearchCriteriaTest(BaseNetworkTest):
1235
1236 # This should be defined by subclasses to reflect resource name to test
1237 resource = None
1238
Armando Migliaccio57581c62016-07-01 10:13:19 -07001239 field = 'name'
1240
Ihar Hrachyshkaa8fe5a12016-05-24 14:50:58 +02001241 # NOTE(ihrachys): some names, like those starting with an underscore (_)
1242 # are sorted differently depending on whether the plugin implements native
1243 # sorting support, or not. So we avoid any such cases here, sticking to
1244 # alphanumeric. Also test a case when there are multiple resources with the
1245 # same name
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001246 resource_names = ('test1', 'abc1', 'test10', '123test') + ('test1',)
1247
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001248 force_tenant_isolation = True
1249
Ihar Hrachyshkaa8fe5a12016-05-24 14:50:58 +02001250 list_kwargs = {}
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001251
Ihar Hrachyshkab7940d92016-06-10 13:44:22 +02001252 list_as_admin = False
1253
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001254 def assertSameOrder(self, original, actual):
1255 # gracefully handle iterators passed
1256 original = list(original)
1257 actual = list(actual)
1258 self.assertEqual(len(original), len(actual))
1259 for expected, res in zip(original, actual):
Armando Migliaccio57581c62016-07-01 10:13:19 -07001260 self.assertEqual(expected[self.field], res[self.field])
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001261
1262 @utils.classproperty
1263 def plural_name(self):
1264 return '%ss' % self.resource
1265
Ihar Hrachyshkab7940d92016-06-10 13:44:22 +02001266 @property
1267 def list_client(self):
1268 return self.admin_client if self.list_as_admin else self.client
1269
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001270 def list_method(self, *args, **kwargs):
Ihar Hrachyshkab7940d92016-06-10 13:44:22 +02001271 method = getattr(self.list_client, 'list_%s' % self.plural_name)
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001272 kwargs.update(self.list_kwargs)
1273 return method(*args, **kwargs)
1274
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001275 def get_bare_url(self, url):
1276 base_url = self.client.base_url
zheng.yong74e760a2019-05-22 14:16:14 +08001277 base_url_normalized = utils.normalize_url(base_url)
1278 url_normalized = utils.normalize_url(url)
1279 self.assertTrue(url_normalized.startswith(base_url_normalized))
1280 return url_normalized[len(base_url_normalized):]
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001281
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001282 @classmethod
1283 def _extract_resources(cls, body):
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001284 return body[cls.plural_name]
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001285
1286 def _test_list_sorts(self, direction):
1287 sort_args = {
1288 'sort_dir': direction,
Armando Migliaccio57581c62016-07-01 10:13:19 -07001289 'sort_key': self.field
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001290 }
1291 body = self.list_method(**sort_args)
1292 resources = self._extract_resources(body)
1293 self.assertNotEmpty(
1294 resources, "%s list returned is empty" % self.resource)
Armando Migliaccio57581c62016-07-01 10:13:19 -07001295 retrieved_names = [res[self.field] for res in resources]
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001296 expected = sorted(retrieved_names)
1297 if direction == constants.SORT_DIRECTION_DESC:
1298 expected = list(reversed(expected))
1299 self.assertEqual(expected, retrieved_names)
1300
1301 @_require_sorting
1302 def _test_list_sorts_asc(self):
1303 self._test_list_sorts(constants.SORT_DIRECTION_ASC)
1304
1305 @_require_sorting
1306 def _test_list_sorts_desc(self):
1307 self._test_list_sorts(constants.SORT_DIRECTION_DESC)
1308
1309 @_require_pagination
1310 def _test_list_pagination(self):
1311 for limit in range(1, len(self.resource_names) + 1):
1312 pagination_args = {
1313 'limit': limit,
1314 }
1315 body = self.list_method(**pagination_args)
1316 resources = self._extract_resources(body)
1317 self.assertEqual(limit, len(resources))
1318
1319 @_require_pagination
1320 def _test_list_no_pagination_limit_0(self):
1321 pagination_args = {
1322 'limit': 0,
1323 }
1324 body = self.list_method(**pagination_args)
1325 resources = self._extract_resources(body)
Béla Vancsicsf1806182016-08-23 07:36:18 +02001326 self.assertGreaterEqual(len(resources), len(self.resource_names))
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001327
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001328 def _test_list_pagination_iteratively(self, lister):
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001329 # first, collect all resources for later comparison
1330 sort_args = {
1331 'sort_dir': constants.SORT_DIRECTION_ASC,
Armando Migliaccio57581c62016-07-01 10:13:19 -07001332 'sort_key': self.field
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001333 }
1334 body = self.list_method(**sort_args)
1335 expected_resources = self._extract_resources(body)
1336 self.assertNotEmpty(expected_resources)
1337
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001338 resources = lister(
1339 len(expected_resources), sort_args
1340 )
1341
1342 # finally, compare that the list retrieved in one go is identical to
1343 # the one containing pagination results
1344 self.assertSameOrder(expected_resources, resources)
1345
1346 def _list_all_with_marker(self, niterations, sort_args):
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001347 # paginate resources one by one, using last fetched resource as a
1348 # marker
1349 resources = []
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001350 for i in range(niterations):
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001351 pagination_args = sort_args.copy()
1352 pagination_args['limit'] = 1
1353 if resources:
1354 pagination_args['marker'] = resources[-1]['id']
1355 body = self.list_method(**pagination_args)
1356 resources_ = self._extract_resources(body)
1357 self.assertEqual(1, len(resources_))
1358 resources.extend(resources_)
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001359 return resources
Ihar Hrachyshka59382252016-04-05 15:54:33 +02001360
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001361 @_require_pagination
1362 @_require_sorting
1363 def _test_list_pagination_with_marker(self):
1364 self._test_list_pagination_iteratively(self._list_all_with_marker)
1365
1366 def _list_all_with_hrefs(self, niterations, sort_args):
1367 # paginate resources one by one, using next href links
1368 resources = []
1369 prev_links = {}
1370
1371 for i in range(niterations):
1372 if prev_links:
1373 uri = self.get_bare_url(prev_links['next'])
1374 else:
Ihar Hrachyshka7f79fe62016-06-07 21:23:44 +02001375 sort_args.update(self.list_kwargs)
Ihar Hrachyshkab7940d92016-06-10 13:44:22 +02001376 uri = self.list_client.build_uri(
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001377 self.plural_name, limit=1, **sort_args)
Ihar Hrachyshkab7940d92016-06-10 13:44:22 +02001378 prev_links, body = self.list_client.get_uri_with_links(
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001379 self.plural_name, uri
1380 )
1381 resources_ = self._extract_resources(body)
1382 self.assertEqual(1, len(resources_))
1383 resources.extend(resources_)
1384
1385 # The last element is empty and does not contain 'next' link
1386 uri = self.get_bare_url(prev_links['next'])
1387 prev_links, body = self.client.get_uri_with_links(
1388 self.plural_name, uri
1389 )
1390 self.assertNotIn('next', prev_links)
1391
1392 # Now walk backwards and compare results
1393 resources2 = []
1394 for i in range(niterations):
1395 uri = self.get_bare_url(prev_links['previous'])
Ihar Hrachyshkab7940d92016-06-10 13:44:22 +02001396 prev_links, body = self.list_client.get_uri_with_links(
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001397 self.plural_name, uri
1398 )
1399 resources_ = self._extract_resources(body)
1400 self.assertEqual(1, len(resources_))
1401 resources2.extend(resources_)
1402
1403 self.assertSameOrder(resources, reversed(resources2))
1404
1405 return resources
1406
1407 @_require_pagination
1408 @_require_sorting
1409 def _test_list_pagination_with_href_links(self):
1410 self._test_list_pagination_iteratively(self._list_all_with_hrefs)
1411
1412 @_require_pagination
1413 @_require_sorting
1414 def _test_list_pagination_page_reverse_with_href_links(
1415 self, direction=constants.SORT_DIRECTION_ASC):
1416 pagination_args = {
1417 'sort_dir': direction,
Armando Migliaccio57581c62016-07-01 10:13:19 -07001418 'sort_key': self.field,
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001419 }
1420 body = self.list_method(**pagination_args)
1421 expected_resources = self._extract_resources(body)
1422
1423 page_size = 2
1424 pagination_args['limit'] = page_size
1425
1426 prev_links = {}
1427 resources = []
1428 num_resources = len(expected_resources)
1429 niterations = int(math.ceil(float(num_resources) / page_size))
1430 for i in range(niterations):
1431 if prev_links:
1432 uri = self.get_bare_url(prev_links['previous'])
1433 else:
Ihar Hrachyshka7f79fe62016-06-07 21:23:44 +02001434 pagination_args.update(self.list_kwargs)
Ihar Hrachyshkab7940d92016-06-10 13:44:22 +02001435 uri = self.list_client.build_uri(
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001436 self.plural_name, page_reverse=True, **pagination_args)
Ihar Hrachyshkab7940d92016-06-10 13:44:22 +02001437 prev_links, body = self.list_client.get_uri_with_links(
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001438 self.plural_name, uri
1439 )
1440 resources_ = self._extract_resources(body)
Béla Vancsicsf1806182016-08-23 07:36:18 +02001441 self.assertGreaterEqual(page_size, len(resources_))
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001442 resources.extend(reversed(resources_))
1443
1444 self.assertSameOrder(expected_resources, reversed(resources))
1445
1446 @_require_pagination
1447 @_require_sorting
1448 def _test_list_pagination_page_reverse_asc(self):
1449 self._test_list_pagination_page_reverse(
1450 direction=constants.SORT_DIRECTION_ASC)
1451
1452 @_require_pagination
1453 @_require_sorting
1454 def _test_list_pagination_page_reverse_desc(self):
1455 self._test_list_pagination_page_reverse(
1456 direction=constants.SORT_DIRECTION_DESC)
1457
1458 def _test_list_pagination_page_reverse(self, direction):
1459 pagination_args = {
1460 'sort_dir': direction,
Armando Migliaccio57581c62016-07-01 10:13:19 -07001461 'sort_key': self.field,
Ihar Hrachyshkaaeb03a02016-05-18 20:03:18 +02001462 'limit': 3,
1463 }
1464 body = self.list_method(**pagination_args)
1465 expected_resources = self._extract_resources(body)
1466
1467 pagination_args['limit'] -= 1
1468 pagination_args['marker'] = expected_resources[-1]['id']
1469 pagination_args['page_reverse'] = True
1470 body = self.list_method(**pagination_args)
1471
1472 self.assertSameOrder(
1473 # the last entry is not included in 2nd result when used as a
1474 # marker
1475 expected_resources[:-1],
1476 self._extract_resources(body))
Victor Morales1be97b42016-09-05 08:50:06 -05001477
Hongbin Lu54f55922018-07-12 19:05:39 +00001478 @tutils.requires_ext(extension="filter-validation", service="network")
1479 def _test_list_validation_filters(
1480 self, validation_args, filter_is_valid=True):
1481 if not filter_is_valid:
1482 self.assertRaises(lib_exc.BadRequest, self.list_method,
1483 **validation_args)
1484 else:
1485 body = self.list_method(**validation_args)
1486 resources = self._extract_resources(body)
1487 for resource in resources:
1488 self.assertIn(resource['name'], self.resource_names)