diff --git a/openstack_dashboard/dashboards/project/access_and_security/floating_ips/tests.py b/openstack_dashboard/dashboards/project/access_and_security/floating_ips/tests.py
index b189195c58..e6c85215cc 100644
--- a/openstack_dashboard/dashboards/project/access_and_security/floating_ips/tests.py
+++ b/openstack_dashboard/dashboards/project/access_and_security/floating_ips/tests.py
@@ -25,8 +25,6 @@ from mox3.mox import IsA # noqa
import six
from openstack_dashboard import api
-from openstack_dashboard.dashboards.project.access_and_security \
- .floating_ips import tables
from openstack_dashboard.test import helpers as test
from openstack_dashboard.usage import quotas
@@ -214,6 +212,68 @@ class FloatingIpViewTests(test.TestCase):
res = self.client.post(INDEX_URL, {"action": action})
self.assertRedirectsNoFollow(res, INDEX_URL)
+ @test.create_stubs({api.network: ('floating_ip_supported',
+ 'tenant_floating_ip_list',
+ 'security_group_list',
+ 'floating_ip_pools_list',),
+ api.nova: ('keypair_list',
+ 'server_list',),
+ quotas: ('tenant_quota_usages',),
+ api.base: ('is_service_enabled',)})
+ def test_allocate_button_attributes(self):
+ keypairs = self.keypairs.list()
+ floating_ips = self.floating_ips.list()
+ floating_pools = self.pools.list()
+ quota_data = self.quota_usages.first()
+ quota_data['floating_ips']['available'] = 10
+ sec_groups = self.security_groups.list()
+
+ api.network.floating_ip_supported(
+ IsA(http.HttpRequest)) \
+ .AndReturn(True)
+ api.network.tenant_floating_ip_list(
+ IsA(http.HttpRequest)) \
+ .AndReturn(floating_ips)
+ api.network.security_group_list(
+ IsA(http.HttpRequest)).MultipleTimes()\
+ .AndReturn(sec_groups)
+ api.network.floating_ip_pools_list(
+ IsA(http.HttpRequest)) \
+ .AndReturn(floating_pools)
+ api.nova.keypair_list(
+ IsA(http.HttpRequest)) \
+ .AndReturn(keypairs)
+ api.nova.server_list(
+ IsA(http.HttpRequest)) \
+ .AndReturn([self.servers.list(), False])
+ quotas.tenant_quota_usages(
+ IsA(http.HttpRequest)).MultipleTimes() \
+ .AndReturn(quota_data)
+
+ api.base.is_service_enabled(
+ IsA(http.HttpRequest),
+ 'network').MultipleTimes() \
+ .AndReturn(True)
+ api.base.is_service_enabled(
+ IsA(http.HttpRequest),
+ 'ec2').MultipleTimes() \
+ .AndReturn(False)
+
+ self.mox.ReplayAll()
+
+ res = self.client.get(INDEX_URL +
+ "?tab=access_security_tabs__floating_ips_tab")
+
+ allocate_action = self.getAndAssertTableAction(res, 'floating_ips',
+ 'allocate')
+ self.assertEqual(set(['ajax-modal']), set(allocate_action.classes))
+ self.assertEqual('Allocate IP To Project',
+ six.text_type(allocate_action.verbose_name))
+ self.assertEqual(None, allocate_action.policy_rules)
+
+ url = 'horizon:project:access_and_security:floating_ips:allocate'
+ self.assertEqual(url, allocate_action.url)
+
@test.create_stubs({api.network: ('floating_ip_supported',
'tenant_floating_ip_list',
'security_group_list',
@@ -266,19 +326,12 @@ class FloatingIpViewTests(test.TestCase):
res = self.client.get(INDEX_URL +
"?tab=access_security_tabs__floating_ips_tab")
- allocate_link = tables.AllocateIP()
- url = allocate_link.get_link_url()
- classes = (list(allocate_link.get_default_classes())
- + list(allocate_link.classes))
- link_name = "%s (%s)" % (six.text_type(allocate_link.verbose_name),
- "Quota exceeded")
- expected_string = (""
- ""
- "%s"
- % (url, link_name, " ".join(classes), link_name))
- self.assertContains(res, expected_string, html=True,
- msg_prefix="The create button is not disabled")
+ allocate_action = self.getAndAssertTableAction(res, 'floating_ips',
+ 'allocate')
+ self.assertTrue('disabled' in allocate_action.classes,
+ 'The create button should be disabled')
+ self.assertEqual('Allocate IP To Project (Quota exceeded)',
+ six.text_type(allocate_action.verbose_name))
class FloatingIpNeutronViewTests(FloatingIpViewTests):
diff --git a/openstack_dashboard/dashboards/project/access_and_security/tests.py b/openstack_dashboard/dashboards/project/access_and_security/tests.py
index f6f374a76b..046c74fd1e 100644
--- a/openstack_dashboard/dashboards/project/access_and_security/tests.py
+++ b/openstack_dashboard/dashboards/project/access_and_security/tests.py
@@ -27,8 +27,6 @@ from horizon.workflows import views
from openstack_dashboard import api
from openstack_dashboard.dashboards.project.access_and_security \
import api_access
-from openstack_dashboard.dashboards.project.access_and_security \
- .security_groups import tables
from openstack_dashboard.test import helpers as test
from openstack_dashboard.usage import quotas
@@ -163,6 +161,70 @@ class SecurityGroupTabTests(test.TestCase):
def setUp(self):
super(SecurityGroupTabTests, self).setUp()
+ @test.create_stubs({api.network: ('floating_ip_supported',
+ 'tenant_floating_ip_list',
+ 'security_group_list',
+ 'floating_ip_pools_list',),
+ api.nova: ('keypair_list',
+ 'server_list',),
+ quotas: ('tenant_quota_usages',),
+ api.base: ('is_service_enabled',)})
+ def test_create_button_attributes(self):
+ keypairs = self.keypairs.list()
+ floating_ips = self.floating_ips.list()
+ floating_pools = self.pools.list()
+ sec_groups = self.security_groups.list()
+ quota_data = self.quota_usages.first()
+ quota_data['security_groups']['available'] = 10
+
+ api.network.floating_ip_supported(
+ IsA(http.HttpRequest)) \
+ .AndReturn(True)
+ api.network.tenant_floating_ip_list(
+ IsA(http.HttpRequest)) \
+ .AndReturn(floating_ips)
+ api.network.floating_ip_pools_list(
+ IsA(http.HttpRequest)) \
+ .AndReturn(floating_pools)
+ api.network.security_group_list(
+ IsA(http.HttpRequest)) \
+ .AndReturn(sec_groups)
+ api.nova.keypair_list(
+ IsA(http.HttpRequest)) \
+ .AndReturn(keypairs)
+ api.nova.server_list(
+ IsA(http.HttpRequest)) \
+ .AndReturn([self.servers.list(), False])
+ quotas.tenant_quota_usages(
+ IsA(http.HttpRequest)).MultipleTimes() \
+ .AndReturn(quota_data)
+
+ api.base.is_service_enabled(
+ IsA(http.HttpRequest), 'network').MultipleTimes() \
+ .AndReturn(True)
+ api.base.is_service_enabled(
+ IsA(http.HttpRequest), 'ec2').MultipleTimes() \
+ .AndReturn(False)
+
+ self.mox.ReplayAll()
+
+ res = self.client.get(INDEX_URL +
+ "?tab=access_security_tabs__security_groups_tab")
+
+ security_groups = res.context['security_groups_table'].data
+ self.assertItemsEqual(security_groups, self.security_groups.list())
+
+ create_action = self.getAndAssertTableAction(res, 'security_groups',
+ 'create')
+
+ self.assertEqual('Create Security Group',
+ six.text_type(create_action.verbose_name))
+ self.assertEqual(None, create_action.policy_rules)
+ self.assertEqual(set(['ajax-modal']), set(create_action.classes))
+
+ url = 'horizon:project:access_and_security:security_groups:create'
+ self.assertEqual(url, create_action.url)
+
@test.create_stubs({api.network: ('floating_ip_supported',
'tenant_floating_ip_list',
'security_group_list',
@@ -217,18 +279,10 @@ class SecurityGroupTabTests(test.TestCase):
security_groups = res.context['security_groups_table'].data
self.assertItemsEqual(security_groups, self.security_groups.list())
- create_link = tables.CreateGroup()
- url = create_link.get_link_url()
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- link_name = "%s (%s)" % (six.text_type(create_link.verbose_name),
- "Quota exceeded")
- expected_string = "" \
- "%s" \
- % (url, link_name, " ".join(classes), link_name)
- self.assertContains(res, expected_string, html=True,
- msg_prefix="The create button is not disabled")
+ create_action = self.getAndAssertTableAction(res, 'security_groups',
+ 'create')
+ self.assertTrue('disabled' in create_action.classes,
+ 'The create button should be disabled')
def test_create_button_disabled_when_quota_exceeded_neutron_disabled(self):
self._test_create_button_disabled_when_quota_exceeded(False)
diff --git a/openstack_dashboard/dashboards/project/instances/tests.py b/openstack_dashboard/dashboards/project/instances/tests.py
index 637b6dfe4c..7da47eda41 100644
--- a/openstack_dashboard/dashboards/project/instances/tests.py
+++ b/openstack_dashboard/dashboards/project/instances/tests.py
@@ -28,7 +28,6 @@ from django.core.urlresolvers import reverse
from django.forms import widgets
from django import http
import django.test
-from django.utils import encoding
from django.utils.http import urlencode
from mox3.mox import IgnoreArg # noqa
from mox3.mox import IsA # noqa
@@ -3587,6 +3586,54 @@ class InstanceTests(helpers.TestCase):
self.test_launch_form_instance_non_int_volume_size(
test_with_profile=True)
+ @helpers.create_stubs({
+ api.nova: ('flavor_list', 'server_list', 'tenant_absolute_limits',
+ 'extension_supported',),
+ api.glance: ('image_list_detailed',),
+ api.network: ('floating_ip_simple_associate_supported',
+ 'floating_ip_supported',
+ 'servers_update_addresses',),
+ })
+ def test_launch_button_attributes(self):
+ servers = self.servers.list()
+ limits = self.limits['absolute']
+ limits['totalInstancesUsed'] = 0
+
+ api.nova.extension_supported('AdminActions',
+ IsA(http.HttpRequest)) \
+ .MultipleTimes().AndReturn(True)
+ api.nova.extension_supported('Shelve', IsA(http.HttpRequest)) \
+ .MultipleTimes().AndReturn(True)
+ api.nova.flavor_list(IsA(http.HttpRequest)) \
+ .AndReturn(self.flavors.list())
+ api.glance.image_list_detailed(IgnoreArg()) \
+ .AndReturn((self.images.list(), False, False))
+ search_opts = {'marker': None, 'paginate': True}
+ api.nova.server_list(IsA(http.HttpRequest), search_opts=search_opts) \
+ .AndReturn([servers, False])
+ api.network.servers_update_addresses(IsA(http.HttpRequest), servers)
+ api.nova.tenant_absolute_limits(IsA(http.HttpRequest), reserved=True) \
+ .MultipleTimes().AndReturn(limits)
+ api.network.floating_ip_supported(IsA(http.HttpRequest)) \
+ .MultipleTimes().AndReturn(True)
+ api.network.floating_ip_simple_associate_supported(
+ IsA(http.HttpRequest)).MultipleTimes().AndReturn(True)
+
+ self.mox.ReplayAll()
+
+ tables.LaunchLink()
+ res = self.client.get(INDEX_URL)
+
+ launch_action = self.getAndAssertTableAction(res, 'instances',
+ 'launch')
+
+ self.assertEqual(set(['ajax-modal', 'ajax-update', 'btn-launch']),
+ set(launch_action.classes))
+ self.assertEqual('Launch Instance', launch_action.verbose_name)
+ self.assertEqual('horizon:project:instances:launch', launch_action.url)
+ self.assertEqual((('compute', 'compute:create'),),
+ launch_action.policy_rules)
+
@helpers.create_stubs({
api.nova: ('flavor_list', 'server_list', 'tenant_absolute_limits',
'extension_supported',),
@@ -3622,27 +3669,16 @@ class InstanceTests(helpers.TestCase):
self.mox.ReplayAll()
- launch = tables.LaunchLink()
- url = launch.get_link_url()
- classes = list(launch.get_default_classes()) + list(launch.classes)
- link_name = "%s (%s)" % (six.text_type(launch.verbose_name),
- "Quota exceeded")
-
+ tables.LaunchLink()
res = self.client.get(INDEX_URL)
- if django.VERSION < (1, 8, 0):
- resp_charset = res._charset
- else:
- resp_charset = res.charset
- expected_string = encoding.smart_str(u'''
-
- %s
- ''' % (url, link_name, " ".join(classes), link_name), resp_charset)
- self.assertContains(res, expected_string, html=True,
- msg_prefix="The launch button is not disabled")
+ launch_action = self.getAndAssertTableAction(
+ res, 'instances', 'launch')
+
+ self.assertTrue('disabled' in launch_action.classes,
+ 'The launch button should be disabled')
+ self.assertEqual('Launch Instance (Quota exceeded)',
+ six.text_type(launch_action.verbose_name))
@helpers.create_stubs({api.glance: ('image_list_detailed',),
api.neutron: ('network_list',),
diff --git a/openstack_dashboard/dashboards/project/networks/tests.py b/openstack_dashboard/dashboards/project/networks/tests.py
index 4b50302d85..7e433bd371 100644
--- a/openstack_dashboard/dashboards/project/networks/tests.py
+++ b/openstack_dashboard/dashboards/project/networks/tests.py
@@ -19,10 +19,9 @@ from django.utils.html import escape
from horizon.workflows import views
from mox3.mox import IsA # noqa
+import six
from openstack_dashboard import api
-from openstack_dashboard.dashboards.project.networks.subnets import tables\
- as subnets_tables
from openstack_dashboard.dashboards.project.networks import tables\
as networks_tables
from openstack_dashboard.dashboards.project.networks import workflows
@@ -2013,7 +2012,8 @@ class NetworkSubnetTests(test.TestCase):
class NetworkViewTests(test.TestCase, NetworkStubMixin):
def _test_create_button_shown_when_quota_disabled(
- self, expected_string):
+ self,
+ find_button_fn):
# if quota_data doesnt contain a networks|subnets|routers key or
# these keys are empty dicts, its disabled
quota_data = self.neutron_quota_usages.first()
@@ -2033,11 +2033,14 @@ class NetworkViewTests(test.TestCase, NetworkStubMixin):
networks = res.context['networks_table'].data
self.assertItemsEqual(networks, self.networks.list())
- self.assertContains(res, expected_string, True, html=True,
- msg_prefix="The enabled create button not shown")
+
+ button = find_button_fn(res)
+ self.assertFalse('disabled' in button.classes,
+ "The create button should not be disabled")
+ return button
def _test_create_button_disabled_when_quota_exceeded(
- self, expected_string, network_quota=5, subnet_quota=5):
+ self, find_button_fn, network_quota=5, subnet_quota=5, ):
quota_data = self.neutron_quota_usages.first()
@@ -2056,69 +2059,55 @@ class NetworkViewTests(test.TestCase, NetworkStubMixin):
networks = res.context['networks_table'].data
self.assertItemsEqual(networks, self.networks.list())
- self.assertContains(res, expected_string, True, html=True,
- msg_prefix="The create button is not disabled")
+
+ button = find_button_fn(res)
+ self.assertTrue('disabled' in button.classes,
+ "The create button should be disabled")
+ return button
@test.create_stubs({api.neutron: ('network_list',),
quotas: ('tenant_quota_usages',)})
def test_network_create_button_disabled_when_quota_exceeded_index(self):
- create_link = networks_tables.CreateNetwork()
- url = create_link.get_link_url()
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- link_name = "%s (%s)" % (create_link.verbose_name, "Quota exceeded")
- expected_string = "" \
- "%s" \
- % (url, link_name, " ".join(classes), link_name)
- self._test_create_button_disabled_when_quota_exceeded(expected_string,
- network_quota=0
- )
+ networks_tables.CreateNetwork()
+
+ def _find_net_button(res):
+ return self.getAndAssertTableAction(res, 'networks', 'create')
+ self._test_create_button_disabled_when_quota_exceeded(_find_net_button,
+ network_quota=0)
@test.create_stubs({api.neutron: ('network_list',),
quotas: ('tenant_quota_usages',)})
def test_subnet_create_button_disabled_when_quota_exceeded_index(self):
network_id = self.networks.first().id
- create_link = networks_tables.CreateSubnet()
- url = reverse(create_link.get_link_url(), args=[network_id])
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- link_name = "%s (%s)" % (create_link.verbose_name, "Quota exceeded")
- expected_string = "%s" \
- % (url, " ".join(classes), network_id, link_name)
- self._test_create_button_disabled_when_quota_exceeded(expected_string,
- subnet_quota=0
- )
+ networks_tables.CreateSubnet()
+
+ def _find_subnet_button(res):
+ return self.getAndAssertTableRowAction(res, 'networks',
+ 'subnet', network_id)
+
+ self._test_create_button_disabled_when_quota_exceeded(
+ _find_subnet_button, subnet_quota=0)
@test.create_stubs({api.neutron: ('network_list',),
quotas: ('tenant_quota_usages',)})
def test_network_create_button_shown_when_quota_disabled_index(self):
# if quota_data doesnt contain a networks["available"] key its disabled
- create_link = networks_tables.CreateNetwork()
- url = create_link.get_link_url()
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- expected_string = "" \
- "%s" \
- % (url, create_link.verbose_name, " ".join(classes),
- create_link.verbose_name)
- self._test_create_button_shown_when_quota_disabled(expected_string)
+ networks_tables.CreateNetwork()
+ self._test_create_button_shown_when_quota_disabled(
+ lambda res: self.getAndAssertTableAction(res, 'networks', 'create')
+ )
@test.create_stubs({api.neutron: ('network_list',),
quotas: ('tenant_quota_usages',)})
def test_subnet_create_button_shown_when_quota_disabled_index(self):
# if quota_data doesnt contain a subnets["available"] key, its disabled
network_id = self.networks.first().id
- create_link = networks_tables.CreateSubnet()
- url = reverse(create_link.get_link_url(), args=[network_id])
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- expected_string = "%s" \
- % (url, " ".join(classes), network_id, create_link.verbose_name)
- self._test_create_button_shown_when_quota_disabled(expected_string)
+
+ def _find_subnet_button(res):
+ return self.getAndAssertTableRowAction(res, 'networks',
+ 'subnet', network_id)
+
+ self._test_create_button_shown_when_quota_disabled(_find_subnet_button)
@test.create_stubs({api.neutron: ('network_get',
'subnet_list',
@@ -2155,17 +2144,65 @@ class NetworkViewTests(test.TestCase, NetworkStubMixin):
subnets = res.context['subnets_table'].data
self.assertItemsEqual(subnets, self.subnets.list())
- class FakeTable(object):
- kwargs = {'network_id': network_id}
- create_link = subnets_tables.CreateSubnet()
- create_link.table = FakeTable()
- url = create_link.get_link_url()
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- link_name = "%s (%s)" % (create_link.verbose_name, "Quota exceeded")
- expected_string = "" \
- "%s" \
- % (url, link_name, " ".join(classes), link_name)
- self.assertContains(res, expected_string, html=True,
- msg_prefix="The create button is not disabled")
+ create_action = self.getAndAssertTableAction(res, 'subnets', 'create')
+ self.assertTrue('disabled' in create_action.classes,
+ 'The create button should be disabled')
+
+ @test.create_stubs({api.neutron: ('network_list',),
+ quotas: ('tenant_quota_usages',)})
+ def test_create_button_attributes(self):
+ create_action = self._test_create_button_shown_when_quota_disabled(
+ lambda res: self.getAndAssertTableAction(res, 'networks', 'create')
+ )
+
+ self.assertEqual(set(['ajax-modal']), set(create_action.classes))
+ self.assertEqual('horizon:project:networks:create', create_action.url)
+ self.assertEqual('Create Network',
+ six.text_type(create_action.verbose_name))
+ self.assertEqual((('network', 'create_network'),),
+ create_action.policy_rules)
+
+ @test.create_stubs({api.neutron: ('network_get',
+ 'subnet_list',
+ 'port_list',
+ 'is_extension_supported',),
+ quotas: ('tenant_quota_usages',)})
+ def test_create_subnet_button_attributes(self):
+ network_id = self.networks.first().id
+ quota_data = self.neutron_quota_usages.first()
+ quota_data['subnets']['available'] = 1
+
+ api.neutron.network_get(
+ IsA(http.HttpRequest), network_id)\
+ .MultipleTimes().AndReturn(self.networks.first())
+ api.neutron.subnet_list(
+ IsA(http.HttpRequest), network_id=network_id)\
+ .AndReturn(self.subnets.list())
+ api.neutron.port_list(
+ IsA(http.HttpRequest), network_id=network_id)\
+ .AndReturn([self.ports.first()])
+ api.neutron.is_extension_supported(
+ IsA(http.HttpRequest), 'mac-learning')\
+ .AndReturn(False)
+ quotas.tenant_quota_usages(
+ IsA(http.HttpRequest)) \
+ .MultipleTimes().AndReturn(quota_data)
+
+ self.mox.ReplayAll()
+
+ res = self.client.get(reverse('horizon:project:networks:detail',
+ args=[network_id]))
+ self.assertTemplateUsed(res, 'project/networks/detail.html')
+
+ subnets = res.context['subnets_table'].data
+ self.assertItemsEqual(subnets, self.subnets.list())
+
+ create_action = self.getAndAssertTableAction(res, 'subnets', 'create')
+
+ self.assertEqual(set(['ajax-modal']), set(create_action.classes))
+ self.assertEqual('horizon:project:networks:addsubnet',
+ create_action.url)
+ self.assertEqual('Create Subnet',
+ six.text_type(create_action.verbose_name))
+ self.assertEqual((('network', 'create_subnet'),),
+ create_action.policy_rules)
diff --git a/openstack_dashboard/dashboards/project/routers/tests.py b/openstack_dashboard/dashboards/project/routers/tests.py
index 5aad93ddb5..ce0116c123 100644
--- a/openstack_dashboard/dashboards/project/routers/tests.py
+++ b/openstack_dashboard/dashboards/project/routers/tests.py
@@ -23,7 +23,6 @@ import six
from openstack_dashboard import api
from openstack_dashboard.dashboards.project.routers.extensions.routerrules\
import rulemanager
-from openstack_dashboard.dashboards.project.routers import tables
from openstack_dashboard.test import helpers as test
from openstack_dashboard.usage import quotas
@@ -938,18 +937,11 @@ class RouterViewTests(RouterMixin, test.TestCase):
routers = res.context['Routers_table'].data
self.assertItemsEqual(routers, self.routers.list())
- create_link = tables.CreateRouter()
- url = create_link.get_link_url()
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- link_name = "%s (%s)" % (six.text_type(create_link.verbose_name),
- "Quota exceeded")
- expected_string = "" \
- "%s" \
- % (url, link_name, " ".join(classes), link_name)
- self.assertContains(res, expected_string, html=True,
- msg_prefix="The create button is not disabled")
+ create_action = self.getAndAssertTableAction(res, 'Routers', 'create')
+ self.assertTrue('disabled' in create_action.classes,
+ 'Create button is not disabled')
+ self.assertEqual('Create Router (Quota exceeded)',
+ create_action.verbose_name)
@test.create_stubs({api.neutron: ('router_list', 'network_list'),
quotas: ('tenant_quota_usages',)})
@@ -973,14 +965,38 @@ class RouterViewTests(RouterMixin, test.TestCase):
routers = res.context['Routers_table'].data
self.assertItemsEqual(routers, self.routers.list())
- create_link = tables.CreateRouter()
- url = create_link.get_link_url()
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- link_name = "%s" % (six.text_type(create_link.verbose_name))
- expected_string = "" \
- "%s" \
- % (url, link_name, " ".join(classes), link_name)
- self.assertContains(res, expected_string, html=True,
- msg_prefix="The create button is not displayed")
+ create_action = self.getAndAssertTableAction(res, 'Routers', 'create')
+ self.assertFalse('disabled' in create_action.classes,
+ 'Create button should not be disabled')
+ self.assertEqual('Create Router',
+ create_action.verbose_name)
+
+ @test.create_stubs({api.neutron: ('router_list', 'network_list'),
+ quotas: ('tenant_quota_usages',)})
+ def test_create_button_attributes(self):
+ quota_data = self.neutron_quota_usages.first()
+ quota_data['routers']['available'] = 10
+ api.neutron.router_list(
+ IsA(http.HttpRequest),
+ tenant_id=self.tenant.id,
+ search_opts=None).AndReturn(self.routers.list())
+ quotas.tenant_quota_usages(
+ IsA(http.HttpRequest)) \
+ .MultipleTimes().AndReturn(quota_data)
+
+ self._mock_external_network_list()
+ self.mox.ReplayAll()
+
+ res = self.client.get(self.INDEX_URL)
+ self.assertTemplateUsed(res, 'project/routers/index.html')
+
+ routers = res.context['Routers_table'].data
+ self.assertItemsEqual(routers, self.routers.list())
+
+ create_action = self.getAndAssertTableAction(res, 'Routers', 'create')
+ self.assertEqual(set(['ajax-modal']), set(create_action.classes))
+ self.assertEqual('Create Router',
+ six.text_type(create_action.verbose_name))
+ self.assertEqual('horizon:project:routers:create', create_action.url)
+ self.assertEqual((('network', 'create_router'),),
+ create_action.policy_rules)
diff --git a/openstack_dashboard/dashboards/project/volumes/volumes/tests.py b/openstack_dashboard/dashboards/project/volumes/volumes/tests.py
index ab7fbb641f..b6a70d0554 100644
--- a/openstack_dashboard/dashboards/project/volumes/volumes/tests.py
+++ b/openstack_dashboard/dashboards/project/volumes/volumes/tests.py
@@ -15,7 +15,6 @@
# 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 django
from django.core.urlresolvers import reverse
from django.forms import widgets
@@ -24,11 +23,10 @@ from django.test.utils import override_settings
from mox3.mox import IsA # noqa
import six
+from six import moves
from openstack_dashboard import api
from openstack_dashboard.api import cinder
-from openstack_dashboard.dashboards.project.volumes \
- .volumes import tables
from openstack_dashboard.test import helpers as test
from openstack_dashboard.usage import quotas
@@ -1021,6 +1019,50 @@ class VolumeViewTests(test.TestCase):
server.id)
self.assertEqual(res.status_code, 200)
+ def _get_volume_row_action_from_ajax(self, res, action_name, row_id):
+ def _matches_row_id(context_row):
+ return (len(context_row.dicts) > 1 and
+ isinstance(context_row.dicts[1], dict) and
+ context_row.dicts[1].get('row_id', None) == row_id)
+
+ matching = list(moves.filter(lambda r: _matches_row_id(r),
+ res.context))
+ self.assertTrue(len(matching) > 1,
+ "Expected at least one row matching %s" % row_id)
+ row = matching[-1].dicts[1]
+ matching_actions = list(moves.filter(lambda a: a.name == action_name,
+ row['row_actions']))
+ self.assertEqual(1, len(matching_actions),
+ "Expected one row action named '%s'" % action_name)
+ return matching_actions[0]
+
+ @test.create_stubs({cinder: ('tenant_absolute_limits',
+ 'volume_get',)})
+ def test_create_snapshot_button_attributes(self):
+ limits = {'maxTotalSnapshots': 2}
+ limits['totalSnapshotsUsed'] = 1
+ volume = self.cinder_volumes.first()
+
+ cinder.volume_get(IsA(http.HttpRequest), volume.id).AndReturn(volume)
+ cinder.tenant_absolute_limits(IsA(http.HttpRequest)).AndReturn(limits)
+ self.mox.ReplayAll()
+
+ res_url = (VOLUME_INDEX_URL +
+ "?action=row_update&table=volumes&obj_id=" + volume.id)
+
+ res = self.client.get(res_url, {},
+ HTTP_X_REQUESTED_WITH='XMLHttpRequest')
+
+ snapshot_action = self._get_volume_row_action_from_ajax(
+ res, 'snapshots', volume.id)
+ self.assertEqual('horizon:project:volumes:volumes:create_snapshot',
+ snapshot_action.url)
+ self.assertEqual(set(['ajax-modal']), set(snapshot_action.classes))
+ self.assertEqual('Create Snapshot',
+ six.text_type(snapshot_action.verbose_name))
+ self.assertEqual((('volume', 'volume:create_snapshot'),),
+ snapshot_action.policy_rules)
+
@test.create_stubs({cinder: ('tenant_absolute_limits',
'volume_get',)})
def test_create_snapshot_button_disabled_when_quota_exceeded(self):
@@ -1032,25 +1074,57 @@ class VolumeViewTests(test.TestCase):
cinder.tenant_absolute_limits(IsA(http.HttpRequest)).AndReturn(limits)
self.mox.ReplayAll()
- create_link = tables.CreateSnapshot()
- url = reverse(create_link.get_link_url(), args=[volume.id])
res_url = (VOLUME_INDEX_URL +
"?action=row_update&table=volumes&obj_id=" + volume.id)
res = self.client.get(res_url, {},
HTTP_X_REQUESTED_WITH='XMLHttpRequest')
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- link_name = "%s (%s)" % (six.text_type(create_link.verbose_name),
- "Quota exceeded")
- expected_string = "%s" \
- % (url, " ".join(classes), volume.id, link_name)
+ snapshot_action = self._get_volume_row_action_from_ajax(
+ res, 'snapshots', volume.id)
+ self.assertTrue('disabled' in snapshot_action.classes,
+ 'The create snapshot button should be disabled')
- self.assertContains(
- res, expected_string, html=True,
- msg_prefix="The create snapshot button is not disabled")
+ @test.create_stubs({cinder: ('tenant_absolute_limits',
+ 'volume_list',
+ 'volume_snapshot_list',
+ 'volume_backup_supported',),
+ api.nova: ('server_list',)})
+ def test_create_button_attributes(self):
+ limits = self.cinder_limits['absolute']
+ limits['maxTotalVolumes'] = 10
+ limits['totalVolumesUsed'] = 1
+ volumes = self.cinder_volumes.list()
+
+ api.cinder.volume_backup_supported(IsA(http.HttpRequest)). \
+ MultipleTimes().AndReturn(True)
+ cinder.volume_list(IsA(http.HttpRequest), search_opts=None)\
+ .AndReturn(volumes)
+ cinder.volume_snapshot_list(IsA(http.HttpRequest),
+ search_opts=None).\
+ AndReturn([])
+ api.nova.server_list(IsA(http.HttpRequest), search_opts=None)\
+ .AndReturn([self.servers.list(), False])
+ cinder.tenant_absolute_limits(IsA(http.HttpRequest))\
+ .MultipleTimes().AndReturn(limits)
+ self.mox.ReplayAll()
+
+ res = self.client.get(VOLUME_INDEX_URL)
+ self.assertTemplateUsed(res, 'project/volumes/index.html')
+
+ volumes = res.context['volumes_table'].data
+ self.assertItemsEqual(volumes, self.cinder_volumes.list())
+
+ create_action = self.getAndAssertTableAction(res, 'volumes', 'create')
+
+ self.assertEqual(set(['ajax-modal', 'ajax-update', 'btn-create']),
+ set(create_action.classes))
+ self.assertEqual('Create Volume',
+ six.text_type(create_action.verbose_name))
+ self.assertEqual('horizon:project:volumes:volumes:create',
+ create_action.url)
+ self.assertEqual((('volume', 'volume:create'),),
+ create_action.policy_rules)
@test.create_stubs({cinder: ('tenant_absolute_limits',
'volume_list',
@@ -1081,19 +1155,9 @@ class VolumeViewTests(test.TestCase):
volumes = res.context['volumes_table'].data
self.assertItemsEqual(volumes, self.cinder_volumes.list())
- create_link = tables.CreateVolume()
- url = create_link.get_link_url()
- classes = (list(create_link.get_default_classes())
- + list(create_link.classes))
- link_name = "%s (%s)" % (six.text_type(create_link.verbose_name),
- "Quota exceeded")
- expected_string = " "\
- "%s" \
- % (url, link_name, " ".join(classes), link_name)
- self.assertContains(res, expected_string, html=True,
- msg_prefix="The create button is not disabled")
+ create_action = self.getAndAssertTableAction(res, 'volumes', 'create')
+ self.assertTrue('disabled' in create_action.classes,
+ 'The create button should be disabled')
@test.create_stubs({cinder: ('tenant_absolute_limits',
'volume_get',),
diff --git a/openstack_dashboard/test/helpers.py b/openstack_dashboard/test/helpers.py
index b108d6be0e..b01216b0d0 100644
--- a/openstack_dashboard/test/helpers.py
+++ b/openstack_dashboard/test/helpers.py
@@ -280,6 +280,44 @@ class TestCase(horizon_helpers.TestCase):
def assertItemsCollectionEqual(self, response, items_list):
self.assertEqual(response.json, {"items": items_list})
+ def getAndAssertTableRowAction(self, response, table_name,
+ action_name, row_id):
+ table = response.context[table_name + '_table']
+ full_row_id = '%s__row__%s' % (table_name, row_id)
+ rows = list(moves.filter(lambda x: x.id == full_row_id,
+ table.get_rows()))
+ self.assertEqual(1, len(rows),
+ "Did not find a row matching id '%s'" % row_id)
+ row_actions = table.get_row_actions(rows[0])
+
+ msg_args = (table_name, action_name, row_id)
+ self.assertTrue(
+ len(row_actions) > 0,
+ "No action named '%s' found in table '%s' row '%s'" % msg_args)
+
+ self.assertEqual(
+ 1, len(row_actions),
+ "Multiple actions '%s' found in table '%s' row '%s'" % msg_args)
+
+ return row_actions[0]
+
+ def getAndAssertTableAction(self, response, table_name, action_name):
+
+ table = response.context[table_name + '_table']
+ table_actions = table.get_table_actions()
+ actions = list(moves.filter(lambda x: x.name == action_name,
+ table_actions))
+ msg_args = (table_name, action_name)
+ self.assertTrue(
+ len(actions) > 0,
+ "No action named '%s' found in table '%s'" % msg_args)
+
+ self.assertEqual(
+ 1, len(actions),
+ "More than one action named '%s' found in table '%s'" % msg_args)
+
+ return actions[0]
+
@staticmethod
def mock_rest_request(**args):
mock_args = {