tempest: lbaas l7 switching scenario tests

1. add lbaas l7 switching clients
2. enhance to delete l7 resources if present
3. enhance NSX-v scenario to support l7 switching testing
   3.1 check floatingip assiged to server before ping test
       NSX-v3 can take 500+ second to assign floatingip to VM
   3.2 enhance test_lbaas_round_robin to support l7 test
   3.3 test_lbaas_l7_switching_ops only test URL starts_with
       validation.
4. l7_switching_ops requires around 2000 seconds. When run test
   set OS_TEST_TIMEOUT=2400 to avoid fixture timeout.
5. add CONF.nsxv.bugs_to_resolve to skip tests that can not be
   run at sepcific NSX plugin environment.
6. doc/README-LBaaS.rst describes limitation and how to run
   tests at different plugin's and backends.

Change-Id: Ib2ee4ce57d45882d76e25ce7b7bba3d825bf34ab
This commit is contained in:
Alex Kang 2016-10-12 11:15:11 +00:00
parent 455bb910d2
commit 2065418288
17 changed files with 647 additions and 136 deletions

View File

@ -114,6 +114,11 @@ NSXvGroup = [
help="router_type is NSXv extension." help="router_type is NSXv extension."
"Set it to True allow tests to remove this attribute" "Set it to True allow tests to remove this attribute"
" when creating router."), " when creating router."),
cfg.ListOpt('bugs_to_resolve',
default=[],
help="Bugs to be resolved. Define this at tempest.conf and"
" test case testtools.skipIf(condition, reasons) to"
" skip test cannot be run at specific plugin env."),
] ]

View File

@ -0,0 +1,74 @@
Overview
========
This document describes what LBaaS tests are not supported at different
NSX plugin's and backends.
NOTE::
All LBaaS API & Scenario tests should PASS with exceptions
due to NSX plugins and features supported by backend.
For how tests can be skipped for specific plugin and backend,
please refer to paragraph "Config for Test Execution".
NOTE::
We no longer support LBaaS v1. So this document and LBaaS tests
only applys to releases from Mitaka/Marvin or later.
Limitation:
-----------
NSX-v with VMware LBaaS driver::
#. LBaaS networks need to attach to exclusive router
#. One tenant per subnet
#. L7 switching not supported
NSX-v3 with Octavia driver::
#. upstream implemenation - all tests should PASS.
#. scenario tests take long time, it might fail with fixture timeout.
Config for Test execution:
--------------------------
Following configuration attributes used to controll test execution::
#. no_router_type at group/session nsxv
Default is False, and is used to run LBaaS tests in NSX-v environment.
To run in NSX-t environment, set it to True
#. bugs_to_resolve at group/session nsxv
For test to skip if bug-ID presented in this attribute.
The test will use testtools.skipIf(condition, reason) to skip if its ID in the bugs_to_resolve list.
local.conf:
----------
NSX-v::
[nsxv]
no_router_type=False
bugs_to_resolve=1641902,1715126,1703396,1739510
NSX-v3::
[compute]
build_timeout=900
build_interval=2
[nsxv]
no_router_type=True
Execution:
----------
#. Use testr list-tests command to generate test suite for run API and Scenario tests::
tools/with_venv.sh testr list-tests nsxv.api.lbaas
tools/with_venv.sh testr list-tests nsxv.scenarion.test_lbaas
#. l7 switching tests take long time to complete. If got fixture timeout, do::
OS_TEST_TIMEOUT=2400 ./run_tempest.sh -t test_lbaas_l7_switching_ops

View File

@ -0,0 +1,58 @@
# 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.
from tempest.lib.services.network import base
class L7PoliciesClient(base.BaseNetworkClient):
resource = 'l7policy'
resource_plural = 'l7policies'
resource_base_path = '/lbaas/l7policies'
resource_object_path = '/lbaas/l7policies/%s'
def create_l7policy(self, **kwargs):
uri = self.resource_base_path
post_data = {self.resource: kwargs}
return self.create_resource(uri, post_data)
def update_l7policy(self, policy_id, **kwargs):
uri = self.resource_object_path % (policy_id)
post_data = {self.resource: kwargs}
return self.update_resource(uri, post_data)
def show_l7policy(self, policy_id, **fields):
uri = self.resource_object_path % (policy_id)
return self.show_resource(uri, **fields)
def delete_l7policy(self, policy_id):
uri = self.resource_object_path % (policy_id)
return self.delete_resource(uri)
def list_l7policies(self, **filters):
uri = self.resource_base_path
return self.list_resources(uri, **filters)
def get_client(client_mgr):
"""create a lbaas l7policies client from manager or networks_client"""
manager = getattr(client_mgr, 'manager', client_mgr)
net_client = getattr(manager, 'networks_client')
try:
_params = manager.default_params_with_timeout_values.copy()
except Exception:
_params = {}
client = L7PoliciesClient(net_client.auth_provider,
net_client.service,
net_client.region,
net_client.endpoint_type,
**_params)
return client

View File

@ -0,0 +1,58 @@
# 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.
from tempest.lib.services.network import base
class L7RulesClient(base.BaseNetworkClient):
resource = 'rule'
resource_plural = 'rules'
resource_base_path = '/lbaas/l7policies/%s/rules'
resource_object_path = '/lbaas/l7policies/%s/rules/%s'
def create_l7rule(self, policy_id, **kwargs):
uri = self.resource_base_path % policy_id
post_data = {self.resource: kwargs}
return self.create_resource(uri, post_data)
def update_l7rule(self, policy_id, rule_id, **kwargs):
uri = self.resource_object_path % (policy_id, rule_id)
post_data = {self.resource: kwargs}
return self.update_resource(uri, post_data)
def show_l7rule(self, policy_id, rule_id, **fields):
uri = self.resource_object_path % (policy_id, rule_id)
return self.show_resource(uri, **fields)
def delete_l7rule(self, policy_id, rule_id):
uri = self.resource_object_path % (policy_id, rule_id)
return self.delete_resource(uri)
def list_l7rules(self, policy_id, **filters):
uri = self.resource_base_path % policy_id
return self.list_resources(uri, **filters)
def get_client(client_mgr):
"""create a lbaas l7rules client from manager or networks_client"""
manager = getattr(client_mgr, 'manager', client_mgr)
net_client = getattr(manager, 'networks_client')
try:
_params = manager.default_params_with_timeout_values.copy()
except Exception:
_params = {}
client = L7RulesClient(net_client.auth_provider,
net_client.service,
net_client.region,
net_client.endpoint_type,
**_params)
return client

View File

