Source code for openstack_dashboard.dashboards.project.images_and_snapshots.images.tables

# Copyright 2012 Nebula, 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.

import logging
from collections import defaultdict

from django.conf import settings
from django.core.urlresolvers import reverse
from django.template import defaultfilters as filters
from django.utils.http import urlencode
from django.utils.translation import ugettext_lazy as _

from horizon import tables
from horizon.utils.memoized import memoized

from openstack_dashboard import api


LOG = logging.getLogger(__name__)


class LaunchImage(tables.LinkAction):
    name = "launch_image"
[docs] verbose_name = _("Launch") url = "horizon:project:instances:launch" classes = ("btn-launch", "ajax-modal") def get_link_url(self, datum): base_url = reverse(self.url)
[docs] data_type_plural = _("Images") def allowed(self, request, image=None): if image:
[docs] return image.owner == request.user.tenant_id # Return True to allow table-level bulk delete action to appear. return True def delete(self, request, obj_id): api.glance.image_delete(request, obj_id)
[docs] class CreateImage(tables.LinkAction): name = "create"
[docs] verbose_name = _("Create Image") url = "horizon:project:images_and_snapshots:images:create" classes = ("ajax-modal", "btn-create") class EditImage(tables.LinkAction): name = "edit"
[docs] verbose_name = _("Edit") url = "horizon:project:images_and_snapshots:images:update" classes = ("ajax-modal", "btn-edit") def allowed(self, request, image=None): if image:
[docs] return image.status in ("active",) and \ image.owner == request.user.tenant_id # We don't have bulk editing, so if there isn't an image that's # authorized, don't allow the action. return False def filter_tenants(): return getattr(settings, 'IMAGES_LIST_FILTER_TENANTS', [])
[docs] @memoized def filter_tenant_ids():
return map(lambda ft: ft['tenant'], filter_tenants()) class OwnerFilter(tables.FixedFilterAction): def get_fixed_buttons(self):
[docs] def make_dict(text, tenant, icon):
[docs] return dict(text=text, value=tenant, icon=icon) buttons = [make_dict('Project', 'project', 'icon-home')] for button_dict in filter_tenants(): new_dict = button_dict.copy() new_dict['value'] = new_dict['tenant'] buttons.append(new_dict) buttons.append(make_dict('Shared with Me', 'shared', 'icon-share')) buttons.append(make_dict('Public', 'public', 'icon-fire')) return buttons def categorize(self, table, images): user_tenant_id = table.request.user.tenant_id
[docs] tenants = defaultdict(list) for im in images: categories = get_image_categories(im, user_tenant_id) for category in categories: tenants[category].append(im) return tenants def get_image_categories(im, user_tenant_id): categories = []
[docs] if im.is_public: categories.append('public') if im.owner == user_tenant_id: categories.append('project') elif im.owner in filter_tenant_ids(): categories.append(im.owner) elif not im.is_public: categories.append('shared') return categories def get_image_type(image): return getattr(image, "properties", {}).get("image_type", _("Image"))
[docs] def get_format(image): format = getattr(image, "disk_format", "")
[docs] # The "container_format" attribute can actually be set to None, # which will raise an error if you call upper() on it. if format is not None: return format.upper() class UpdateRow(tables.Row): ajax = True
[docs] def get_data(self, request, image_id): image = api.glance.image_get(request, image_id)
[docs] return image def load_cells(self, image=None): super(UpdateRow, self).load_cells(image)
[docs] # Tag the row with the image category for client-side filtering. image = self.datum my_tenant_id = self.table.request.user.tenant_id image_categories = get_image_categories(image, my_tenant_id) for category in image_categories: self.classes.append('category-' + category) class ImagesTable(tables.DataTable): STATUS_CHOICES = (
[docs] ("active", True), ("saving", None), ("queued", None), ("pending_delete", None), ("killed", False), ("deleted", False), ) name = tables.Column("name", link=("horizon:project:images_and_snapshots:" "images:detail"), verbose_name=_("Image Name")) image_type = tables.Column(get_image_type, verbose_name=_("Type"), filters=(filters.title,)) status = tables.Column("status", filters=(filters.title,), verbose_name=_("Status"), status=True, status_choices=STATUS_CHOICES) public = tables.Column("is_public", verbose_name=_("Public"), empty_value=False, filters=(filters.yesno, filters.capfirst)) disk_format = tables.Column(get_format, verbose_name=_("Format")) class Meta: name = "images"
[docs] row_class = UpdateRow status_columns = ["status"] verbose_name = _("Images") # Hide the image_type column. Done this way so subclasses still get # all the columns by default. columns = ["name", "status", "public", "disk_format"] table_actions = (OwnerFilter, CreateImage, DeleteImage,) row_actions = (LaunchImage, EditImage, DeleteImage,) pagination_param = "image_marker"