Determine security group API dynamically

This commit removes enable_security_group from local_settings.py
and determines which security group API should be used (nova or
neutron).

Closes-Bug: #1227804

As described in bug 1203413, there is a case where Nova security group
with Neutron driver causes a problem. The type of 'name' attribute in
add_security_group_to_instance and remove_security_group_from_instance
depends on the backend, integer for nova security group driver and
UUID for quantum security group driver to make it work as expected.

enable_security_group config parameter produces a situation where
Nova security group with Neutron driver. We can avoid this situation
by removing this parameter when using Horizon.

Change-Id: I713c6ad166e142929f0a708e93a8fedb0de48640
This commit is contained in:
Akihiro MOTOKI 2013-09-20 11:27:06 +09:00
parent 2ed62bb2e5
commit 6f1307e472
8 changed files with 96 additions and 27 deletions

View File

@ -37,12 +37,8 @@ class NetworkClient(object):
else: else:
self.floating_ips = nova.FloatingIpManager(request) self.floating_ips = nova.FloatingIpManager(request)
# Not all qunantum plugins support security group, if (neutron_enabled and
# so we have enable_security_group configuration parameter. neutron.is_security_group_extension_supported(request)):
neutron_sg_enabled = getattr(settings,
'OPENSTACK_NEUTRON_NETWORK',
{}).get('enable_security_group', True)
if neutron_enabled and neutron_sg_enabled:
self.secgroups = neutron.SecurityGroupManager(request) self.secgroups = neutron.SecurityGroupManager(request)
else: else:
self.secgroups = nova.SecurityGroupManager(request) self.secgroups = nova.SecurityGroupManager(request)
@ -87,6 +83,10 @@ def floating_ip_target_get_by_instance(request, instance_id):
instance_id) instance_id)
def floating_ip_simple_associate_supported(request):
return NetworkClient(request).floating_ips.is_simple_associate_supported()
def security_group_list(request): def security_group_list(request):
return NetworkClient(request).secgroups.list() return NetworkClient(request).secgroups.list()

View File

@ -27,6 +27,8 @@ from django.conf import settings # noqa
from django.utils.datastructures import SortedDict # noqa from django.utils.datastructures import SortedDict # noqa
from django.utils.translation import ugettext_lazy as _ # noqa from django.utils.translation import ugettext_lazy as _ # noqa
from horizon.utils.memoized import memoized # noqa
from openstack_dashboard.api import base from openstack_dashboard.api import base
from openstack_dashboard.api import network_base from openstack_dashboard.api import network_base
from openstack_dashboard.api import nova from openstack_dashboard.api import nova
@ -714,6 +716,12 @@ def tenant_quota_update(request, tenant_id, **kwargs):
return neutronclient(request).update_quota(tenant_id, quotas) return neutronclient(request).update_quota(tenant_id, quotas)
def agent_list(request):
agents = neutronclient(request).list_agents()
return [Agent(a) for a in agents['agents']]
@memoized
def list_extensions(request): def list_extensions(request):
extensions_list = neutronclient(request).list_extensions() extensions_list = neutronclient(request).list_extensions()
if 'extensions' in extensions_list: if 'extensions' in extensions_list:
@ -722,11 +730,7 @@ def list_extensions(request):
return {} return {}
def agent_list(request): @memoized
agents = neutronclient(request).list_agents()
return [Agent(a) for a in agents['agents']]
def is_extension_supported(request, extension_alias): def is_extension_supported(request, extension_alias):
extensions = list_extensions(request) extensions = list_extensions(request)
@ -737,15 +741,20 @@ def is_extension_supported(request, extension_alias):
return False return False
@memoized
def is_quotas_extension_supported(request): def is_quotas_extension_supported(request):
network_config = getattr(settings, 'OPENSTACK_NEUTRON_NETWORK', {}) network_config = getattr(settings, 'OPENSTACK_NEUTRON_NETWORK', {})
if network_config.get('enable_quotas', False) and \ if (network_config.get('enable_quotas', False) and
is_extension_supported(request, 'quotas'): is_extension_supported(request, 'quotas')):
return True return True
else: else:
return False return False
def is_security_group_extension_supported(request):
return is_extension_supported(request, 'security-group')
# Using this mechanism till a better plugin/sub-plugin detection # Using this mechanism till a better plugin/sub-plugin detection
# mechanism is available. # mechanism is available.
# Using local_settings to detect if the "router" dashboard # Using local_settings to detect if the "router" dashboard

