Add "Create Router" button to admin panel
Co-Authored-By: Akihiro Motoki <amotoki@gmail.com> Change-Id: I623acbad9a326845603c7a9f480d05265d5b279e
This commit is contained in:
parent
4ad6b95dd9
commit
19a6c9bc61
@ -11,9 +11,30 @@
|
|||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
from django.urls import reverse_lazy
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
|
from horizon import forms
|
||||||
|
|
||||||
|
from openstack_dashboard import api
|
||||||
from openstack_dashboard.dashboards.project.routers import forms as r_forms
|
from openstack_dashboard.dashboards.project.routers import forms as r_forms
|
||||||
|
|
||||||
|
|
||||||
|
class CreateForm(r_forms.CreateForm):
|
||||||
|
tenant_id = forms.ThemableChoiceField(label=_("Project"))
|
||||||
|
# Other fields which are not defined in field_order will be
|
||||||
|
# placed in the default order.
|
||||||
|
field_order = ['name', 'tenant_id']
|
||||||
|
failure_url = 'horizon:admin:routers:index'
|
||||||
|
|
||||||
|
def __init__(self, request, *args, **kwargs):
|
||||||
|
super(CreateForm, self).__init__(request, *args, **kwargs)
|
||||||
|
tenant_choices = [('', _("Select a project"))]
|
||||||
|
tenants, __ = api.keystone.tenant_list(request)
|
||||||
|
for tenant in tenants:
|
||||||
|
if tenant.enabled:
|
||||||
|
tenant_choices.append((tenant.id, tenant.name))
|
||||||
|
self.fields['tenant_id'].choices = tenant_choices
|
||||||
|
|
||||||
|
|
||||||
class UpdateForm(r_forms.UpdateForm):
|
class UpdateForm(r_forms.UpdateForm):
|
||||||
redirect_url = reverse_lazy('horizon:admin:routers:index')
|
redirect_url = reverse_lazy('horizon:admin:routers:index')
|
||||||
|
@ -23,6 +23,10 @@ class DeleteRouter(r_tables.DeleteRouter):
|
|||||||
redirect_url = "horizon:admin:routers:index"
|
redirect_url = "horizon:admin:routers:index"
|
||||||
|
|
||||||
|
|
||||||
|
class CreateRouter(r_tables.CreateRouter):
|
||||||
|
url = "horizon:admin:routers:create"
|
||||||
|
|
||||||
|
|
||||||
class EditRouter(r_tables.EditRouter):
|
class EditRouter(r_tables.EditRouter):
|
||||||
url = "horizon:admin:routers:update"
|
url = "horizon:admin:routers:update"
|
||||||
|
|
||||||
@ -52,7 +56,8 @@ class RoutersTable(r_tables.RoutersTable):
|
|||||||
verbose_name = _("Routers")
|
verbose_name = _("Routers")
|
||||||
status_columns = ["status"]
|
status_columns = ["status"]
|
||||||
row_class = UpdateRow
|
row_class = UpdateRow
|
||||||
table_actions = (DeleteRouter, AdminRoutersFilterAction)
|
table_actions = (CreateRouter, DeleteRouter,
|
||||||
|
AdminRoutersFilterAction)
|
||||||
row_actions = (EditRouter, DeleteRouter,)
|
row_actions = (EditRouter, DeleteRouter,)
|
||||||
columns = ('tenant', 'name', 'status', 'distributed', 'ext_net',
|
columns = ('tenant', 'name', 'status', 'distributed', 'ext_net',
|
||||||
'ha', 'availability_zones', 'admin_state',)
|
'ha', 'availability_zones', 'admin_state',)
|
||||||
|
@ -19,6 +19,7 @@ import mock
|
|||||||
from openstack_dashboard import api
|
from openstack_dashboard import api
|
||||||
from openstack_dashboard.dashboards.project.routers import tests as r_test
|
from openstack_dashboard.dashboards.project.routers import tests as r_test
|
||||||
from openstack_dashboard.test import helpers as test
|
from openstack_dashboard.test import helpers as test
|
||||||
|
from openstack_dashboard.usage import quotas
|
||||||
|
|
||||||
INDEX_TEMPLATE = 'horizon/common/_data_table_view.html'
|
INDEX_TEMPLATE = 'horizon/common/_data_table_view.html'
|
||||||
|
|
||||||
@ -72,10 +73,13 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
@test.create_mocks({api.neutron: ('router_list',
|
@test.create_mocks({api.neutron: ('router_list',
|
||||||
'network_list',
|
'network_list',
|
||||||
'is_extension_supported'),
|
'is_extension_supported'),
|
||||||
api.keystone: ('tenant_list',)})
|
api.keystone: ('tenant_list',),
|
||||||
|
quotas: ('tenant_quota_usages',)})
|
||||||
def test_index(self):
|
def test_index(self):
|
||||||
tenants = self.tenants.list()
|
tenants = self.tenants.list()
|
||||||
|
quota_data = self.neutron_quota_usages.first()
|
||||||
self.mock_router_list.return_value = self.routers.list()
|
self.mock_router_list.return_value = self.routers.list()
|
||||||
|
self.mock_tenant_quota_usages.return_value = quota_data
|
||||||
self.mock_tenant_list.return_value = [tenants, False]
|
self.mock_tenant_list.return_value = [tenants, False]
|
||||||
self.mock_is_extension_supported.return_value = True
|
self.mock_is_extension_supported.return_value = True
|
||||||
self._mock_external_network_list()
|
self._mock_external_network_list()
|
||||||
@ -88,14 +92,20 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
|
|
||||||
self.mock_router_list.assert_called_once_with(test.IsHttpRequest())
|
self.mock_router_list.assert_called_once_with(test.IsHttpRequest())
|
||||||
self.mock_tenant_list.assert_called_once_with(test.IsHttpRequest())
|
self.mock_tenant_list.assert_called_once_with(test.IsHttpRequest())
|
||||||
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
|
self.mock_tenant_quota_usages, 2,
|
||||||
|
mock.call(test.IsHttpRequest(), targets=('router',)))
|
||||||
self.mock_is_extension_supported.assert_called_once_with(
|
self.mock_is_extension_supported.assert_called_once_with(
|
||||||
test.IsHttpRequest(), "router_availability_zone")
|
test.IsHttpRequest(), "router_availability_zone")
|
||||||
self._check_mock_external_network_list()
|
self._check_mock_external_network_list()
|
||||||
|
|
||||||
@test.create_mocks({api.neutron: ('router_list',
|
@test.create_mocks({api.neutron: ('router_list',
|
||||||
'is_extension_supported')})
|
'is_extension_supported'),
|
||||||
|
quotas: ('tenant_quota_usages',)})
|
||||||
def test_index_router_list_exception(self):
|
def test_index_router_list_exception(self):
|
||||||
|
quota_data = self.neutron_quota_usages.first()
|
||||||
self.mock_router_list.side_effect = self.exceptions.neutron
|
self.mock_router_list.side_effect = self.exceptions.neutron
|
||||||
|
self.mock_tenant_quota_usages.return_value = quota_data
|
||||||
self.mock_is_extension_supported.return_value = True
|
self.mock_is_extension_supported.return_value = True
|
||||||
|
|
||||||
res = self.client.get(self.INDEX_URL)
|
res = self.client.get(self.INDEX_URL)
|
||||||
@ -104,6 +114,9 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
self.assertEqual(len(res.context['table'].data), 0)
|
self.assertEqual(len(res.context['table'].data), 0)
|
||||||
self.assertMessageCount(res, error=1)
|
self.assertMessageCount(res, error=1)
|
||||||
self.mock_router_list.assert_called_once_with(test.IsHttpRequest())
|
self.mock_router_list.assert_called_once_with(test.IsHttpRequest())
|
||||||
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
|
self.mock_tenant_quota_usages, 2,
|
||||||
|
mock.call(test.IsHttpRequest(), targets=('router',)))
|
||||||
self.mock_is_extension_supported.assert_called_once_with(
|
self.mock_is_extension_supported.assert_called_once_with(
|
||||||
test.IsHttpRequest(), "router_availability_zone")
|
test.IsHttpRequest(), "router_availability_zone")
|
||||||
|
|
||||||
@ -111,13 +124,16 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
'router_list_on_l3_agent',
|
'router_list_on_l3_agent',
|
||||||
'network_list',
|
'network_list',
|
||||||
'is_extension_supported'),
|
'is_extension_supported'),
|
||||||
api.keystone: ('tenant_list',)})
|
api.keystone: ('tenant_list',),
|
||||||
|
quotas: ('tenant_quota_usages',)})
|
||||||
def test_list_by_l3_agent(self):
|
def test_list_by_l3_agent(self):
|
||||||
tenants = self.tenants.list()
|
tenants = self.tenants.list()
|
||||||
|
quota_data = self.neutron_quota_usages.first()
|
||||||
agent = self.agents.list()[1]
|
agent = self.agents.list()[1]
|
||||||
self.mock_agent_list.return_value = [agent]
|
self.mock_agent_list.return_value = [agent]
|
||||||
self.mock_router_list_on_l3_agent.return_value = self.routers.list()
|
self.mock_router_list_on_l3_agent.return_value = self.routers.list()
|
||||||
self.mock_tenant_list.return_value = [tenants, False]
|
self.mock_tenant_list.return_value = [tenants, False]
|
||||||
|
self.mock_tenant_quota_usages.return_value = quota_data
|
||||||
self.mock_is_extension_supported.return_value = True
|
self.mock_is_extension_supported.return_value = True
|
||||||
self._mock_external_network_list()
|
self._mock_external_network_list()
|
||||||
|
|
||||||
@ -134,6 +150,9 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
self.mock_router_list_on_l3_agent.assert_called_once_with(
|
self.mock_router_list_on_l3_agent.assert_called_once_with(
|
||||||
test.IsHttpRequest(), agent.id, search_opts=None)
|
test.IsHttpRequest(), agent.id, search_opts=None)
|
||||||
self.mock_tenant_list.assert_called_once_with(test.IsHttpRequest())
|
self.mock_tenant_list.assert_called_once_with(test.IsHttpRequest())
|
||||||
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
|
self.mock_tenant_quota_usages, 2,
|
||||||
|
mock.call(test.IsHttpRequest(), targets=('router',)))
|
||||||
self.mock_is_extension_supported.assert_called_once_with(
|
self.mock_is_extension_supported.assert_called_once_with(
|
||||||
test.IsHttpRequest(), "router_availability_zone")
|
test.IsHttpRequest(), "router_availability_zone")
|
||||||
self._check_mock_external_network_list()
|
self._check_mock_external_network_list()
|
||||||
@ -141,10 +160,13 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
@test.create_mocks({api.neutron: ('router_list',
|
@test.create_mocks({api.neutron: ('router_list',
|
||||||
'network_list',
|
'network_list',
|
||||||
'is_extension_supported'),
|
'is_extension_supported'),
|
||||||
api.keystone: ('tenant_list',)})
|
api.keystone: ('tenant_list',),
|
||||||
|
quotas: ('tenant_quota_usages',)})
|
||||||
def test_set_external_network_empty(self):
|
def test_set_external_network_empty(self):
|
||||||
router = self.routers.first()
|
router = self.routers.first()
|
||||||
|
quota_data = self.neutron_quota_usages.first()
|
||||||
self.mock_router_list.return_value = [router]
|
self.mock_router_list.return_value = [router]
|
||||||
|
self.mock_tenant_quota_usages.return_value = quota_data
|
||||||
self.mock_is_extension_supported.return_value = True
|
self.mock_is_extension_supported.return_value = True
|
||||||
self.mock_tenant_list.return_value = [self.tenants.list(), False]
|
self.mock_tenant_list.return_value = [self.tenants.list(), False]
|
||||||
self._mock_external_network_list(alter_ids=True)
|
self._mock_external_network_list(alter_ids=True)
|
||||||
@ -159,6 +181,9 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
self.assertMessageCount(res, error=1)
|
self.assertMessageCount(res, error=1)
|
||||||
|
|
||||||
self.mock_router_list.assert_called_once_with(test.IsHttpRequest())
|
self.mock_router_list.assert_called_once_with(test.IsHttpRequest())
|
||||||
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
|
self.mock_tenant_quota_usages, 2,
|
||||||
|
mock.call(test.IsHttpRequest(), targets=('router',)))
|
||||||
self.mock_is_extension_supported.assert_called_once_with(
|
self.mock_is_extension_supported.assert_called_once_with(
|
||||||
test.IsHttpRequest(), "router_availability_zone")
|
test.IsHttpRequest(), "router_availability_zone")
|
||||||
self.mock_tenant_list.assert_called_once_with(test.IsHttpRequest())
|
self.mock_tenant_list.assert_called_once_with(test.IsHttpRequest())
|
||||||
@ -173,14 +198,17 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
'port_list',
|
'port_list',
|
||||||
'router_delete',
|
'router_delete',
|
||||||
'is_extension_supported'),
|
'is_extension_supported'),
|
||||||
api.keystone: ('tenant_list',)})
|
api.keystone: ('tenant_list',),
|
||||||
|
quotas: ('tenant_quota_usages',)})
|
||||||
def test_router_delete(self):
|
def test_router_delete(self):
|
||||||
router = self.routers.first()
|
router = self.routers.first()
|
||||||
tenants = self.tenants.list()
|
tenants = self.tenants.list()
|
||||||
|
quota_data = self.neutron_quota_usages.first()
|
||||||
|
|
||||||
self.mock_router_list.return_value = self.routers.list()
|
self.mock_router_list.return_value = self.routers.list()
|
||||||
self.mock_tenant_list.return_value = [tenants, False]
|
self.mock_tenant_list.return_value = [tenants, False]
|
||||||
self._mock_external_network_list(count=3)
|
self._mock_external_network_list(count=3)
|
||||||
|
self.mock_tenant_quota_usages.return_value = quota_data
|
||||||
self.mock_is_extension_supported.return_value = True
|
self.mock_is_extension_supported.return_value = True
|
||||||
self.mock_port_list.return_value = []
|
self.mock_port_list.return_value = []
|
||||||
self.mock_router_delete.return_value = None
|
self.mock_router_delete.return_value = None
|
||||||
@ -201,6 +229,9 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
self.mock_tenant_list, 3,
|
self.mock_tenant_list, 3,
|
||||||
mock.call(test.IsHttpRequest()))
|
mock.call(test.IsHttpRequest()))
|
||||||
self._check_mock_external_network_list(count=3)
|
self._check_mock_external_network_list(count=3)
|
||||||
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
|
self.mock_tenant_quota_usages, 4,
|
||||||
|
mock.call(test.IsHttpRequest(), targets=('router',)))
|
||||||
self.assert_mock_multiple_calls_with_same_arguments(
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
self.mock_is_extension_supported, 3,
|
self.mock_is_extension_supported, 3,
|
||||||
mock.call(test.IsHttpRequest(), 'router_availability_zone'))
|
mock.call(test.IsHttpRequest(), 'router_availability_zone'))
|
||||||
@ -215,15 +246,18 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
'router_remove_interface',
|
'router_remove_interface',
|
||||||
'router_delete',
|
'router_delete',
|
||||||
'is_extension_supported'),
|
'is_extension_supported'),
|
||||||
api.keystone: ('tenant_list',)})
|
api.keystone: ('tenant_list',),
|
||||||
|
quotas: ('tenant_quota_usages',)})
|
||||||
def test_router_with_interface_delete(self):
|
def test_router_with_interface_delete(self):
|
||||||
router = self.routers.first()
|
router = self.routers.first()
|
||||||
ports = self.ports.list()
|
ports = self.ports.list()
|
||||||
tenants = self.tenants.list()
|
tenants = self.tenants.list()
|
||||||
|
quota_data = self.neutron_quota_usages.first()
|
||||||
|
|
||||||
self.mock_router_list.return_value = self.routers.list()
|
self.mock_router_list.return_value = self.routers.list()
|
||||||
self.mock_tenant_list.return_value = [tenants, False]
|
self.mock_tenant_list.return_value = [tenants, False]
|
||||||
self._mock_external_network_list(count=3)
|
self._mock_external_network_list(count=3)
|
||||||
|
self.mock_tenant_quota_usages.return_value = quota_data
|
||||||
self.mock_is_extension_supported.return_value = True
|
self.mock_is_extension_supported.return_value = True
|
||||||
self.mock_port_list.return_value = ports
|
self.mock_port_list.return_value = ports
|
||||||
self.mock_router_remove_interface.return_value = None
|
self.mock_router_remove_interface.return_value = None
|
||||||
@ -245,6 +279,9 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
self.mock_tenant_list, 3,
|
self.mock_tenant_list, 3,
|
||||||
mock.call(test.IsHttpRequest()))
|
mock.call(test.IsHttpRequest()))
|
||||||
self._check_mock_external_network_list(count=3)
|
self._check_mock_external_network_list(count=3)
|
||||||
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
|
self.mock_tenant_quota_usages, 4,
|
||||||
|
mock.call(test.IsHttpRequest(), targets=('router',)))
|
||||||
self.assert_mock_multiple_calls_with_same_arguments(
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
self.mock_is_extension_supported, 3,
|
self.mock_is_extension_supported, 3,
|
||||||
mock.call(test.IsHttpRequest(), 'router_availability_zone'))
|
mock.call(test.IsHttpRequest(), 'router_availability_zone'))
|
||||||
@ -257,9 +294,12 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
self.mock_router_delete.assert_called_once_with(
|
self.mock_router_delete.assert_called_once_with(
|
||||||
test.IsHttpRequest(), router.id)
|
test.IsHttpRequest(), router.id)
|
||||||
|
|
||||||
@test.create_mocks({api.neutron: ('is_extension_supported',)})
|
@test.create_mocks({api.neutron: ('is_extension_supported',),
|
||||||
|
quotas: ('tenant_quota_usages',)})
|
||||||
@test.update_settings(FILTER_DATA_FIRST={'admin.routers': True})
|
@test.update_settings(FILTER_DATA_FIRST={'admin.routers': True})
|
||||||
def test_routers_list_with_admin_filter_first(self):
|
def test_routers_list_with_admin_filter_first(self):
|
||||||
|
quota_data = self.neutron_quota_usages.first()
|
||||||
|
self.mock_tenant_quota_usages.return_value = quota_data
|
||||||
self.mock_is_extension_supported.return_value = True
|
self.mock_is_extension_supported.return_value = True
|
||||||
|
|
||||||
res = self.client.get(self.INDEX_URL)
|
res = self.client.get(self.INDEX_URL)
|
||||||
@ -267,14 +307,20 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
routers = res.context['table'].data
|
routers = res.context['table'].data
|
||||||
self.assertItemsEqual(routers, [])
|
self.assertItemsEqual(routers, [])
|
||||||
|
|
||||||
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
|
self.mock_tenant_quota_usages, 2,
|
||||||
|
mock.call(test.IsHttpRequest(), targets=('router',)))
|
||||||
self.mock_is_extension_supported.assert_called_once_with(
|
self.mock_is_extension_supported.assert_called_once_with(
|
||||||
test.IsHttpRequest(), 'router_availability_zone')
|
test.IsHttpRequest(), 'router_availability_zone')
|
||||||
|
|
||||||
@test.create_mocks({api.neutron: ('is_extension_supported',),
|
@test.create_mocks({api.neutron: ('is_extension_supported',),
|
||||||
api.keystone: ('tenant_list',)})
|
api.keystone: ('tenant_list',),
|
||||||
|
quotas: ('tenant_quota_usages',)})
|
||||||
def test_routers_list_with_non_exist_tenant_filter(self):
|
def test_routers_list_with_non_exist_tenant_filter(self):
|
||||||
self.mock_is_extension_supported.return_value = True
|
self.mock_is_extension_supported.return_value = True
|
||||||
self.mock_tenant_list.return_value = [self.tenants.list(), False]
|
self.mock_tenant_list.return_value = [self.tenants.list(), False]
|
||||||
|
quota_data = self.neutron_quota_usages.first()
|
||||||
|
self.mock_tenant_quota_usages.return_value = quota_data
|
||||||
|
|
||||||
self.client.post(
|
self.client.post(
|
||||||
self.INDEX_URL,
|
self.INDEX_URL,
|
||||||
@ -285,12 +331,19 @@ class RouterTests(RouterMixin, r_test.RouterTestCase, test.BaseAdminViewTests):
|
|||||||
routers = res.context['table'].data
|
routers = res.context['table'].data
|
||||||
self.assertItemsEqual(routers, [])
|
self.assertItemsEqual(routers, [])
|
||||||
|
|
||||||
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
|
self.mock_tenant_quota_usages, 2,
|
||||||
|
mock.call(test.IsHttpRequest(), targets=('router',)))
|
||||||
self.assert_mock_multiple_calls_with_same_arguments(
|
self.assert_mock_multiple_calls_with_same_arguments(
|
||||||
self.mock_is_extension_supported, 2,
|
self.mock_is_extension_supported, 2,
|
||||||
mock.call(test.IsHttpRequest(), "router_availability_zone"))
|
mock.call(test.IsHttpRequest(), "router_availability_zone"))
|
||||||
self.mock_tenant_list.assert_called_once_with(test.IsHttpRequest())
|
self.mock_tenant_list.assert_called_once_with(test.IsHttpRequest())
|
||||||
|
|
||||||
|
|
||||||
|
class RouterViewTests(r_test.RouterViewTests):
|
||||||
|
DASHBOARD = 'admin'
|
||||||
|
|
||||||
|
|
||||||
class RouterTestsNoL3Agent(RouterTests):
|
class RouterTestsNoL3Agent(RouterTests):
|
||||||
|
|
||||||
support_l3_agent = False
|
support_l3_agent = False
|
||||||
|
@ -22,6 +22,7 @@ ROUTER_URL = r'^(?P<router_id>[^/]+)/%s'
|
|||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
url(r'^$', views.IndexView.as_view(), name='index'),
|
url(r'^$', views.IndexView.as_view(), name='index'),
|
||||||
|
url(r'^create/$', views.CreateView.as_view(), name='create'),
|
||||||
url(ROUTER_URL % '$',
|
url(ROUTER_URL % '$',
|
||||||
views.DetailView.as_view(),
|
views.DetailView.as_view(),
|
||||||
name='detail'),
|
name='detail'),
|
||||||
|
@ -120,6 +120,13 @@ class DetailView(r_views.DetailView):
|
|||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
|
class CreateView(r_views.CreateView):
|
||||||
|
form_class = rforms.CreateForm
|
||||||
|
template_name = 'project/routers/create.html'
|
||||||
|
success_url = reverse_lazy("horizon:admin:routers:index")
|
||||||
|
submit_url = reverse_lazy("horizon:admin:routers:create")
|
||||||
|
|
||||||
|
|
||||||
class UpdateView(r_views.UpdateView):
|
class UpdateView(r_views.UpdateView):
|
||||||
form_class = rforms.UpdateForm
|
form_class = rforms.UpdateForm
|
||||||
template_name = 'project/routers/update.html'
|
template_name = 'project/routers/update.html'
|
||||||
|
@ -119,6 +119,10 @@ class CreateForm(forms.SelfHandlingForm):
|
|||||||
try:
|
try:
|
||||||
params = {'name': data['name'],
|
params = {'name': data['name'],
|
||||||
'admin_state_up': data['admin_state_up']}
|
'admin_state_up': data['admin_state_up']}
|
||||||
|
# NOTE: admin form allows to specify tenant_id.
|
||||||
|
# We have the logic here to simplify the logic.
|
||||||
|
if 'tenant_id' in data and data['tenant_id']:
|
||||||
|
params['tenant_id'] = data['tenant_id']
|
||||||
if 'external_network' in data and data['external_network']:
|
if 'external_network' in data and data['external_network']:
|
||||||
params['external_gateway_info'] = {'network_id':
|
params['external_gateway_info'] = {'network_id':
|
||||||
data['external_network']}
|
data['external_network']}
|
||||||
|
@ -0,0 +1,4 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- |
|
||||||
|
Add "Create Router" button to Admin/Network/Routers panel.
|
Loading…
x
Reference in New Issue
Block a user