@ -28,6 +28,8 @@ from tempest.lib import exceptions
from vmware_nsx_tempest._i18n import _LI from vmware_nsx_tempest._i18n import _LI
from vmware_nsx_tempest.services.lbaas import health_monitors_client from vmware_nsx_tempest.services.lbaas import health_monitors_client
from vmware_nsx_tempest.services.lbaas import l7policies_client
from vmware_nsx_tempest.services.lbaas import l7rules_client
from vmware_nsx_tempest.services.lbaas import listeners_client from vmware_nsx_tempest.services.lbaas import listeners_client
from vmware_nsx_tempest.services.lbaas import load_balancers_client from vmware_nsx_tempest.services.lbaas import load_balancers_client
from vmware_nsx_tempest.services.lbaas import members_client from vmware_nsx_tempest.services.lbaas import members_client
@ -68,6 +70,9 @@ class BaseTestCase(base.BaseNetworkTest):
cls.pools_client = pools_client.get_client(mgr) cls.pools_client = pools_client.get_client(mgr)
cls.members_client = members_client.get_client(mgr) cls.members_client = members_client.get_client(mgr)
cls.health_monitors_client = health_monitors_client.get_client(mgr) cls.health_monitors_client = health_monitors_client.get_client(mgr)
# l7-switching clients
cls.l7policies_client = l7policies_client.get_client(cls.manager)
cls.l7rules_client = l7rules_client.get_client(cls.manager)
@classmethod @classmethod
def setup_lbaas_core_network(cls): def setup_lbaas_core_network(cls):
@ -95,30 +100,21 @@ class BaseTestCase(base.BaseNetworkTest):
except exceptions.NotFound: except exceptions.NotFound:
continue continue
for listener in lb.get('listeners', []): for listener in lb.get('listeners', []):
for pool in listener.get('pools'): for policy in listener.get('l7policies'):
# delete pool's health-monitor
hm = pool.get('healthmonitor')
if hm:
test_utils.call_and_ignore_notfound_exc(
cls.health_monitors_client.delete_health_monitor,
pool.get('healthmonitor').get('id'))
cls._wait_for_load_balancer_status(lb_id)
# delete pool's members
members = pool.get('members', [])
for member in members:
test_utils.call_and_ignore_notfound_exc(
cls.members_client.delete_member,
pool.get('id'), member.get('id'))
cls._wait_for_load_balancer_status(lb_id)
# delete pool
test_utils.call_and_ignore_notfound_exc( test_utils.call_and_ignore_notfound_exc(
cls.pools_client.delete_pool, pool.get('id')) cls.l7policies_client.delete_l7policy,
policy.get('id'))
cls._wait_for_load_balancer_status(lb_id) cls._wait_for_load_balancer_status(lb_id)
for pool in listener.get('pools'):
cls.delete_lb_pool_resources(lb_id, pool)
# delete listener # delete listener
test_utils.call_and_ignore_notfound_exc( test_utils.call_and_ignore_notfound_exc(
cls.listeners_client.delete_listener, cls.listeners_client.delete_listener,
listener.get('id')) listener.get('id'))
cls._wait_for_load_balancer_status(lb_id) cls._wait_for_load_balancer_status(lb_id)
# delete pools not attached to listener, but loadbalancer
for pool in lb.get('pools', []):
cls.delete_lb_pool_resources(lb_id, pool)
# delete load-balancer # delete load-balancer
test_utils.call_and_ignore_notfound_exc( test_utils.call_and_ignore_notfound_exc(
cls._delete_load_balancer, lb_id) cls._delete_load_balancer, lb_id)
@ -126,6 +122,27 @@ class BaseTestCase(base.BaseNetworkTest):
cls.delete_router(cls.router) cls.delete_router(cls.router)
super(BaseTestCase, cls).resource_cleanup() super(BaseTestCase, cls).resource_cleanup()
@classmethod
def delete_lb_pool_resources(cls, lb_id, pool):
# delete pool's health-monitor
hm = pool.get('healthmonitor')
if hm:
test_utils.call_and_ignore_notfound_exc(
cls.health_monitors_client.delete_health_monitor,
pool.get('healthmonitor').get('id'))
cls._wait_for_load_balancer_status(lb_id)
# delete pool's members
members = pool.get('members', [])
for member in members:
test_utils.call_and_ignore_notfound_exc(
cls.members_client.delete_member,
pool.get('id'), member.get('id'))
cls._wait_for_load_balancer_status(lb_id)
# delete pool
test_utils.call_and_ignore_notfound_exc(
cls.pools_client.delete_pool, pool.get('id'))
cls._wait_for_load_balancer_status(lb_id)
@classmethod @classmethod
def setUpClass(cls): def setUpClass(cls):
cls.LOG = logging.getLogger(cls._get_full_case_name()) cls.LOG = logging.getLogger(cls._get_full_case_name())
@ -137,16 +154,16 @@ class BaseTestCase(base.BaseNetworkTest):
def tearDown(cls): def tearDown(cls):
super(BaseTestCase, cls).tearDown() super(BaseTestCase, cls).tearDown()
cls.LOG.info(_LI('Finished: {0}\n').format(cls._testMethodName)) cls.LOG.info(_LI('Finished: {0}').format(cls._testMethodName))
@classmethod @classmethod
def _create_load_balancer(cls, wait=True, **lb_kwargs): def _create_load_balancer(cls, wait=True, **lb_kwargs):
lb = cls.load_balancers_client.create_load_balancer(**lb_kwargs) lb = cls.load_balancers_client.create_load_balancer(**lb_kwargs)
lb = lb.get('loadbalancer', lb) lb = lb.get('loadbalancer', lb)
cls._lbs_to_delete.append(lb.get('id'))
if wait: if wait:
cls._wait_for_load_balancer_status(lb.get('id')) cls._wait_for_load_balancer_status(lb.get('id'))
cls._lbs_to_delete.append(lb.get('id'))
port = cls.ports_client.show_port(lb['vip_port_id']) port = cls.ports_client.show_port(lb['vip_port_id'])
cls.ports.append(port['port']) cls.ports.append(port['port'])
return lb return lb
@ -414,8 +431,9 @@ class BaseAdminTestCase(BaseTestCase):
def resource_setup(cls): def resource_setup(cls):
super(BaseAdminTestCase, cls).resource_setup() super(BaseAdminTestCase, cls).resource_setup()
mgr = cls.get_client_manager(credential_type='admin') cls.admin_mgr = cls.get_client_manager(credential_type='admin')
cls.create_lbaas_clients(mgr) cls.admin_tenant_id = cls.admin_mgr.networks_client.tenant_id
cls.create_lbaas_clients(cls.admin_mgr)
cls.setup_lbaas_core_network() cls.setup_lbaas_core_network()
@classmethod @classmethod

View File