View File

@ -353,8 +353,7 @@ class AssociateIP(tables.LinkAction):
classes = ("ajax-modal", "btn-associate") classes = ("ajax-modal", "btn-associate")
def allowed(self, request, instance): def allowed(self, request, instance):
fip = api.network.NetworkClient(request).floating_ips if api.network.floating_ip_simple_associate_supported(request):
if fip.is_simple_associate_supported():
return False return False
return not is_deleting(instance) return not is_deleting(instance)
@ -373,8 +372,7 @@ class SimpleAssociateIP(tables.Action):
classes = ("btn-associate-simple",) classes = ("btn-associate-simple",)
def allowed(self, request, instance): def allowed(self, request, instance):
fip = api.network.NetworkClient(request).floating_ips if not api.network.floating_ip_simple_associate_supported(request):
if not fip.is_simple_associate_supported():
return False return False
return not is_deleting(instance) return not is_deleting(instance)

View File

@ -49,7 +49,10 @@ class InstanceTests(test.TestCase):
@test.create_stubs({api.nova: ('flavor_list', @test.create_stubs({api.nova: ('flavor_list',
'server_list', 'server_list',
'tenant_absolute_limits', 'tenant_absolute_limits',
'extension_supported',)}) 'extension_supported',),
api.network:
('floating_ip_simple_associate_supported',),
})
def test_index(self): def test_index(self):
api.nova.extension_supported('AdminActions', api.nova.extension_supported('AdminActions',
IsA(http.HttpRequest)) \ IsA(http.HttpRequest)) \
@ -61,6 +64,8 @@ class InstanceTests(test.TestCase):
.AndReturn([self.servers.list(), False]) .AndReturn([self.servers.list(), False])
api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \ api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \
.MultipleTimes().AndReturn(self.limits['absolute']) .MultipleTimes().AndReturn(self.limits['absolute'])
api.network.floating_ip_simple_associate_supported(
IsA(http.HttpRequest)).MultipleTimes().AndReturn(True)
self.mox.ReplayAll() self.mox.ReplayAll()
@ -93,7 +98,10 @@ class InstanceTests(test.TestCase):
'server_list', 'server_list',
'flavor_get', 'flavor_get',
'tenant_absolute_limits', 'tenant_absolute_limits',
'extension_supported',)}) 'extension_supported',),
api.network:
('floating_ip_simple_associate_supported',),
})
def test_index_flavor_list_exception(self): def test_index_flavor_list_exception(self):
servers = self.servers.list() servers = self.servers.list()
flavors = self.flavors.list() flavors = self.flavors.list()
@ -111,6 +119,8 @@ class InstanceTests(test.TestCase):
AndReturn(full_flavors[server.flavor["id"]]) AndReturn(full_flavors[server.flavor["id"]])
api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \ api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \
.MultipleTimes().AndReturn(self.limits['absolute']) .MultipleTimes().AndReturn(self.limits['absolute'])
api.network.floating_ip_simple_associate_supported(
IsA(http.HttpRequest)).MultipleTimes().AndReturn(True)
self.mox.ReplayAll() self.mox.ReplayAll()
@ -125,7 +135,10 @@ class InstanceTests(test.TestCase):
'server_list', 'server_list',
'flavor_get', 'flavor_get',
'tenant_absolute_limits', 'tenant_absolute_limits',
'extension_supported',)}) 'extension_supported',),
api.network:
('floating_ip_simple_associate_supported',),
})
def test_index_flavor_get_exception(self): def test_index_flavor_get_exception(self):
servers = self.servers.list() servers = self.servers.list()
flavors = self.flavors.list() flavors = self.flavors.list()
@ -146,6 +159,8 @@ class InstanceTests(test.TestCase):
AndRaise(self.exceptions.nova) AndRaise(self.exceptions.nova)
api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \ api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \
.MultipleTimes().AndReturn(self.limits['absolute']) .MultipleTimes().AndReturn(self.limits['absolute'])
api.network.floating_ip_simple_associate_supported(
IsA(http.HttpRequest)).MultipleTimes().AndReturn(True)
self.mox.ReplayAll() self.mox.ReplayAll()
@ -1588,7 +1603,10 @@ class InstanceTests(test.TestCase):
@test.create_stubs({api.nova: ('flavor_list', 'server_list', @test.create_stubs({api.nova: ('flavor_list', 'server_list',
'tenant_absolute_limits', 'tenant_absolute_limits',
'extension_supported',)}) 'extension_supported',),
api.network:
('floating_ip_simple_associate_supported',),
})
def test_launch_button_disabled_when_quota_exceeded(self): def test_launch_button_disabled_when_quota_exceeded(self):
limits = self.limits['absolute'] limits = self.limits['absolute']
limits['totalInstancesUsed'] = limits['maxTotalInstances'] limits['totalInstancesUsed'] = limits['maxTotalInstances']
@ -1603,6 +1621,8 @@ class InstanceTests(test.TestCase):
.AndReturn([self.servers.list(), False]) .AndReturn([self.servers.list(), False])
api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \ api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \
.MultipleTimes().AndReturn(limits) .MultipleTimes().AndReturn(limits)
api.network.floating_ip_simple_associate_supported(
IsA(http.HttpRequest)).MultipleTimes().AndReturn(True)
self.mox.ReplayAll() self.mox.ReplayAll()
@ -1621,7 +1641,10 @@ class InstanceTests(test.TestCase):
@test.create_stubs({api.nova: ('flavor_list', 'server_list', @test.create_stubs({api.nova: ('flavor_list', 'server_list',
'tenant_absolute_limits', 'tenant_absolute_limits',
'extension_supported',)}) 'extension_supported',),
api.network:
('floating_ip_simple_associate_supported',),
})
def test_index_options_after_migrate(self): def test_index_options_after_migrate(self):
server = self.servers.first() server = self.servers.first()
server.status = "VERIFY_RESIZE" server.status = "VERIFY_RESIZE"
@ -1635,6 +1658,8 @@ class InstanceTests(test.TestCase):
.AndReturn([self.servers.list(), False]) .AndReturn([self.servers.list(), False])
api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \ api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \
.MultipleTimes().AndReturn(self.limits['absolute']) .MultipleTimes().AndReturn(self.limits['absolute'])
api.network.floating_ip_simple_associate_supported(
IsA(http.HttpRequest)).MultipleTimes().AndReturn(True)
self.mox.ReplayAll() self.mox.ReplayAll()

