Add volume group list/show support for admin panel

This commit allows admin to list /show cinder volume groups
using horizon dashboard.

TODO:
1. Modify/Delete volume groups

Partially-Implements blueprint cinder-generic-volume-groups

Change-Id: I75d463204cf83492b30523f46dd0507bbb86dd2e
This commit is contained in:
manchandavishal 2018-12-12 07:14:01 +00:00
parent 254e3791d3
commit c2a3c62039
9 changed files with 299 additions and 0 deletions

View File

@ -0,0 +1,20 @@
# Copyright 2017 NEC Corporation
#
# 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 openstack_dashboard.dashboards.project.volume_groups \
import panel as volume_groups_panel
class VolumeGroups(volume_groups_panel.VolumeGroups):
policy_rules = (("volume", "context_is_admin"),)

View File

@ -0,0 +1,31 @@
# 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 django.utils.translation import ugettext_lazy as _
from horizon import tables
from openstack_dashboard.dashboards.project.volume_groups \
import tables as volume_groups_tables
class GroupsTable(volume_groups_tables.GroupsTable):
# TODO(vishalmanchanda): Add Project Info.column in table
name = tables.WrappingColumn("name_or_id",
verbose_name=_("Name"),
link="horizon:admin:volume_groups:detail")
class Meta(object):
name = "volume_groups"
verbose_name = _("Volume Groups")
table_actions = (
volume_groups_tables.GroupsFilterAction,
)

View File

@ -0,0 +1,27 @@
# 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 django.urls import reverse
from openstack_dashboard.dashboards.project.volume_groups \
import tabs as volume_groups_tabs
class OverviewTab(volume_groups_tabs.OverviewTab):
template_name = ("admin/volume_groups/_detail_overview.html")
def get_redirect_url(self):
return reverse('horizon:admin:volume_groups:index')
class GroupsDetailTabs(volume_groups_tabs.GroupsDetailTabs):
tabs = (OverviewTab,)

View File

@ -0,0 +1,42 @@
{% load i18n sizeformat parse_date %}
<div class="detail">
<dl class="dl-horizontal">
<dt>{% trans "Name" %}</dt>
<dd>{{ group.name|default:_("-") }}</dd>
<dt>{% trans "ID" %}</dt>
<dd>{{ group.id }}</dd>
<dt>{% trans "Description" %}</dt>
<dd>{{ group.description|default:_("-") }}</dd>
<dt>{% trans "Status" %}</dt>
<dd>{{ group.status|capfirst }}</dd>
<dt>{% trans "Availability Zone" %}</dt>
<dd>{{ group.availability_zone }}</dd>
<dt>{% trans "Group Type" %}</dt>
<dd>{{ group.group_type }}</dd>
<dt>{% trans "Created" %}</dt>
<dd>{{ group.created_at|parse_isotime }}</dd>
<dt>{% trans "Replication Status" %}</dt>
<dd>{{ group.replication_status }}</dd>
</dl>
<h4>{% trans "Volume Types" %}</h4>
<hr class="header_rule">
<dl class="dl-horizontal">
{% for vol_type_name in group.volume_type_names %}
<dd>{{ vol_type_name }}</dd>
{% endfor %}
</dl>
<h4>{% trans "Volumes" %}</h4>
<hr class="header_rule">
<dl class="dl-horizontal">
{% for vol in group.volume_names %}
<dd><a href="{% url 'horizon:admin:volumes:detail' vol.id %}">{{ vol.name }}</a></dd>
{% empty %}
<dd>
<em>{% trans "No assigned volumes" %}</em>
</dd>
{% endfor %}
</dl>
</div>

View File

