From 2e5ffd27df0f8db087e53be9517fed1b5fefcab0 Mon Sep 17 00:00:00 2001 From: Alex Kang Date: Thu, 25 Aug 2016 13:12:43 -0700 Subject: [PATCH] Tempest: router_size create and update tests 1. Move router size tests from test_router_types.py to test_router_sizes.py 2. Add "change router_size" tests Begins with Mitaka, nsxv can change router's size. 3. Use NSX-v backend to validate router's size is changed correctly. 4. Add CONF.nsxv.create_router_http_timeout to set the http_timeout for router client. 5. Due to bug#1716695 - keep creating routers. This fix creates its own router client to change the retry time from default 60 to CONF.nsxv.create_router_http_timeout. Plese refer to docstr of class RouterSizeBaseTest for detail. Change-Id: Iaf07f78db898c1bf94dda5d716eae9ef6e0bcc2c --- vmware_nsx_tempest/config.py | 4 + .../tests/nsxv/api/test_router_sizes.py | 205 ++++++++++++++++++ .../tests/nsxv/api/test_router_types.py | 62 +----- 3 files changed, 211 insertions(+), 60 deletions(-) create mode 100644 vmware_nsx_tempest/tests/nsxv/api/test_router_sizes.py diff --git a/vmware_nsx_tempest/config.py b/vmware_nsx_tempest/config.py index bda0cac2cc..6f475efd78 100644 --- a/vmware_nsx_tempest/config.py +++ b/vmware_nsx_tempest/config.py @@ -105,6 +105,10 @@ NSXvGroup = [ cfg.IntOpt('provider_vlan_id', default=888, help="The default vlan_id for admin vlan."), + cfg.IntOpt('create_router_http_timeout', + default=900, + help="Specific for router_size tests. This value defines" + " how long http.request should retry."), cfg.BoolOpt('no_router_type', default=False, help="router_type is NSXv extension." diff --git a/vmware_nsx_tempest/tests/nsxv/api/test_router_sizes.py b/vmware_nsx_tempest/tests/nsxv/api/test_router_sizes.py new file mode 100644 index 0000000000..2f12aa1f2f --- /dev/null +++ b/vmware_nsx_tempest/tests/nsxv/api/test_router_sizes.py @@ -0,0 +1,205 @@ +# Copyright 2016 VMware Inc +# All Rights Reserved +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +import re +import time + +from oslo_log import log as logging + +from tempest.api.network import base_routers as base +from tempest import config +from tempest.lib.common.utils import data_utils +from tempest.lib.services import network as net_clients +from tempest import test +from vmware_nsx_tempest.services import nsxv_client + +CONF = config.CONF +LOG = logging.getLogger(__name__) +ROUTER_SIZE = ('compact', 'large', 'xlarge', 'quadlarge') + + +class RouterSizeBaseTest(base.BaseRouterTest): + """Base class to test creating routers with different router sizes: + + NSX-v allows exclusive router to be created with one of ROUTER_SIZE. + Starts with VIO-3.0 it can update its router_size after created. + + tempest internally uses urllib3 and by default it will retry very 60 + seconds. However this retry mechanism causes bug#1716696. + + A better solution is to change request's retry-time so it will not + cause neutront keep creating routers while router was not created + in time. + + Methods should be used to change retry-time are: + + create_exclusive_router & change_router_size + + The retry-time is http_timeout in request.__init__() and is + defined by CONF.nsxv.create_router_http_timeout. + """ + + @classmethod + def skip_checks(cls): + super(RouterSizeBaseTest, cls).skip_checks() + if not test.is_extension_enabled('nsxv-router-type', 'network'): + msg = "router-type extension is not enabled" + raise cls.skipException(msg) + + @classmethod + def setup_clients(cls): + super(RouterSizeBaseTest, cls).setup_clients() + + @classmethod + def resource_setup(cls): + super(RouterSizeBaseTest, cls).resource_setup() + cls.tenant_cidr = (CONF.network.project_network_cidr + if cls._ip_version == 4 else + CONF.network.project_network_v6_cidr) + manager_ip = re.search(r"(\d{1,3}\.){3}\d{1,3}", + CONF.nsxv.manager_uri).group(0) + cls.vsm = nsxv_client.VSMClient( + manager_ip, CONF.nsxv.user, CONF.nsxv.password) + + def setUp(self): + super(RouterSizeBaseTest, self).setUp() + params = {'build_interval': self.routers_client.build_interval, + 'build_timeout': self.routers_client.build_timeout} + http_timeout = CONF.nsxv.create_router_http_timeout + self.router_sizes_client = net_clients.RoutersClient( + self.routers_client.auth_provider, + self.routers_client.service, + self.routers_client.region, + self.routers_client.endpoint_type, + http_timeout=http_timeout, + **params) + + def create_exclusive_router(self, router_size): + name = data_utils.rand_name('rtr1-%s' % router_size) + LOG.debug("create router with size=%s", router_size) + ext_gw_info = dict( + network_id=CONF.network.public_network_id) + rtr_cfg = dict( + name=name, admin_state_up=False, + external_gateway_info=ext_gw_info, + router_type='exclusive', + router_size=router_size) + router = self.router_sizes_client.create_router(**rtr_cfg) + router = router.get('router', router) + self.routers.append(router) + self.assertEqual(router['name'], name) + self.check_router_nsx_name(router, router_size) + return router + + def change_router_size(self, router, new_router_size): + LOG.debug("update router to size=%s", new_router_size) + update_router = self.router_sizes_client.update_router( + router['id'], router_size=new_router_size)['router'] + self.assertEqual(update_router['router_size'], new_router_size) + self.check_router_nsx_name(update_router, new_router_size) + return router + + def check_router_nsx_name(self, router, router_size=None): + router_nsxv_name = self.get_router_nsx_name(router) + exc_edge = self.vsm.get_edge(router_nsxv_name) + self.assertTrue(exc_edge is not None) + self.assertEqual(exc_edge['edgeType'], 'gatewayServices') + if router_size: + edge_type = exc_edge['appliancesSummary']['applianceSize'] + LOG.debug("check router size at backend is %s", router_size) + self.assertEqual(edge_type, router_size) + return router_nsxv_name + + def get_router_nsx_name(self, router): + router_nsxv_name = '%s-%s' % (router['name'], router['id']) + return router_nsxv_name + + def do_create_update_delete_router_with_size(self, + router_size, + del_waitfor=10.0, + del_interval=1.5): + router = self.create_exclusive_router(router_size) + updated_name = 'updated-' + router['name'] + update_router = self.router_sizes_client.update_router( + router['id'], name=updated_name)['router'] + self.assertEqual(update_router['name'], updated_name) + # change router name, the backend also change + router = self.router_sizes_client.show_router( + router['id'])['router'] + nsxv_edge_name = self.check_router_nsx_name(router, router_size) + # Delete the exclusive router and verify it has been deleted + # from nsxv backend + self.router_sizes_client.delete_router(router['id']) + list_body = self.router_sizes_client.list_routers() + routers_list = [r['id'] for r in list_body['routers']] + self.assertNotIn(router['id'], routers_list) + wait_till = time.time() + del_waitfor + while (time.time() < wait_till): + try: + self.assertEqual(self.vsm.get_edge(nsxv_edge_name), None) + return + except Exception: + time.sleep(del_interval) + # last try. Fail if nesx_edge still exists + fail_msg = ("%s router nsxv_edge[%s] still exists after %s seconds." % + (router_size, nsxv_edge_name, del_waitfor)) + self.assertEqual(self.vsm.get_edge(nsxv_edge_name), None, fail_msg) + + def do_router_size_change_test(self, router_size, new_router_size_list): + router = self.create_exclusive_router(router_size) + for new_router_size in new_router_size_list: + self.change_router_size(router, new_router_size) + + +class CompactRouterTest(RouterSizeBaseTest): + @test.attr(type='nsxv') + @test.idempotent_id('d75fbcd5-c8cb-49ea-a868-ada12fd8c87f') + def test_create_update_delete_compact_router(self): + self.do_create_update_delete_router_with_size('compact') + + +class LargeRouterTest(RouterSizeBaseTest): + @test.attr(type='nsxv') + @test.idempotent_id('da00c74f-81e6-4ef9-8aca-8e0345b376e9') + def test_create_update_delete_large_router(self): + self.do_create_update_delete_router_with_size('large', 20.0) + + +class XlargeRouterTest(RouterSizeBaseTest): + @test.attr(type='nsxv') + @test.idempotent_id('091dad07-6044-4ca3-b16c-54a3ef92254b') + def test_create_update_delete_xlarge_router(self): + self.do_create_update_delete_router_with_size('xlarge', 20.0) + + +class QuadlargeRouterTest(RouterSizeBaseTest): + @test.attr(type='nsxv') + @test.idempotent_id('0f69bf8a-4b06-47ac-a3f7-eedba95fd395') + def test_create_update_delete_quadlarge_router(self): + self.do_create_update_delete_router_with_size('quadlarge', 30.0) + + +class RouterSizeChangeTest(RouterSizeBaseTest): + @test.idempotent_id('3201b0a9-702c-46cf-8512-f166a6ea5109') + def test_router_size_1sizeup_change(self): + self.do_router_size_change_test( + 'compact', + ('large', 'xlarge', 'quadlarge')) + + @test.idempotent_id('c7ee9f78-4938-4bdd-b39c-1d736d41a84b') + def test_router_size_outofseq_change(self): + self.do_router_size_change_test( + "large", + ('quadlarge', 'compact', 'xlarge', 'large')) diff --git a/vmware_nsx_tempest/tests/nsxv/api/test_router_types.py b/vmware_nsx_tempest/tests/nsxv/api/test_router_types.py index cf67a528b4..3686af3b80 100644 --- a/vmware_nsx_tempest/tests/nsxv/api/test_router_types.py +++ b/vmware_nsx_tempest/tests/nsxv/api/test_router_types.py @@ -14,23 +14,20 @@ # under the License. import re -import time - -from tempest.lib.common.utils import data_utils from tempest.api.network import base_routers as base from tempest import config +from tempest.lib.common.utils import data_utils from tempest import test from vmware_nsx_tempest.services import nsxv_client CONF = config.CONF -ROUTER_SIZE = ('compact', 'large', 'xlarge', 'quadlarge') class ExcRouterTest(base.BaseRouterTest): """ Test class for exclusive router type, which is 1:1 mapping of - NSX-v service edge. Tests will sipped if the router-type + NSX-v service edge. Tests will skipped if the router-type extension is not enabled. """ @@ -141,58 +138,3 @@ class ExcRouterTest(base.BaseRouterTest): self.assertNotIn(router['router']['id'], routers_list) nsxv_edge_name = "%s-%s" % (name, router['router']['id']) self.assertEqual(self.vsm.get_edge(nsxv_edge_name), None) - - @test.attr(type='nsxv') - @test.idempotent_id('d75fbcd5-c8cb-49ea-a868-ada12fd8c87f') - def test_create_update_delete_compact_router(self): - self.do_create_update_delete_router_with_size('compact') - - @test.attr(type='nsxv') - @test.idempotent_id('da00c74f-81e6-4ef9-8aca-8e0345b376e9') - def test_create_update_delete_large_router(self): - self.do_create_update_delete_router_with_size('large', 20.0) - - @test.attr(type='nsxv') - @test.idempotent_id('091dad07-6044-4ca3-b16c-54a3ef92254b') - def test_create_update_delete_xlarge_router(self): - self.do_create_update_delete_router_with_size('xlarge', 20.0) - - @test.attr(type='nsxv') - @test.idempotent_id('0f69bf8a-4b06-47ac-a3f7-eedba95fd395') - def test_create_update_delete_quadlarge_router(self): - self.do_create_update_delete_router_with_size('quadlarge', 30.0) - - def do_create_update_delete_router_with_size(self, - router_size, - del_waitfor=10.0, - del_interval=1.5): - name = data_utils.rand_name('rtr-%s' % router_size) - router = self.routers_client.create_router( - name=name, external_gateway_info={ - "network_id": CONF.network.public_network_id}, - admin_state_up=False, router_type='exclusive', - router_size=router_size) - self.assertEqual(router['router']['name'], name) - # Update the name of the exclusive router - updated_name = 'updated' + name - update_body = self.routers_client.update_router( - router['router']['id'], name=updated_name) - self.assertEqual(update_body['router']['name'], updated_name) - # Delete the exclusive router and verify it has been deleted - # from nsxv backend - self.routers_client.delete_router(router['router']['id']) - list_body = self.routers_client.list_routers() - routers_list = [r['id'] for r in list_body['routers']] - self.assertNotIn(router['router']['id'], routers_list) - nsxv_edge_name = "%s-%s" % (name, router['router']['id']) - wait_till = time.time() + del_waitfor - while (time.time() < wait_till): - try: - self.assertEqual(self.vsm.get_edge(nsxv_edge_name), None) - return - except Exception: - time.sleep(del_interval) - # last try. Fail if nesx_edge still exists - fail_msg = ("%s router nsxv_edge[%s] still exists after %s seconds." % - (router_size, nsxv_edge_name, del_waitfor)) - self.assertEqual(self.vsm.get_edge(nsxv_edge_name), None, fail_msg)