View File

@ -160,7 +160,6 @@ OPENSTACK_NEUTRON_NETWORK = {
'enable_lb': False, 'enable_lb': False,
'enable_firewall': False, 'enable_firewall': False,
'enable_quotas': True, 'enable_quotas': True,
'enable_security_group': True,
'enable_vpn': False, 'enable_vpn': False,
# The profile_support option is used to detect if an external router can be # The profile_support option is used to detect if an external router can be
# configured via the dashboard. When using specific plugins the # configured via the dashboard. When using specific plugins the

View File

@ -27,6 +27,43 @@ from openstack_dashboard import api
from openstack_dashboard.test import helpers as test from openstack_dashboard.test import helpers as test
class NetworkClientTestCase(test.APITestCase):
def test_networkclient_no_neutron(self):
self.mox.StubOutWithMock(api.base, 'is_service_enabled')
api.base.is_service_enabled(IsA(http.HttpRequest), 'network') \
.AndReturn(False)
self.mox.ReplayAll()
nc = api.network.NetworkClient(self.request)
self.assertIsInstance(nc.floating_ips, api.nova.FloatingIpManager)
self.assertIsInstance(nc.secgroups, api.nova.SecurityGroupManager)
def test_networkclient_neutron(self):
self.mox.StubOutWithMock(api.base, 'is_service_enabled')
api.base.is_service_enabled(IsA(http.HttpRequest), 'network') \
.AndReturn(True)
self.neutronclient = self.stub_neutronclient()
self.neutronclient.list_extensions() \
.AndReturn({'extensions': self.api_extensions.list()})
self.mox.ReplayAll()
nc = api.network.NetworkClient(self.request)
self.assertIsInstance(nc.floating_ips, api.neutron.FloatingIpManager)
self.assertIsInstance(nc.secgroups, api.neutron.SecurityGroupManager)
def test_networkclient_neutron_with_nova_security_group(self):
self.mox.StubOutWithMock(api.base, 'is_service_enabled')
api.base.is_service_enabled(IsA(http.HttpRequest), 'network') \
.AndReturn(True)
self.neutronclient = self.stub_neutronclient()
self.neutronclient.list_extensions().AndReturn({'extensions': []})
self.mox.ReplayAll()
nc = api.network.NetworkClient(self.request)
self.assertIsInstance(nc.floating_ips, api.neutron.FloatingIpManager)
self.assertIsInstance(nc.secgroups, api.nova.SecurityGroupManager)
class NetworkApiNovaTestBase(test.APITestCase): class NetworkApiNovaTestBase(test.APITestCase):
def setUp(self): def setUp(self):
super(NetworkApiNovaTestBase, self).setUp() super(NetworkApiNovaTestBase, self).setUp()
@ -156,6 +193,8 @@ class NetworkApiNeutronTestBase(test.APITestCase):
api.base.is_service_enabled(IsA(http.HttpRequest), 'network') \ api.base.is_service_enabled(IsA(http.HttpRequest), 'network') \
.AndReturn(True) .AndReturn(True)
self.qclient = self.stub_neutronclient() self.qclient = self.stub_neutronclient()
self.qclient.list_extensions() \
.AndReturn({'extensions': self.api_extensions.list()})
class NetworkApiNeutronSecurityGroupTests(NetworkApiNeutronTestBase): class NetworkApiNeutronSecurityGroupTests(NetworkApiNeutronTestBase):

View File

@ -274,7 +274,7 @@ class NeutronApiTests(test.APITestCase):
def test_is_extension_supported(self): def test_is_extension_supported(self):
neutronclient = self.stub_neutronclient() neutronclient = self.stub_neutronclient()
neutronclient.list_extensions().MultipleTimes() \ neutronclient.list_extensions().MultipleTimes() \
.AndReturn(self.api_extensions.first()) .AndReturn({'extensions': self.api_extensions.list()})
self.mox.ReplayAll() self.mox.ReplayAll()
self.assertTrue( self.assertTrue(

View File

@ -534,9 +534,8 @@ def data(TEST):
extension_2 = {"name": "Quota management support", extension_2 = {"name": "Quota management support",
"alias": "quotas", "alias": "quotas",
"description": "Expose functions for quotas management"} "description": "Expose functions for quotas management"}
extensions = {} TEST.api_extensions.add(extension_1)
extensions['extensions'] = [extension_1, extension_2] TEST.api_extensions.add(extension_2)
TEST.api_extensions.add(extensions)
#------------------------------------------------------------ #------------------------------------------------------------
# 1st agent # 1st agent