@ -14,7 +14,6 @@ from oslo_log import log as logging
from oslo_utils import uuidutils from oslo_utils import uuidutils
from tempest import config from tempest import config
from tempest.lib import decorators
from tempest.lib import exceptions as ex from tempest.lib import exceptions as ex
from tempest import test from tempest import test
@ -70,12 +69,12 @@ class TestHealthMonitors(base.BaseAdminTestCase):
self.assertEqual(admin_tenant_id, hm_tenant_id) self.assertEqual(admin_tenant_id, hm_tenant_id)
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1638148")
@test.idempotent_id('acbff982-15d6-43c5-a015-e72b7df30998') @test.idempotent_id('acbff982-15d6-43c5-a015-e72b7df30998')
def test_create_health_monitor_empty_tenant_id_field(self): def test_create_health_monitor_empty_tenant_id_field(self):
"""Test with admin user """Test with admin user
creating health monitor with an empty tenant id field should fail. creating health monitor with an empty tenant id field should fail.
Kilo: @decorators.skip_because(bug="1638148")
""" """
self.assertRaises(ex.BadRequest, self._create_health_monitor, self.assertRaises(ex.BadRequest, self._create_health_monitor,
type='HTTP', delay=3, max_retries=10, type='HTTP', delay=3, max_retries=10,

View File

@ -10,7 +10,6 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
from tempest.lib import decorators
from tempest.lib import exceptions as ex from tempest.lib import exceptions as ex
from tempest import test from tempest import test
@ -305,11 +304,11 @@ class TestHealthMonitors(base.BaseTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('0d637b7f-52ea-429f-8f97-584a5a9118aa') @test.idempotent_id('0d637b7f-52ea-429f-8f97-584a5a9118aa')
@decorators.skip_because(bug="1641652")
def test_create_health_monitor_invalid_url_path(self): def test_create_health_monitor_invalid_url_path(self):
"""Test if a non_admin user """Test if a non_admin user
can create a health monitor with invalid url_path can create a health monitor with invalid url_path
Kilo: @decorators.skip_because(bug="1641652")
""" """
self.assertRaises(ex.BadRequest, self._create_health_monitor, self.assertRaises(ex.BadRequest, self._create_health_monitor,
type='HTTP', delay=3, max_retries=10, timeout=5, type='HTTP', delay=3, max_retries=10, timeout=5,
@ -317,11 +316,11 @@ class TestHealthMonitors(base.BaseTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('7d4061c4-1fbc-43c3-81b5-2d099a120297') @test.idempotent_id('7d4061c4-1fbc-43c3-81b5-2d099a120297')
@decorators.skip_because(bug="1641643")
def test_create_health_monitor_invalid_http_method(self): def test_create_health_monitor_invalid_http_method(self):
"""Test if a non_admin user """Test if a non_admin user
can create a health monitor with invalid http_method can create a health monitor with invalid http_method
Kilo: @decorators.skip_because(bug="1641643")
""" """
self.assertRaises(ex.BadRequest, self._create_health_monitor, self.assertRaises(ex.BadRequest, self._create_health_monitor,
type='HTTP', delay=3, max_retries=10, timeout=5, type='HTTP', delay=3, max_retries=10, timeout=5,
@ -379,18 +378,22 @@ class TestHealthMonitors(base.BaseTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('9c8e8fe8-a3a2-481b-9ac8-eb9ecccd8330') @test.idempotent_id('9c8e8fe8-a3a2-481b-9ac8-eb9ecccd8330')
@decorators.skip_because(bug="1639340")
def test_create_health_monitor_empty_max_http_method(self): def test_create_health_monitor_empty_max_http_method(self):
"""Test create health monitor with empty http_method""" """Test create health monitor with empty http_method
Kilo: @decorators.skip_because(bug="1639340")
"""
self.assertRaises(ex.BadRequest, self._create_health_monitor, self.assertRaises(ex.BadRequest, self._create_health_monitor,
type='HTTP', delay=3, max_retries=10, timeout=5, type='HTTP', delay=3, max_retries=10, timeout=5,
pool_id=self.pool.get('id'), http_method='') pool_id=self.pool.get('id'), http_method='')
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('9016c846-fc7c-4063-9f01-61fad37c435d') @test.idempotent_id('9016c846-fc7c-4063-9f01-61fad37c435d')
@decorators.skip_because(bug="1639340")
def test_create_health_monitor_empty_max_url_path(self): def test_create_health_monitor_empty_max_url_path(self):
"""Test create health monitor with empty url_path""" """Test create health monitor with empty url_path
Kilo: @decorators.skip_because(bug="1639340")
"""
self.assertRaises(ex.BadRequest, self._create_health_monitor, self.assertRaises(ex.BadRequest, self._create_health_monitor,
type='HTTP', delay=3, max_retries=10, timeout=5, type='HTTP', delay=3, max_retries=10, timeout=5,
pool_id=self.pool.get('id'), url_path='') pool_id=self.pool.get('id'), url_path='')
@ -559,8 +562,8 @@ class TestHealthMonitors(base.BaseTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('1e2fb718-de77-46a3-8897-6f5aff6cab5e') @test.idempotent_id('1e2fb718-de77-46a3-8897-6f5aff6cab5e')
@decorators.skip_because(bug="1641643")
def test_update_health_monitor_invalid_http_method(self): def test_update_health_monitor_invalid_http_method(self):
"""Kilo: @decorators.skip_because(bug="1641643")"""
hm = self._create_health_monitor(type='HTTP', delay=3, hm = self._create_health_monitor(type='HTTP', delay=3,
max_retries=10, timeout=5, max_retries=10, timeout=5,
pool_id=self.pool.get('id')) pool_id=self.pool.get('id'))
@ -570,8 +573,8 @@ class TestHealthMonitors(base.BaseTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('07d62a55-18b3-4b74-acb2-b73a0b5e4364') @test.idempotent_id('07d62a55-18b3-4b74-acb2-b73a0b5e4364')
@decorators.skip_because(bug="1641652")
def test_update_health_monitor_invalid_url_path(self): def test_update_health_monitor_invalid_url_path(self):
"""Kilo: @decorators.skip_because(bug="1641652")"""
hm = self._create_health_monitor(type='HTTP', delay=3, hm = self._create_health_monitor(type='HTTP', delay=3,
max_retries=10, timeout=5, max_retries=10, timeout=5,
pool_id=self.pool.get('id')) pool_id=self.pool.get('id'))
@ -631,8 +634,8 @@ class TestHealthMonitors(base.BaseTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('0c464bb3-ff84-4816-9237-4583e4da9881') @test.idempotent_id('0c464bb3-ff84-4816-9237-4583e4da9881')
@decorators.skip_because(bug="1639340")
def test_update_health_monitor_empty_empty_http_method(self): def test_update_health_monitor_empty_empty_http_method(self):
"""Kilo: @decorators.skip_because(bug="1639340")"""
hm = self._create_health_monitor(type='HTTP', delay=3, hm = self._create_health_monitor(type='HTTP', delay=3,
max_retries=10, timeout=5, max_retries=10, timeout=5,
pool_id=self.pool.get('id')) pool_id=self.pool.get('id'))
@ -642,8 +645,8 @@ class TestHealthMonitors(base.BaseTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('3e87c0a8-ef15-457c-a58f-270de8c5c76c') @test.idempotent_id('3e87c0a8-ef15-457c-a58f-270de8c5c76c')
@decorators.skip_because(bug="1639340")
def test_update_health_monitor_empty_url_path(self): def test_update_health_monitor_empty_url_path(self):
"""Kilo: @decorators.skip_because(bug="1639340")"""
hm = self._create_health_monitor(type='HTTP', delay=3, hm = self._create_health_monitor(type='HTTP', delay=3,
max_retries=10, timeout=5, max_retries=10, timeout=5,
pool_id=self.pool.get('id')) pool_id=self.pool.get('id'))
@ -673,8 +676,8 @@ class TestHealthMonitors(base.BaseTestCase):
@test.attr(type=['smoke', 'negative']) @test.attr(type=['smoke', 'negative'])
@test.idempotent_id('fe44e0d9-957b-44cf-806b-af7819444864') @test.idempotent_id('fe44e0d9-957b-44cf-806b-af7819444864')
@decorators.skip_because(bug="1639340")
def test_delete_health_monitor(self): def test_delete_health_monitor(self):
"""Kilo: @decorators.skip_because(bug="1639340")"""
hm = self._create_health_monitor(cleanup=False, type='HTTP', delay=3, hm = self._create_health_monitor(cleanup=False, type='HTTP', delay=3,
max_retries=10, timeout=5, max_retries=10, timeout=5,
pool_id=self.pool.get('id')) pool_id=self.pool.get('id'))

View File

@ -13,7 +13,6 @@
from oslo_log import log as logging from oslo_log import log as logging
from tempest import config from tempest import config
from tempest.lib import decorators
from tempest.lib import exceptions as ex from tempest.lib import exceptions as ex
from tempest import test from tempest import test
@ -54,10 +53,12 @@ class ListenersTest(base.BaseAdminTestCase):
super(ListenersTest, cls).resource_cleanup() super(ListenersTest, cls).resource_cleanup()
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1638738")
@test.idempotent_id('f84bfb35-7f73-4576-b2ca-26193850d2bf') @test.idempotent_id('f84bfb35-7f73-4576-b2ca-26193850d2bf')
def test_create_listener_empty_tenant_id(self): def test_create_listener_empty_tenant_id(self):
"""Test create listener with an empty tenant id should fail""" """Test create listener with an empty tenant id should fail
Kilo: @decorators.skip_because(bug="1638738")
"""
create_new_listener_kwargs = self.create_listener_kwargs create_new_listener_kwargs = self.create_listener_kwargs
create_new_listener_kwargs['protocol_port'] = 8081 create_new_listener_kwargs['protocol_port'] = 8081
create_new_listener_kwargs['tenant_id'] = "" create_new_listener_kwargs['tenant_id'] = ""

View File

@ -13,7 +13,6 @@
from oslo_log import log as logging from oslo_log import log as logging
from tempest import config from tempest import config
from tempest.lib import decorators
from tempest.lib import exceptions from tempest.lib import exceptions
from tempest import test from tempest import test
@ -220,10 +219,12 @@ class ListenersTest(base.BaseTestCase):
listener_ids=[self.listener_id]) listener_ids=[self.listener_id])
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('59d32fd7-06f6-4466-bdd4-0be23b15970c') @test.idempotent_id('59d32fd7-06f6-4466-bdd4-0be23b15970c')
def test_create_listener_invalid_name(self): def test_create_listener_invalid_name(self):
"""Test create listener with an invalid name""" """Test create listener with an invalid name
Kilo: @decorators.skip_because(bug="1637877")
"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.BadRequest,
self._create_listener, self._create_listener,
loadbalancer_id=self.load_balancer_id, loadbalancer_id=self.load_balancer_id,
@ -234,10 +235,12 @@ class ListenersTest(base.BaseTestCase):
listener_ids=[self.listener_id]) listener_ids=[self.listener_id])
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('95457f70-2c1a-4c14-aa80-db8e803d78a9') @test.idempotent_id('95457f70-2c1a-4c14-aa80-db8e803d78a9')
def test_create_listener_invalid_description(self): def test_create_listener_invalid_description(self):
"""Test create listener with an invalid description""" """Test create listener with an invalid description
Kilo: @decorators.skip_because(bug="1637877")
"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.BadRequest,
self._create_listener, self._create_listener,
loadbalancer_id=self.load_balancer_id, loadbalancer_id=self.load_balancer_id,
@ -312,10 +315,12 @@ class ListenersTest(base.BaseTestCase):
listener_ids=[self.listener_id]) listener_ids=[self.listener_id])
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1638701")
@test.idempotent_id('46fc3784-d676-42f7-953b-a23c1d62323d') @test.idempotent_id('46fc3784-d676-42f7-953b-a23c1d62323d')
def test_create_listener_empty_tenant_id(self): def test_create_listener_empty_tenant_id(self):
"""Test create listener with an empty tenant id""" """Test create listener with an empty tenant id
Kilo: @decorators.skip_because(bug="1638701")
"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.BadRequest,
self._create_listener, self._create_listener,
loadbalancer_id=self.load_balancer_id, loadbalancer_id=self.load_balancer_id,
@ -417,10 +422,12 @@ class ListenersTest(base.BaseTestCase):
listener_ids=[self.listener_id]) listener_ids=[self.listener_id])
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('7c0efb63-90d9-43d0-b959-eb841ef39832') @test.idempotent_id('7c0efb63-90d9-43d0-b959-eb841ef39832')
def test_update_listener_invalid_name(self): def test_update_listener_invalid_name(self):
"""Test update a listener with an invalid name""" """Test update a listener with an invalid name
Kilo: @decorators.skip_because(bug="1637877")
"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.BadRequest,
self._update_listener, self._update_listener,
listener_id=self.listener_id, listener_id=self.listener_id,
@ -429,10 +436,12 @@ class ListenersTest(base.BaseTestCase):
listener_ids=[self.listener_id]) listener_ids=[self.listener_id])
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('ba9bfad8-dbb0-4cbc-b2e3-52bf72bc1fc5') @test.idempotent_id('ba9bfad8-dbb0-4cbc-b2e3-52bf72bc1fc5')
def test_update_listener_invalid_description(self): def test_update_listener_invalid_description(self):
"""Test update a listener with an invalid description""" """Test update a listener with an invalid description
Kilo: @decorators.skip_because(bug="1637877")
"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.BadRequest,
self._update_listener, self._update_listener,
listener_id=self.listener_id, listener_id=self.listener_id,

View File

@ -11,9 +11,9 @@
# under the License. # under the License.
from oslo_log import log as logging from oslo_log import log as logging
import testtools
from tempest import config from tempest import config
from tempest.lib import decorators
from tempest.lib import exceptions as ex from tempest.lib import exceptions as ex
from tempest import test from tempest import test
@ -47,13 +47,15 @@ class LoadBalancersTest(base.BaseAdminTestCase):
cls.load_balancer_id = cls.load_balancer['id'] cls.load_balancer_id = cls.load_balancer['id']
@test.attr(type='smoke') @test.attr(type='smoke')
@decorators.skip_because(bug="1641902") @testtools.skipIf('1641902' in CONF.nsxv.bugs_to_resolve,
"skip_because bug=1641902")
@test.idempotent_id('0008ae1e-77a2-45d9-b81e-0e3119b5a26d') @test.idempotent_id('0008ae1e-77a2-45d9-b81e-0e3119b5a26d')
def test_create_load_balancer_missing_tenant_id_field_for_admin(self): def test_create_load_balancer_missing_tenant_id_field_for_admin(self):
"""Test create load balancer with a missing tenant id field. """Test create load balancer with a missing tenant id field.
Verify tenant_id matches when creating loadbalancer vs. Verify tenant_id matches when creating loadbalancer vs.
load balancer(admin tenant) load balancer(admin tenant)
Kilo: @decorators.skip_because(bug="1641902")
""" """
load_balancer = self._create_load_balancer( load_balancer = self._create_load_balancer(
vip_subnet_id=self.subnet['id']) vip_subnet_id=self.subnet['id'])
@ -65,13 +67,15 @@ class LoadBalancersTest(base.BaseAdminTestCase):
self._wait_for_load_balancer_status(load_balancer['id']) self._wait_for_load_balancer_status(load_balancer['id'])
@test.attr(type='smoke') @test.attr(type='smoke')
@decorators.skip_because(bug="1638571") @testtools.skipIf('1715126' in CONF.nsxv.bugs_to_resolve,
"skip_because bug=1715126")
@test.idempotent_id('37620941-47c1-40b2-84d8-db17ff823ebc') @test.idempotent_id('37620941-47c1-40b2-84d8-db17ff823ebc')
def test_create_load_balancer_missing_tenant_id_for_other_tenant(self): def test_create_load_balancer_missing_tenant_id_for_other_tenant(self):
"""Test create load balancer with a missing tenant id field. """Test create load balancer with a missing tenant id field.
Verify tenant_id does not match of subnet(non-admin tenant) vs. Verify tenant_id does not match of subnet(non-admin tenant) vs.
load balancer(admin tenant) load balancer(admin tenant)
Kilo: @decorators.skip_because(bug="1638571")
""" """
load_balancer = self._create_load_balancer( load_balancer = self._create_load_balancer(
vip_subnet_id=self.subnet['id']) vip_subnet_id=self.subnet['id'])
@ -81,11 +85,13 @@ class LoadBalancersTest(base.BaseAdminTestCase):
self._wait_for_load_balancer_status(load_balancer['id']) self._wait_for_load_balancer_status(load_balancer['id'])
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1638148")
# Empty tenant_id causing ServerFault
@test.idempotent_id('5bf483f5-ae28-47f5-8805-642da0ffcb40') @test.idempotent_id('5bf483f5-ae28-47f5-8805-642da0ffcb40')
# Empty tenant_id causing ServerFault
def test_create_load_balancer_empty_tenant_id_field(self): def test_create_load_balancer_empty_tenant_id_field(self):
"""Test create load balancer with empty tenant_id field should fail""" """Test create load balancer with empty tenant_id field should fail
Kilo: @decorators.skip_because(bug="1638148")
"""
self.assertRaises(ex.BadRequest, self.assertRaises(ex.BadRequest,
self._create_load_balancer, self._create_load_balancer,
vip_subnet_id=self.subnet['id'], vip_subnet_id=self.subnet['id'],
@ -93,11 +99,13 @@ class LoadBalancersTest(base.BaseAdminTestCase):
tenant_id="") tenant_id="")
@test.attr(type='smoke') @test.attr(type='smoke')
@decorators.skip_because(bug="1638571")
@test.idempotent_id('19fc8a44-1280-49f3-be5b-0d30e6e43363') @test.idempotent_id('19fc8a44-1280-49f3-be5b-0d30e6e43363')
# 2nd tenant_id at the same subnet not supported; got serverFault # NSX-v: 2nd tenant_id at the same subnet not supported; got serverFault
def test_create_load_balancer_for_another_tenant(self): def test_create_load_balancer_for_another_tenant(self):
"""Test create load balancer for other tenant""" """Test create load balancer for other tenant
Kilo: @decorators.skip_because(bug="1638571")
"""
tenant = 'deffb4d7c0584e89a8ec99551565713c' tenant = 'deffb4d7c0584e89a8ec99551565713c'
load_balancer = self._create_load_balancer( load_balancer = self._create_load_balancer(
vip_subnet_id=self.subnet['id'], vip_subnet_id=self.subnet['id'],

View File

@ -13,9 +13,9 @@
import netaddr import netaddr
from oslo_log import log as logging from oslo_log import log as logging
import testtools
from tempest import config from tempest import config
from tempest.lib import decorators
from tempest.lib import exceptions from tempest.lib import exceptions
from tempest import test from tempest import test
@ -240,10 +240,12 @@ class LoadBalancersTest(base.BaseTestCase):
tenant_id="&^%123") tenant_id="&^%123")
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('b8c56e4a-9644-4119-8fc9-130841caf662') @test.idempotent_id('b8c56e4a-9644-4119-8fc9-130841caf662')
def test_create_load_balancer_invalid_name(self): def test_create_load_balancer_invalid_name(self):
"""Test create load balancer with an invalid name""" """Test create load balancer with an invalid name
Kilo: @decorators.skip_because(bug="1637877")
"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.BadRequest,
self._create_load_balancer, self._create_load_balancer,
wait=False, wait=False,
@ -252,10 +254,12 @@ class LoadBalancersTest(base.BaseTestCase):
name='n' * 256) name='n' * 256)
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('d638ae60-7de5-45da-a7d9-53eca4998980') @test.idempotent_id('d638ae60-7de5-45da-a7d9-53eca4998980')
def test_create_load_balancer_invalid_description(self): def test_create_load_balancer_invalid_description(self):
"""Test create load balancer with an invalid description""" """Test create load balancer with an invalid description
Kilo: @decorators.skip_because(bug="1637877")
"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.BadRequest,
self._create_load_balancer, self._create_load_balancer,
wait=False, wait=False,
@ -307,21 +311,23 @@ class LoadBalancersTest(base.BaseTestCase):
tenant_id=tenant) tenant_id=tenant)
@test.attr(type='negative') @test.attr(type='negative')
@testtools.skipIf('1703396' in CONF.nsxv.bugs_to_resolve,
"skip_because bug=1703396")
@test.idempotent_id('9963cbf5-97d0-4ab9-96e5-6cbd65c98714') @test.idempotent_id('9963cbf5-97d0-4ab9-96e5-6cbd65c98714')
# TODO(akang): upstream is exceptions.NotFound
def test_create_load_balancer_invalid_flavor_field(self): def test_create_load_balancer_invalid_flavor_field(self):
"""Test create load balancer with an invalid flavor field""" """Test create load balancer with an invalid flavor field"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.NotFound,
self._create_load_balancer, self._create_load_balancer,
vip_subnet_id=self.subnet['id'], vip_subnet_id=self.subnet['id'],
flavor_id="NO_SUCH_FLAVOR") flavor_id="NO_SUCH_FLAVOR")
@test.attr(type='negative') @test.attr(type='negative')
@testtools.skipIf('1703396' in CONF.nsxv.bugs_to_resolve,
"skip_because bug=1703396")
@test.idempotent_id('f7319e32-0fad-450e-8f53-7567f56e8223') @test.idempotent_id('f7319e32-0fad-450e-8f53-7567f56e8223')
# TODO(akang): upstream is exceptions.Conflict
def test_create_load_balancer_provider_flavor_conflict(self): def test_create_load_balancer_provider_flavor_conflict(self):
"""Test create load balancer with both a provider and a flavor""" """Test create load balancer with both a provider and a flavor"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.Conflict,
self._create_load_balancer, self._create_load_balancer,
vip_subnet_id=self.subnet['id'], vip_subnet_id=self.subnet['id'],
flavor_id="NO_SUCH_FLAVOR", flavor_id="NO_SUCH_FLAVOR",
@ -348,10 +354,12 @@ class LoadBalancersTest(base.BaseTestCase):
self.assertEqual(load_balancer.get('name'), "") self.assertEqual(load_balancer.get('name'), "")
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('551be885-215d-4941-8870-651cbc871162') @test.idempotent_id('551be885-215d-4941-8870-651cbc871162')
def test_update_load_balancer_invalid_name(self): def test_update_load_balancer_invalid_name(self):
"""Test update load balancer with invalid name""" """Test update load balancer with invalid name
Kilo: @decorators.skip_because(bug="1637877")
"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.BadRequest,
self._update_load_balancer, self._update_load_balancer,
load_balancer_id=self.load_balancer_id, load_balancer_id=self.load_balancer_id,
@ -372,10 +380,12 @@ class LoadBalancersTest(base.BaseTestCase):
self.assertEqual(load_balancer_initial, load_balancer_new) self.assertEqual(load_balancer_initial, load_balancer_new)
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('ab3550c6-8b21-463c-bc5d-e79cbae3432f') @test.idempotent_id('ab3550c6-8b21-463c-bc5d-e79cbae3432f')
def test_update_load_balancer_invalid_description(self): def test_update_load_balancer_invalid_description(self):
"""Test update load balancer with invalid description""" """Test update load balancer with invalid description
Kilo: @decorators.skip_because(bug="1637877")
"""
self.assertRaises(exceptions.BadRequest, self.assertRaises(exceptions.BadRequest,
self._update_load_balancer, self._update_load_balancer,
load_balancer_id=self.load_balancer_id, load_balancer_id=self.load_balancer_id,

View File

@ -13,7 +13,6 @@
from oslo_log import log as logging from oslo_log import log as logging
from tempest import config from tempest import config
from tempest.lib import decorators
from tempest.lib import exceptions as ex from tempest.lib import exceptions as ex
from tempest import test from tempest import test
@ -71,9 +70,11 @@ class MemberTest(base.BaseAdminTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('01c9ea0c-bdfe-4108-95d1-69ecdc0a1f26') @test.idempotent_id('01c9ea0c-bdfe-4108-95d1-69ecdc0a1f26')
@decorators.skip_because(bug="1638148")
def test_create_member_empty_tenant_id(self): def test_create_member_empty_tenant_id(self):
"""Test create member with an empty tenant_id should fail""" """Test create member with an empty tenant_id should fail
Kilo: @decorators.skip_because(bug="1638148")
"""
member_opts = {} member_opts = {}
member_opts['address'] = "127.0.0.1" member_opts['address'] = "127.0.0.1"
member_opts['protocol_port'] = 80 member_opts['protocol_port'] = 80

View File

@ -10,7 +10,6 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
from tempest.lib import decorators
from tempest.lib import exceptions as ex from tempest.lib import exceptions as ex
from tempest import test from tempest import test
@ -64,9 +63,11 @@ class TestPools(base.BaseAdminTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('71b9d3e1-3f13-4c84-a905-054c9cd3d4aa') @test.idempotent_id('71b9d3e1-3f13-4c84-a905-054c9cd3d4aa')
@decorators.skip_because(bug="1638148")
def test_create_pool_using_empty_tenant_field(self): def test_create_pool_using_empty_tenant_field(self):
"""Test create pool with empty tenant field should fail""" """Test create pool with empty tenant field should fail
Kilo: @decorators.skip_because(bug="1638148")
"""
self.assertRaises(ex.BadRequest, self._create_pool, self.assertRaises(ex.BadRequest, self._create_pool,
protocol='HTTP', protocol='HTTP',
tenant_id="", tenant_id="",

View File

@ -10,7 +10,6 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
from tempest.lib import decorators
from tempest.lib import exceptions as ex from tempest.lib import exceptions as ex
from tempest import test from tempest import test
@ -381,11 +380,11 @@ class TestPools(base.BaseTestCase):
@test.attr(type='negative') @test.attr(type='negative')
@test.idempotent_id('cb564af8-89aa-40ca-850e-55418da0f235') @test.idempotent_id('cb564af8-89aa-40ca-850e-55418da0f235')
@decorators.skip_because(bug="1637877")
def test_create_pool_invalid_name_field(self): def test_create_pool_invalid_name_field(self):
"""known bug with """known bug with
input more than 255 chars Test create pool with invalid name field input more than 255 chars Test create pool with invalid name field
Kilo: @decorators.skip_because(bug="1637877")
""" """
self.assertRaises(ex.BadRequest, self._create_pool, self.assertRaises(ex.BadRequest, self._create_pool,
protocol='HTTP', protocol='HTTP',
@ -394,12 +393,12 @@ class TestPools(base.BaseTestCase):
name='n' * 256) name='n' * 256)
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('7f4472be-feb7-4ab7-9fb9-97e08f1fa787') @test.idempotent_id('7f4472be-feb7-4ab7-9fb9-97e08f1fa787')
def test_create_pool_invalid_desc_field(self): def test_create_pool_invalid_desc_field(self):
"""known bug with """known bug with
input more than 255 chars Test create pool with invalid desc field input more than 255 chars Test create pool with invalid desc field
Kilo: @decorators.skip_because(bug="1637877")
""" """
self.assertRaises(ex.BadRequest, self._prepare_and_create_pool, self.assertRaises(ex.BadRequest, self._prepare_and_create_pool,
protocol='HTTP', protocol='HTTP',
@ -521,19 +520,23 @@ class TestPools(base.BaseTestCase):
self.assertAlmostEqual(sess_pers, pool.get('session_persistence')) self.assertAlmostEqual(sess_pers, pool.get('session_persistence'))
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('23a9dbaf-105b-450e-95cf-050203b28366') @test.idempotent_id('23a9dbaf-105b-450e-95cf-050203b28366')
def test_update_pool_invalid_name(self): def test_update_pool_invalid_name(self):
"""Test update pool with invalid name""" """Test update pool with invalid name
Kilo: @decorators.skip_because(bug="1637877")
"""
new_pool = self._prepare_and_create_pool() new_pool = self._prepare_and_create_pool()
self.assertRaises(ex.BadRequest, self._update_pool, self.assertRaises(ex.BadRequest, self._update_pool,
new_pool.get('id'), name='n' * 256) new_pool.get('id'), name='n' * 256)
@test.attr(type='negative') @test.attr(type='negative')
@decorators.skip_because(bug="1637877")
@test.idempotent_id('efeeb827-5cb0-4349-8272-b2dbcbf42d22') @test.idempotent_id('efeeb827-5cb0-4349-8272-b2dbcbf42d22')
def test_update_pool_invalid_desc(self): def test_update_pool_invalid_desc(self):
"""Test update pool with invalid desc""" """Test update pool with invalid desc
Kilo: @decorators.skip_because(bug="1637877")
"""
new_pool = self._prepare_and_create_pool() new_pool = self._prepare_and_create_pool()
self.assertRaises(ex.BadRequest, self._update_pool, self.assertRaises(ex.BadRequest, self._update_pool,
new_pool.get('id'), new_pool.get('id'),

View File

@ -16,6 +16,7 @@
import collections import collections
import os import os
import re import re
import shlex
import subprocess import subprocess
import time import time
import traceback import traceback
@ -25,6 +26,7 @@ from tempest.common import waiters
from tempest import config from tempest import config
from tempest.lib.common.utils import data_utils from tempest.lib.common.utils import data_utils
from tempest.lib.common.utils import test_utils from tempest.lib.common.utils import test_utils
from tempest.lib import exceptions
from tempest.scenario import manager from tempest.scenario import manager
from tempest import test from tempest import test
@ -90,8 +92,8 @@ class TopoDeployScenarioManager(manager.NetworkScenarioTest):
@classmethod @classmethod
def check_preconditions(cls): def check_preconditions(cls):
super(TopoDeployScenarioManager, cls).check_preconditions() super(TopoDeployScenarioManager, cls).check_preconditions()
if not (CONF.network.project_networks_reachable if not (CONF.network.project_networks_reachable or
or CONF.network.public_network_id): CONF.network.public_network_id):
msg = ('Either project_networks_reachable must be "true", or ' msg = ('Either project_networks_reachable must be "true", or '
'public_network_id must be defined.') 'public_network_id must be defined.')
cls.enabled = False cls.enabled = False
@ -386,19 +388,61 @@ class TopoDeployScenarioManager(manager.NetworkScenarioTest):
return HELO.create_subnet(self, network, **kwargs) return HELO.create_subnet(self, network, **kwargs)
def create_floatingip_for_server(self, server, external_network_id=None, def create_floatingip_for_server(self, server, external_network_id=None,
port_id=None, client_mgr=None): port_id=None, client_mgr=None,
and_check_assigned=True):
client_mgr = client_mgr or self.manager client_mgr = client_mgr or self.manager
net_floatingip = self.create_floating_ip( net_floatingip = self.create_floating_ip(
server, server,
external_network_id=external_network_id, external_network_id=external_network_id,
port_id=port_id, port_id=port_id,
client=client_mgr.floating_ips_client) client=client_mgr.floating_ips_client)
if port_id:
# attached to port, will not check ip assignement & reachability
return net_floatingip
if not and_check_assigned:
# caller will do the floatingip assigned to server and ping tests
return net_floatingip
self._waitfor_floatingip_assigned_to_server(client_mgr.servers_client,
server.get('id'))
server_pingable = self._waitfor_associated_floatingip(net_floatingip) server_pingable = self._waitfor_associated_floatingip(net_floatingip)
self.assertTrue( self.assertTrue(
server_pingable, server_pingable,
msg="Expect server to be reachable after floatingip assigned.") msg="Expect server to be reachable after floatingip assigned.")
return net_floatingip return net_floatingip
def _waitfor_floatingip_assigned_to_server(self, server_client, server_id,
on_network=None,
extra_timeout=60):
timeout = server_client.build_timeout + extra_timeout
interval = server_client.build_interval
start_time = time.time()
while time.time() - start_time < timeout:
sv = server_client.show_server(server_id)
sv = sv.get('server', sv)
fip = self.get_server_ip_address(sv, 'floating')
if fip:
elapse_time = time.time() - start_time
xmsg = ("%s Take %d seconds to assign floatingip to server[%s]"
% ("OS-STATS:", int(elapse_time), sv.get('name')))
LOG.debug(xmsg)
return fip
time.sleep(interval)
raise Exception(
"Server[%s] did not get its floatingip in %s seconds" %
(server_id, timeout))
def get_server_ip_address(self, server, ip_type='fixed',
network_name=None):
if network_name and server['addresses'].get(network_name):
s_if = network_name
else:
s_if = server['addresses'].keys()[0]
for s_address in server['addresses'][s_if]:
if s_address['OS-EXT-IPS:type'] == ip_type:
return s_address.get('addr')
return None
def _waitfor_associated_floatingip(self, net_floatingip): def _waitfor_associated_floatingip(self, net_floatingip):
host_ip = net_floatingip['floating_ip_address'] host_ip = net_floatingip['floating_ip_address']
return self.waitfor_host_connected(host_ip) return self.waitfor_host_connected(host_ip)
@ -705,3 +749,23 @@ def waitfor_servers_terminated(tenant_servers_client, pause=2.0):
if len(s_list) < 1: if len(s_list) < 1:
return return
time.sleep(pause) time.sleep(pause)
def copy_file_to_host(file_from, dest, host, username, pkey):
dest = "%s@%s:%s" % (username, host, dest)
cmd = "scp -v -o UserKnownHostsFile=/dev/null " \
"-o StrictHostKeyChecking=no " \
"-i %(pkey)s %(file1)s %(dest)s" % {'pkey': pkey,
'file1': file_from,
'dest': dest}
args = shlex.split(cmd.encode('utf-8'))
subprocess_args = {'stdout': subprocess.PIPE,
'stderr': subprocess.STDOUT}
proc = subprocess.Popen(args, **subprocess_args)
stdout, stderr = proc.communicate()
if proc.returncode != 0:
raise exceptions.SSHExecCommandFailed(cmd,
proc.returncode,
stdout,
stderr)
return stdout

View File

@ -0,0 +1,170 @@
# 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 time
from tempest import config
from tempest import test
from vmware_nsx_tempest.services.lbaas import l7policies_client
from vmware_nsx_tempest.services.lbaas import l7rules_client
from vmware_nsx_tempest.tests.nsxv.scenario import (
test_lbaas_round_robin_ops as lbaas_ops)
CONF = config.CONF
class TestL7SwitchingOps(lbaas_ops.LBaasRoundRobinBaseTest):
"""This test validates lbaas l7 switching with round-robin opertion.
Test leverage test_lbaas_round_robin to create the basic round-robin
operation, and then build l7 pool and members to forwarding url path
starts_with value specified.
Manual operation can be found at test proc: https://goo.gl/btDMXy
"""
@classmethod
def skip_checks(cls):
super(TestL7SwitchingOps, cls).skip_checks()
if '1739510' in CONF.nsxv.bugs_to_resolve:
msg = ("skip lbaas_l7_switching_ops because bug=1739150"
" -- l7 switching is not supported")
raise cls.skipException(msg)
@classmethod
def resource_setup(cls):
super(TestL7SwitchingOps, cls).resource_setup()
cls.create_lbaas_clients(cls.manager)
cls.l7policies_client = l7policies_client.get_client(cls.manager)
cls.l7rules_client = l7rules_client.get_client(cls.manager)
@classmethod
def setup_credentials(cls):
super(TestL7SwitchingOps, cls).setup_credentials()
def setUp(self):
super(TestL7SwitchingOps, self).setUp()
self.switching_startswith_value1 = "/api"
self.switching_startswith_value2 = "/api2"
self.pool7 = None
self.l7policy1 = None
self.l7rule1 = None
self.l7rule_kwargs = dict(type='PATH',
compare_type='STARTS_WITH',
value=self.switching_startswith_value1)
def tearDown(self):
lb_id = self.loadbalancer['id']
# teardown lbaas l7 provision
if self.l7policy1:
self.l7policies_client.delete_l7policy(self.l7policy1.get('id'))
self.wait_for_load_balancer_status(lb_id)
if self.pool7:
self.pools_client.delete_pool(self.pool7.get('id'))
self.wait_for_load_balancer_status(lb_id)
super(TestL7SwitchingOps, self).tearDown()
def create_and_start_l7_web_servers(self):
key_name = self.keypair['name']
network_name = self.network['name']
security_groups = [{'name': self.security_group['id']}]
self.server7 = self.create_server_on_network(
self.network, name=(network_name + "-7"),
security_groups=security_groups,
key_name=key_name, wait_on_boot=False,
servers_client=self.manager.servers_client)
self.server8 = self.create_server_on_network(
self.network, name=(network_name + "-8"),
security_groups=security_groups,
key_name=key_name, wait_on_boot=False,
servers_client=self.manager.servers_client)
self.l7_server_list = [self.server7, self.server8]
self.wait_for_servers_become_active(self.l7_server_list)
self.start_web_servers(self.l7_server_list)
def build_l7_switching(self):
subnet_id = self.subnet.get('id')
lb_id = self.loadbalancer['id']
l7_name = self.loadbalancer['name'] + "-7"
redirect_to_listener_id = self.listener.get('id')
# build_l7_pool(loadbalancer_id):
self.pool7 = self.pools_client .create_pool(
loadbalancer_id=lb_id,
lb_algorithm=self.lb_algorithm, protocol=self.protocol_type,
name=l7_name)['pool']
self.wait_for_load_balancer_status(lb_id)
pool_id = self.pool7['id']
self.member7_list = []
for server in self.l7_server_list:
fip = server['_floating_ip']
fixed_ip_address = fip['fixed_ip_address']
member = self.members_client.create_member(
pool_id, subnet_id=subnet_id,
address=fixed_ip_address,
protocol_port=self.protocol_port)
self.wait_for_load_balancer_status(lb_id)
self.member7_list.append(member)
l7policy_kwargs = dict(action="REDIRECT_TO_POOL",
redirect_pool_id=pool_id,
listener_id=redirect_to_listener_id,
name='policy1')
l7policy1 = self.l7policies_client.create_l7policy(**l7policy_kwargs)
self.l7policy1 = l7policy1.get(u'l7policy', l7policy1)
policy_id = self.l7policy1.get('id')
self.l7rule1 = self.l7rules_client.create_l7rule(
policy_id, **self.l7rule_kwargs)['rule']
def check_l7_switching(self, start_path, expected_server_list,
send_count=6):
self.do_http_request(start_path, send_count)
for sv_name, cnt in self.http_cnt.items():
self.assertIn(sv_name, expected_server_list)
self.assertTrue(cnt > 0)
def validate_l7_switching(self):
l7_sv_name_list = [s['name'] for s in self.l7_server_list]
rr_sv_name_list = [s['name'] for s in self.rr_server_list]
# URL prefix api switching to pool7
self.check_l7_switching('api', l7_sv_name_list, 6)
# URL prefix ap/i switching to pool1
self.check_l7_switching('ap/i', rr_sv_name_list, 6)
# URL prefix api2 switching to pool7
self.check_l7_switching('api2', l7_sv_name_list, 6)
# change rule starts_with's value to /api2
# and /api & /api/2 will be swithed to default pool
policy_id = self.l7policy1.get('id')
rule_id = self.l7rule1.get('id')
self.l7rule_kwargs['value'] = self.switching_startswith_value2
self.l7rule2 = self.l7rules_client.update_l7rule(
policy_id, rule_id, **self.l7rule_kwargs)['rule']
time.sleep(2.0)
# URL prefix api switching to pool
self.check_l7_switching('api', rr_sv_name_list, 6)
# URL prefix api switching to pool
self.check_l7_switching('api/2', rr_sv_name_list, 6)
# URL prefix api2 switching to pool7
self.check_l7_switching('api2', l7_sv_name_list, 6)
# URL prefix api2 switching to pool
self.check_l7_switching('xapi2', rr_sv_name_list, 6)
@test.idempotent_id('f11e19e4-16b5-41c7-878d-59b9e943e3ce')
@test.services('compute', 'network')
def test_lbaas_l7_switching_ops(self):
self.create_lbaas_networks()
self.start_web_servers()
self.create_project_lbaas()
self.check_project_lbaas()
# do l7 provision and testing
self.create_and_start_l7_web_servers()
self.build_l7_switching()
self.validate_l7_switching()

View File

@ -26,32 +26,26 @@ from vmware_nsx_tempest.services.lbaas import members_client
from vmware_nsx_tempest.services.lbaas import pools_client from vmware_nsx_tempest.services.lbaas import pools_client
from vmware_nsx_tempest.tests.nsxv.scenario import ( from vmware_nsx_tempest.tests.nsxv.scenario import (
manager_topo_deployment as dmgr) manager_topo_deployment as dmgr)
from vmware_nsx_tempest.tests.nsxv.scenario import test_v1_lbaas_basic_ops
CONF = config.CONF CONF = config.CONF
LOG = dmgr.manager.log.getLogger(__name__) LOG = dmgr.manager.log.getLogger(__name__)
class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager): class LBaasRoundRobinBaseTest(dmgr.TopoDeployScenarioManager):
"""Base class to support LBaaS ROUND-ROBIN test.
"""This test checks basic load balancer V2 ROUND-ROBIN operation. It provides the methods to create loadbalancer network, and
start web servers.
The following is the scenario outline: Default lb_algorithm is ROUND_ROBIND.
1. Create network with exclusive router, and 2 servers
2. SSH to each instance and start web server
3. Create a load balancer with 1 listener, 1 pool, 1 healthmonitor
and 2 members and with ROUND_ROBIN algorithm.
4. Associate loadbalancer's vip_address with a floating ip
5. Send NUM requests to vip's floating ip and check that they are shared
between the two servers.
""" """
tenant_router_attrs = {'router_type': 'exclusive'} tenant_router_attrs = {'router_type': 'exclusive'}
@classmethod @classmethod
def skip_checks(cls): def skip_checks(cls):
super(TestLBaasRoundRobinOps, cls).skip_checks() super(LBaasRoundRobinBaseTest, cls).skip_checks()
cfg = CONF.network cfg = CONF.network
if not test.is_extension_enabled('lbaasv2', 'network'): if not test.is_extension_enabled('lbaasv2', 'network'):
msg = 'lbaasv2 extension is not enabled.' msg = 'lbaasv2 extension is not enabled.'
@ -63,7 +57,7 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
@classmethod @classmethod
def resource_setup(cls): def resource_setup(cls):
super(TestLBaasRoundRobinOps, cls).resource_setup() super(LBaasRoundRobinBaseTest, cls).resource_setup()
cls.create_lbaas_clients(cls.manager) cls.create_lbaas_clients(cls.manager)
@classmethod @classmethod
@ -78,10 +72,10 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
def setup_credentials(cls): def setup_credentials(cls):
# Ask framework to not create network resources for these tests. # Ask framework to not create network resources for these tests.
cls.set_network_resources() cls.set_network_resources()
super(TestLBaasRoundRobinOps, cls).setup_credentials() super(LBaasRoundRobinBaseTest, cls).setup_credentials()
def setUp(self): def setUp(self):
super(TestLBaasRoundRobinOps, self).setUp() super(LBaasRoundRobinBaseTest, self).setUp()
CONF.validation.ssh_shell_prologue = '' CONF.validation.ssh_shell_prologue = ''
self.namestart = 'lbaas-ops' self.namestart = 'lbaas-ops'
self.poke_counters = 10 self.poke_counters = 10
@ -91,6 +85,7 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
self.hm_delay = 4 self.hm_delay = 4
self.hm_max_retries = 3 self.hm_max_retries = 3
self.hm_timeout = 10 self.hm_timeout = 10
self.hm_type = 'PING'
self.server_names = [] self.server_names = []
self.loadbalancer = None self.loadbalancer = None
self.vip_fip = None self.vip_fip = None
@ -108,7 +103,7 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
# make sure servers terminated before teardown network resources # make sure servers terminated before teardown network resources
LOG.debug("tearDown lbaas servers") LOG.debug("tearDown lbaas servers")
server_id_list = [] server_id_list = []
for servid in ['server1', 'server2']: for servid in ['server1', 'server2', 'server7', 'server8']:
server = getattr(self, servid, None) server = getattr(self, servid, None)
if server: if server:
if '_floating_ip' in server: if '_floating_ip' in server:
@ -120,7 +115,7 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
waiters.wait_for_server_termination( waiters.wait_for_server_termination(
self.manager.servers_client, server_id) self.manager.servers_client, server_id)
# delete lbaas network before handing back to framework # delete lbaas network before handing back to framework
super(TestLBaasRoundRobinOps, self).tearDown() super(LBaasRoundRobinBaseTest, self).tearDown()
LOG.debug("tearDown lbaas exiting...") LOG.debug("tearDown lbaas exiting...")
def delete_loadbalancer_resources(self, lb_id): def delete_loadbalancer_resources(self, lb_id):
@ -129,26 +124,17 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
statuses = statuses.get('statuses', statuses) statuses = statuses.get('statuses', statuses)
lb = statuses.get('loadbalancer') lb = statuses.get('loadbalancer')
for listener in lb.get('listeners', []): for listener in lb.get('listeners', []):
for policy in listener.get('l7policies'):
self.l7policies_client.delete_policy(policy.get('id'))
for pool in listener.get('pools'): for pool in listener.get('pools'):
pool_id = pool.get('id') self.delete_lb_pool_resources(lb_id, pool)
hm = pool.get('healthmonitor')
if hm:
test_utils.call_and_ignore_notfound_exc(
self.health_monitors_client.delete_health_monitor,
pool.get('healthmonitor').get('id'))
self.wait_for_load_balancer_status(lb_id)
test_utils.call_and_ignore_notfound_exc(
self.pools_client.delete_pool, pool.get('id'))
self.wait_for_load_balancer_status(lb_id)
for member in pool.get('members', []):
test_utils.call_and_ignore_notfound_exc(
self.members_client.delete_member,
pool_id, member.get('id'))
self.wait_for_load_balancer_status(lb_id)
test_utils.call_and_ignore_notfound_exc( test_utils.call_and_ignore_notfound_exc(
self.listeners_client.delete_listener, self.listeners_client.delete_listener,
listener.get('id')) listener.get('id'))
self.wait_for_load_balancer_status(lb_id) self.wait_for_load_balancer_status(lb_id)
# delete pools not attached to listener, but loadbalancer
for pool in lb.get('pools', []):
self.delete_lb_pool_resources(lb_id, pool)
test_utils.call_and_ignore_notfound_exc( test_utils.call_and_ignore_notfound_exc(
lb_client.delete_load_balancer, lb_id) lb_client.delete_load_balancer, lb_id)
self.load_balancers_client.wait_for_load_balancer_status( self.load_balancers_client.wait_for_load_balancer_status(
@ -156,6 +142,23 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
lbs = lb_client.list_load_balancers()['loadbalancers'] lbs = lb_client.list_load_balancers()['loadbalancers']
self.assertEqual(0, len(lbs)) self.assertEqual(0, len(lbs))
def delete_lb_pool_resources(self, lb_id, pool):
pool_id = pool.get('id')
hm = pool.get('healthmonitor')
if hm:
test_utils.call_and_ignore_notfound_exc(
self.health_monitors_client.delete_health_monitor,
pool.get('healthmonitor').get('id'))
self.wait_for_load_balancer_status(lb_id)
test_utils.call_and_ignore_notfound_exc(
self.pools_client.delete_pool, pool.get('id'))
self.wait_for_load_balancer_status(lb_id)
for member in pool.get('members', []):
test_utils.call_and_ignore_notfound_exc(
self.members_client.delete_member,
pool_id, member.get('id'))
self.wait_for_load_balancer_status(lb_id)
def wait_for_load_balancer_status(self, lb_id): def wait_for_load_balancer_status(self, lb_id):
# Wait for load balancer become ONLINE and ACTIVE # Wait for load balancer become ONLINE and ACTIVE
self.load_balancers_client.wait_for_load_balancer_status(lb_id) self.load_balancers_client.wait_for_load_balancer_status(lb_id)
@ -180,10 +183,11 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
security_groups=security_groups, security_groups=security_groups,
key_name=key_name, key_name=key_name,
servers_client=self.manager.servers_client) servers_client=self.manager.servers_client)
self.wait_for_servers_become_active() self.rr_server_list = [self.server1, self.server2]
self.wait_for_servers_become_active(self.rr_server_list)
def wait_for_servers_become_active(self): def wait_for_servers_become_active(self, server_list):
for serv in [self.server1, self.server2]: for serv in server_list:
waiters.wait_for_server_status( waiters.wait_for_server_status(
self.manager.servers_client, self.manager.servers_client,
serv['id'], 'ACTIVE') serv['id'], 'ACTIVE')
@ -205,13 +209,14 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
tenant_id=self.tenant_id, tenant_id=self.tenant_id,
**rule) **rule)
def start_web_servers(self): def start_web_servers(self, server_list=None):
"""Start predefined servers: """Start predefined servers:
1. SSH to the instance 1. SSH to the instance
2. Start http backends listening on port 80 2. Start http backends listening on port 80
""" """
for server in [self.server1, self.server2]: server_list = server_list or self.rr_server_list
for server in server_list:
fip = self.create_floatingip_for_server( fip = self.create_floatingip_for_server(
server, self.public_network_id, server, self.public_network_id,
client_mgr=self.manager) client_mgr=self.manager)
@ -220,7 +225,7 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
self.start_web_server(server, server_fip, server['name']) self.start_web_server(server, server_fip, server['name'])
# need to wait for web server to be able to response # need to wait for web server to be able to response
time.sleep(self.web_service_start_delay) time.sleep(self.web_service_start_delay)
for server in [self.server1, self.server2]: for server in server_list:
server_name = server['name'] server_name = server['name']
fip = server['_floating_ip'] fip = server['_floating_ip']
web_fip = fip['floating_ip_address'] web_fip = fip['floating_ip_address']
@ -248,10 +253,9 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
with tempfile.NamedTemporaryFile() as key: with tempfile.NamedTemporaryFile() as key:
key.write(private_key) key.write(private_key)
key.flush() key.flush()
test_v1_lbaas_basic_ops.copy_file_to_host( dmgr.copy_file_to_host(script.name,
script.name, "/tmp/script",
"/tmp/script", server_fip, username, key.name)
server_fip, username, key.name)
# Start netcat # Start netcat
start_server = ('while true; do ' start_server = ('while true; do '
@ -294,7 +298,7 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
self.healthmonitor = ( self.healthmonitor = (
self.health_monitors_client.create_health_monitor( self.health_monitors_client.create_health_monitor(
pool_id=pool_id, type=self.protocol_type, pool_id=pool_id, type=self.hm_type,
delay=self.hm_delay, max_retries=self.hm_max_retries, delay=self.hm_delay, max_retries=self.hm_max_retries,
timeout=self.hm_timeout)) timeout=self.hm_timeout))
self.wait_for_load_balancer_status(lb_id) self.wait_for_load_balancer_status(lb_id)
@ -324,20 +328,30 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
port_id=self.loadbalancer['vip_port_id'], port_id=self.loadbalancer['vip_port_id'],
client_mgr=self.manager) client_mgr=self.manager)
self.vip_ip_address = self.vip_fip['floating_ip_address'] self.vip_ip_address = self.vip_fip['floating_ip_address']
time.sleep(1.0) for x in range(1, 8):
self.send_request(self.vip_ip_address) time.sleep(2)
resp = self.send_request(self.vip_ip_address)
if resp:
break
LOG.debug('#%d LBaaS-VIP get NO response from its members', x)
return self.vip_ip_address return self.vip_ip_address
def check_project_lbaas(self): def do_http_request(self, start_path='', send_counts=None):
statuses = self.load_balancers_client.show_load_balancer_status_tree( statuses = self.load_balancers_client.show_load_balancer_status_tree(
self.loadbalancer['id']) self.loadbalancer['id'])
statuses = statuses.get('statuses', statuses) statuses = statuses.get('statuses', statuses)
self.http_cnt = {} self.http_cnt = {}
http = urllib3.PoolManager(retries=10) http = urllib3.PoolManager(retries=10)
url_path = "http://{0}/".format(self.vip_ip_address) send_counts = send_counts or self.poke_counters
for x in range(self.poke_counters): send_counts = (send_counts * 2) / 2
url_path = "http://{0}/{1}".format(self.vip_ip_address, start_path)
for x in range(send_counts):
resp = http.request('GET', url_path) resp = http.request('GET', url_path)
self.count_response(resp.data.strip()) self.count_response(resp.data.strip())
return self.http_cnt
def check_project_lbaas(self):
self.do_http_request(send_counts=self.poke_counters)
# should response from 2 servers # should response from 2 servers
self.assertEqual(2, len(self.http_cnt)) self.assertEqual(2, len(self.http_cnt))
# ROUND_ROUBIN, so equal counts # ROUND_ROUBIN, so equal counts
@ -351,10 +365,25 @@ class TestLBaasRoundRobinOps(dmgr.TopoDeployScenarioManager):
else: else:
self.http_cnt[response] = 1 self.http_cnt[response] = 1
class TestLBaasRoundRobinOps(LBaasRoundRobinBaseTest):
"""This test checks basic load balancer V2 ROUND-ROBIN operation.
The following is the scenario outline:
1. Create network with exclusive router, and 2 servers
2. SSH to each instance and start web server
3. Create a load balancer with 1 listener, 1 pool, 1 healthmonitor
and 2 members and with ROUND_ROBIN algorithm.
4. Associate loadbalancer's vip_address with a floating ip
5. Send NUM requests to vip's floating ip and check that they are shared
between the two servers.
"""
@test.idempotent_id('077d2a5c-4938-448f-a80f-8e65f5cc49d7') @test.idempotent_id('077d2a5c-4938-448f-a80f-8e65f5cc49d7')
@test.services('compute', 'network') @test.services('compute', 'network')
def test_lbaas_round_robin_ops(self): def test_lbaas_round_robin_ops(self):
self.create_lbaas_networks() self.create_lbaas_networks()
self.start_web_servers() self.start_web_servers(self.rr_server_list)
self.create_project_lbaas() self.create_project_lbaas()
self.check_project_lbaas() self.check_project_lbaas()