@ -0,0 +1,85 @@
# 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 django.urls import reverse
from openstack_dashboard import api
from openstack_dashboard.api import cinder
from openstack_dashboard.test import helpers as test
INDEX_URL = reverse('horizon:admin:volume_groups:index')
INDEX_TEMPLATE = 'horizon/common/_data_table_view.html'
class AdminVolumeGroupTests(test.BaseAdminViewTests):
@test.create_mocks({api.cinder: ['group_list_with_vol_type_names',
'group_snapshot_list']})
def test_index(self):
group = self.cinder_groups.list()
vg_snapshot = self.cinder_group_snapshots.list()
self.mock_group_list_with_vol_type_names.return_value = group
self.mock_group_snapshot_list.return_value = vg_snapshot
res = self.client.get(INDEX_URL)
self.assertTemplateUsed(res, INDEX_TEMPLATE)
self.assertIn('volume_groups_table', res.context)
volume_groups_table = res.context['volume_groups_table']
volume_groups = volume_groups_table.data
self.assertEqual(len(volume_groups), 1)
self.mock_group_list_with_vol_type_names.assert_called_once_with(
test.IsHttpRequest(), {'all_tenants': 1})
self.mock_group_snapshot_list.assert_called_once_with(
test.IsHttpRequest())
@test.create_mocks({cinder: ['group_get_with_vol_type_names',
'volume_list',
'group_snapshot_list']})
def test_detail_view(self):
group = self.cinder_groups.first()
volumes = self.cinder_volumes.list()
vg_snapshot = self.cinder_group_snapshots.list()
self.mock_group_get_with_vol_type_names.return_value = group
self.mock_volume_list.return_value = volumes
self.mock_group_snapshot_list.return_value = vg_snapshot
url = reverse('horizon:admin:volume_groups:detail',
args=[group.id])
res = self.client.get(url)
self.assertNoFormErrors(res)
self.assertEqual(res.status_code, 200)
self.assertTrue(group.has_snapshots)
self.mock_group_get_with_vol_type_names.assert_called_once_with(
test.IsHttpRequest(), group.id)
search_opts = {'group_id': group.id}
self.mock_volume_list.assert_called_once_with(
test.IsHttpRequest(), search_opts=search_opts)
self.mock_group_snapshot_list.assert_called_once_with(
test.IsHttpRequest(), search_opts=search_opts)
@test.create_mocks({cinder: ['group_get']})
def test_detail_view_with_exception(self):
group = self.cinder_groups.first()
self.mock_group_get.side_effect = self.exceptions.cinder
url = reverse('horizon:admin:volume_groups:detail',
args=[group.id])
res = self.client.get(url)
self.assertNoFormErrors(res)
self.assertRedirectsNoFollow(res, INDEX_URL)
self.mock_group_get.assert_called_once_with(
test.IsHttpRequest(), group.id)

View File

@ -0,0 +1,23 @@
# 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 django.conf.urls import url
from openstack_dashboard.dashboards.admin.volume_groups import views
urlpatterns = [
url(r'^$', views.IndexView.as_view(), name='index'),
url(r'^(?P<group_id>[^/]+)$',
views.DetailView.as_view(),
name='detail'),
]

View File

@ -0,0 +1,61 @@
# 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 django.urls import reverse
from django.utils.translation import ugettext_lazy as _
from horizon import exceptions
from horizon import tables
from openstack_dashboard import api
from openstack_dashboard.dashboards.admin.volume_groups \
import tables as volume_group_tables
from openstack_dashboard.dashboards.admin.volume_groups \
import tabs as volume_group_tabs
from openstack_dashboard.dashboards.project.volume_groups \
import views as volume_groups_views
class IndexView(tables.DataTableView):
table_class = volume_group_tables.GroupsTable
page_title = _("Groups")
def get_data(self):
try:
groups = api.cinder.group_list_with_vol_type_names(
self.request, {'all_tenants': 1})
except Exception:
groups = []
exceptions.handle(self.request,
_("Unable to retrieve volume groups."))
if not groups:
return groups
group_snapshots = api.cinder.group_snapshot_list(self.request)
snapshot_groups = {gs.group_id for gs in group_snapshots}
for g in groups:
g.has_snapshots = g.id in snapshot_groups
return groups
class DetailView(volume_groups_views.DetailView):
tab_group_class = volume_group_tabs.GroupsDetailTabs
def get_context_data(self, **kwargs):
context = super(DetailView, self).get_context_data(**kwargs)
table = volume_group_tables.GroupsTable(self.request)
context["actions"] = table.render_row_actions(context["group"])
return context
@staticmethod
def get_redirect_url():
return reverse('horizon:admin:volume_groups:index')

View File

@ -0,0 +1,10 @@
# The slug of the panel to be added to HORIZON_CONFIG. Required.
PANEL = 'volume_groups'
# The slug of the dashboard the PANEL associated with. Required.
PANEL_DASHBOARD = 'admin'
# The slug of the panel group the PANEL is associated with.
PANEL_GROUP = 'volume'
# Python panel class of the PANEL to be added.
ADD_PANEL = ('openstack_dashboard.dashboards.admin.volume_groups.panel.'
'VolumeGroups')