
Also, rename instance_id field to proper port_id. Before this change, when instance_id was passed from the Instances page in the URL, the information would be lost on form submission, and if the form contained an error, the it would be redisplayed with the port drop-down containing all ports from all instances. This also made submitting this form slow, as the drop-down would be populated with all ports on form validation. It also creates a bug, where if there are more instances than Nova's pagination allows, the ports from newer instances would no longer appear in the drop-down, and the form would fail to validate, even though the choice of port on initial form was correct. Closes-bug: #1920010 Change-Id: I3ab26c19dc9ea1ed23fcff790d0db919039099eb
374 lines
17 KiB
Python
374 lines
17 KiB
Python
# Copyright 2012 United States Government as represented by the
|
|
# Administrator of the National Aeronautics and Space Administration.
|
|
# All Rights Reserved.
|
|
#
|
|
# Copyright 2012 Nebula, Inc.
|
|
# Copyright (c) 2012 X.commerce, a business unit of eBay Inc.
|
|
#
|
|
# 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 unittest import mock
|
|
|
|
from django.urls import reverse
|
|
from django.utils.http import urlencode
|
|
|
|
from openstack_dashboard import api
|
|
from openstack_dashboard.test import helpers as test
|
|
from openstack_dashboard.usage import quotas
|
|
|
|
from horizon.workflows import views
|
|
|
|
|
|
INDEX_URL = reverse('horizon:project:floating_ips:index')
|
|
NAMESPACE = "horizon:project:floating_ips"
|
|
|
|
|
|
class FloatingIpViewTests(test.TestCase):
|
|
|
|
@test.create_mocks({api.neutron: ('floating_ip_target_list',
|
|
'tenant_floating_ip_list')})
|
|
def test_associate(self):
|
|
self.mock_floating_ip_target_list.return_value = \
|
|
self._get_fip_targets()
|
|
self.mock_tenant_floating_ip_list.return_value = \
|
|
self.floating_ips.list()
|
|
|
|
url = reverse('%s:associate' % NAMESPACE)
|
|
res = self.client.get(url)
|
|
self.assertTemplateUsed(res, views.WorkflowView.template_name)
|
|
workflow = res.context['workflow']
|
|
choices = dict(workflow.steps[0].action.fields['ip_id'].choices)
|
|
# Verify that our "associated" floating IP isn't in the choices list.
|
|
self.assertNotIn(self.floating_ips.first(), choices)
|
|
|
|
self.mock_floating_ip_target_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
|
|
@test.create_mocks({api.neutron: ('floating_ip_target_list_by_instance',
|
|
'tenant_floating_ip_list')})
|
|
def test_associate_with_instance_id(self):
|
|
targets = self._get_fip_targets()
|
|
target = targets[0]
|
|
self.mock_floating_ip_target_list_by_instance.return_value = [target]
|
|
self.mock_tenant_floating_ip_list.return_value = \
|
|
self.floating_ips.list()
|
|
|
|
base_url = reverse('%s:associate' % NAMESPACE)
|
|
params = urlencode({'instance_id': target.instance_id})
|
|
url = '?'.join([base_url, params])
|
|
res = self.client.get(url)
|
|
self.assertTemplateUsed(res, views.WorkflowView.template_name)
|
|
workflow = res.context['workflow']
|
|
choices = dict(workflow.steps[0].action.fields['ip_id'].choices)
|
|
# Verify that our "associated" floating IP isn't in the choices list.
|
|
self.assertNotIn(self.floating_ips.first(), choices)
|
|
|
|
self.mock_floating_ip_target_list_by_instance.assert_called_once_with(
|
|
test.IsHttpRequest(), target.instance_id)
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
|
|
def _get_compute_ports(self):
|
|
return [p for p in self.ports.list()
|
|
if not p.device_owner.startswith('network:')]
|
|
|
|
def _get_fip_targets(self):
|
|
server_dict = dict((s.id, s.name) for s in self.servers.list())
|
|
targets = []
|
|
for p in self._get_compute_ports():
|
|
for ip in p.fixed_ips:
|
|
targets.append(api.neutron.FloatingIpTarget(
|
|
p, ip['ip_address'], server_dict[p.device_id]))
|
|
return targets
|
|
|
|
@staticmethod
|
|
def _get_target_id(port):
|
|
return '%s_%s' % (port.id, port.fixed_ips[0]['ip_address'])
|
|
|
|
@test.create_mocks({api.neutron: ('floating_ip_target_list',
|
|
'tenant_floating_ip_list')})
|
|
def test_associate_with_port_id(self):
|
|
compute_port = self._get_compute_ports()[0]
|
|
associated_fips = [fip.id for fip in self.floating_ips.list()
|
|
if fip.port_id]
|
|
|
|
self.mock_floating_ip_target_list.return_value = \
|
|
self._get_fip_targets()
|
|
self.mock_tenant_floating_ip_list.return_value = \
|
|
self.floating_ips.list()
|
|
|
|
base_url = reverse('%s:associate' % NAMESPACE)
|
|
params = urlencode({'port_id': compute_port.id})
|
|
url = '?'.join([base_url, params])
|
|
res = self.client.get(url)
|
|
self.assertTemplateUsed(res, views.WorkflowView.template_name)
|
|
workflow = res.context['workflow']
|
|
choices = dict(workflow.steps[0].action.fields['ip_id'].choices)
|
|
# Verify that our "associated" floating IP isn't in the choices list.
|
|
self.assertFalse(set(associated_fips) & set(choices.keys()))
|
|
|
|
self.mock_floating_ip_target_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
|
|
@test.create_mocks({api.neutron: ('floating_ip_associate',
|
|
'floating_ip_target_list',
|
|
'tenant_floating_ip_list')})
|
|
def test_associate_post(self):
|
|
floating_ip = [fip for fip in self.floating_ips.list()
|
|
if not fip.port_id][0]
|
|
compute_port = self._get_compute_ports()[0]
|
|
port_target_id = self._get_target_id(compute_port)
|
|
|
|
self.mock_tenant_floating_ip_list.return_value = \
|
|
self.floating_ips.list()
|
|
self.mock_floating_ip_target_list.return_value = \
|
|
self._get_fip_targets()
|
|
self.mock_floating_ip_associate.return_value = None
|
|
|
|
form_data = {'port_id': port_target_id,
|
|
'ip_id': floating_ip.id}
|
|
url = reverse('%s:associate' % NAMESPACE)
|
|
res = self.client.post(url, form_data)
|
|
self.assertRedirectsNoFollow(res, INDEX_URL)
|
|
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_target_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_associate.assert_called_once_with(
|
|
test.IsHttpRequest(), floating_ip.id, port_target_id)
|
|
|
|
@test.create_mocks({api.neutron: ('floating_ip_associate',
|
|
'floating_ip_target_list',
|
|
'tenant_floating_ip_list')})
|
|
def test_associate_post_with_redirect(self):
|
|
floating_ip = [fip for fip in self.floating_ips.list()
|
|
if not fip.port_id][0]
|
|
compute_port = self._get_compute_ports()[0]
|
|
port_target_id = self._get_target_id(compute_port)
|
|
|
|
self.mock_tenant_floating_ip_list.return_value = \
|
|
self.floating_ips.list()
|
|
self.mock_floating_ip_target_list.return_value = \
|
|
self._get_fip_targets()
|
|
self.mock_floating_ip_associate.return_value = None
|
|
|
|
next = reverse("horizon:project:instances:index")
|
|
form_data = {'port_id': port_target_id,
|
|
'next': next,
|
|
'ip_id': floating_ip.id}
|
|
url = reverse('%s:associate' % NAMESPACE)
|
|
res = self.client.post(url, form_data)
|
|
self.assertRedirectsNoFollow(res, next)
|
|
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_target_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_associate.assert_called_once_with(
|
|
test.IsHttpRequest(), floating_ip.id, port_target_id)
|
|
|
|
@test.create_mocks({api.neutron: ('floating_ip_associate',
|
|
'floating_ip_target_list',
|
|
'tenant_floating_ip_list')})
|
|
def test_associate_post_with_exception(self):
|
|
floating_ip = [fip for fip in self.floating_ips.list()
|
|
if not fip.port_id][0]
|
|
compute_port = self._get_compute_ports()[0]
|
|
port_target_id = self._get_target_id(compute_port)
|
|
|
|
self.mock_tenant_floating_ip_list.return_value = \
|
|
self.floating_ips.list()
|
|
self.mock_floating_ip_target_list.return_value = \
|
|
self._get_fip_targets()
|
|
self.mock_floating_ip_associate.side_effect = self.exceptions.nova
|
|
|
|
form_data = {'port_id': port_target_id,
|
|
'ip_id': floating_ip.id}
|
|
url = reverse('%s:associate' % NAMESPACE)
|
|
res = self.client.post(url, form_data)
|
|
self.assertRedirectsNoFollow(res, INDEX_URL)
|
|
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_target_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_associate.assert_called_once_with(
|
|
test.IsHttpRequest(), floating_ip.id, port_target_id)
|
|
|
|
@test.create_mocks({api.nova: ('server_list',),
|
|
api.neutron: ('floating_ip_disassociate',
|
|
'floating_ip_pools_list',
|
|
'is_extension_supported',
|
|
'tenant_floating_ip_list')})
|
|
def test_disassociate_post(self):
|
|
floating_ip = self.floating_ips.first()
|
|
|
|
self.mock_is_extension_supported.return_value = False
|
|
self.mock_server_list.return_value = [self.servers.list(), False]
|
|
self.mock_tenant_floating_ip_list.return_value = \
|
|
self.floating_ips.list()
|
|
self.mock_floating_ip_pools_list.return_value = self.pools.list()
|
|
self.mock_floating_ip_disassociate.return_value = None
|
|
|
|
action = "floating_ips__disassociate__%s" % floating_ip.id
|
|
res = self.client.post(INDEX_URL, {"action": action})
|
|
self.assertMessageCount(success=1)
|
|
self.assertRedirectsNoFollow(res, INDEX_URL)
|
|
|
|
self.mock_server_list.assert_called_once_with(test.IsHttpRequest(),
|
|
detailed=False)
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_pools_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_disassociate.assert_called_once_with(
|
|
test.IsHttpRequest(), floating_ip.id)
|
|
self.mock_is_extension_supported.assert_called_once_with(
|
|
test.IsHttpRequest(), 'dns-integration')
|
|
|
|
@test.create_mocks({api.nova: ('server_list',),
|
|
api.neutron: ('floating_ip_disassociate',
|
|
'floating_ip_pools_list',
|
|
'is_extension_supported',
|
|
'tenant_floating_ip_list')})
|
|
def test_disassociate_post_with_exception(self):
|
|
floating_ip = self.floating_ips.first()
|
|
|
|
self.mock_is_extension_supported.return_value = False
|
|
self.mock_server_list.return_value = [self.servers.list(), False]
|
|
self.mock_tenant_floating_ip_list.return_value = \
|
|
self.floating_ips.list()
|
|
self.mock_floating_ip_pools_list.return_value = self.pools.list()
|
|
self.mock_floating_ip_disassociate.side_effect = self.exceptions.nova
|
|
|
|
action = "floating_ips__disassociate__%s" % floating_ip.id
|
|
res = self.client.post(INDEX_URL, {"action": action})
|
|
self.assertRedirectsNoFollow(res, INDEX_URL)
|
|
|
|
self.mock_server_list.assert_called_once_with(test.IsHttpRequest(),
|
|
detailed=False)
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_pools_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_disassociate.assert_called_once_with(
|
|
test.IsHttpRequest(), floating_ip.id)
|
|
self.mock_is_extension_supported.assert_called_once_with(
|
|
test.IsHttpRequest(), 'dns-integration')
|
|
|
|
@test.create_mocks({api.neutron: ('tenant_floating_ip_list',
|
|
'is_extension_supported',
|
|
'floating_ip_pools_list'),
|
|
api.nova: ('server_list',),
|
|
quotas: ('tenant_quota_usages',)})
|
|
def test_allocate_button_attributes(self):
|
|
floating_ips = self.floating_ips.list()
|
|
floating_pools = self.pools.list()
|
|
quota_data = self.neutron_quota_usages.first()
|
|
|
|
self.mock_is_extension_supported.return_value = False
|
|
self.mock_tenant_floating_ip_list.return_value = floating_ips
|
|
self.mock_floating_ip_pools_list.return_value = floating_pools
|
|
self.mock_server_list.return_value = [self.servers.list(), False]
|
|
self.mock_tenant_quota_usages.return_value = quota_data
|
|
|
|
res = self.client.get(INDEX_URL)
|
|
|
|
allocate_action = self.getAndAssertTableAction(res, 'floating_ips',
|
|
'allocate')
|
|
self.assertEqual(set(['ajax-modal']), set(allocate_action.classes))
|
|
self.assertEqual('Allocate IP To Project', allocate_action.verbose_name)
|
|
self.assertIsNone(allocate_action.policy_rules)
|
|
|
|
url = 'horizon:project:floating_ips:allocate'
|
|
self.assertEqual(url, allocate_action.url)
|
|
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_pools_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_server_list.assert_called_once_with(test.IsHttpRequest(),
|
|
detailed=False)
|
|
self.assert_mock_multiple_calls_with_same_arguments(
|
|
self.mock_tenant_quota_usages, 3,
|
|
mock.call(test.IsHttpRequest(), targets=('floatingip', )))
|
|
self.mock_is_extension_supported.assert_called_once_with(
|
|
test.IsHttpRequest(), 'dns-integration',
|
|
)
|
|
|
|
@test.create_mocks({api.neutron: ('tenant_floating_ip_list',
|
|
'is_extension_supported',
|
|
'floating_ip_pools_list'),
|
|
api.nova: ('server_list',),
|
|
quotas: ('tenant_quota_usages',)})
|
|
def test_allocate_button_disabled_when_quota_exceeded(self):
|
|
floating_ips = self.floating_ips.list()
|
|
floating_pools = self.pools.list()
|
|
quota_data = self.neutron_quota_usages.first()
|
|
quota_data['floatingip']['available'] = 0
|
|
|
|
self.mock_is_extension_supported.return_value = False
|
|
self.mock_tenant_floating_ip_list.return_value = floating_ips
|
|
self.mock_floating_ip_pools_list.return_value = floating_pools
|
|
self.mock_server_list.return_value = [self.servers.list(), False]
|
|
self.mock_tenant_quota_usages.return_value = quota_data
|
|
|
|
res = self.client.get(INDEX_URL)
|
|
|
|
allocate_action = self.getAndAssertTableAction(res, 'floating_ips',
|
|
'allocate')
|
|
self.assertIn('disabled', allocate_action.classes,
|
|
'The create button should be disabled')
|
|
self.assertEqual('Allocate IP To Project (Quota exceeded)',
|
|
allocate_action.verbose_name)
|
|
|
|
self.mock_tenant_floating_ip_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_floating_ip_pools_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|
|
self.mock_server_list.assert_called_once_with(test.IsHttpRequest(),
|
|
detailed=False)
|
|
self.assert_mock_multiple_calls_with_same_arguments(
|
|
self.mock_tenant_quota_usages, 3,
|
|
mock.call(test.IsHttpRequest(), targets=('floatingip', )))
|
|
self.mock_is_extension_supported.assert_called_once_with(
|
|
test.IsHttpRequest(), 'dns-integration',
|
|
)
|
|
|
|
@test.create_mocks({api.neutron: ('floating_ip_pools_list',
|
|
'is_extension_supported'),
|
|
quotas: ('tenant_quota_usages',)})
|
|
@test.update_settings(OPENSTACK_NEUTRON_NETWORK={'enable_quotas': True})
|
|
def test_correct_quotas_displayed(self):
|
|
self.mock_is_extension_supported.side_effect = [False, True, False]
|
|
self.mock_tenant_quota_usages.return_value = \
|
|
self.neutron_quota_usages.first()
|
|
self.mock_floating_ip_pools_list.return_value = self.pools.list()
|
|
|
|
url = reverse('%s:allocate' % NAMESPACE)
|
|
res = self.client.get(url)
|
|
self.assertEqual(res.context['usages']['floatingip']['quota'],
|
|
self.neutron_quotas.first().get('floatingip').limit)
|
|
|
|
self.mock_is_extension_supported.assert_called_once_with(
|
|
test.IsHttpRequest(), 'dns-integration')
|
|
self.mock_tenant_quota_usages.assert_called_once_with(
|
|
test.IsHttpRequest(), targets=('floatingip',))
|
|
self.mock_floating_ip_pools_list.assert_called_once_with(
|
|
test.IsHttpRequest())
|