Source code for openstack_dashboard.dashboards.project.containers.tables

# vim: tabstop=4 shiftwidth=4 softtabstop=4

# 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 django.core.urlresolvers import reverse
from django import shortcuts
from django import template
from django.template import defaultfilters as filters
from django.utils import http
from django.utils import safestring
from django.utils.translation import ugettext_lazy as _

from horizon import exceptions
from horizon import messages
from horizon import tables

from openstack_dashboard import api
from openstack_dashboard.api import swift


LOG = logging.getLogger(__name__)
LOADING_IMAGE = '<img src="/static/dashboard/img/loading.gif" />'


[docs]def wrap_delimiter(name): if name and not name.endswith(swift.FOLDER_DELIMITER): return name + swift.FOLDER_DELIMITER return name
[docs]class ViewContainer(tables.LinkAction): name = "view" verbose_name = _("View Details") url = "horizon:project:containers:container_detail" classes = ("ajax-modal", "btn-view")
[docs]class MakePublicContainer(tables.Action): name = "make_public" verbose_name = _("Make Public") classes = ("btn-edit", )
[docs] def allowed(self, request, container): # Container metadata have not been loaded if not hasattr(container, 'is_public'): return False return not container.is_public
[docs] def single(self, table, request, obj_id): try: api.swift.swift_update_container(request, obj_id, metadata=({'is_public': True})) LOG.info('Updating container "%s" access to public.' % obj_id) messages.success(request, _('Successfully updated container access to ' 'public.')) except Exception: exceptions.handle(request, _('Unable to update container access.')) return shortcuts.redirect('horizon:project:containers:index')
[docs]class MakePrivateContainer(tables.Action): name = "make_private" verbose_name = _("Make Private") classes = ("btn-edit", )
[docs] def allowed(self, request, container): # Container metadata have not been loaded if not hasattr(container, 'is_public'): return False return container.is_public
[docs] def single(self, table, request, obj_id): try: api.swift.swift_update_container(request, obj_id, metadata=({'is_public': False})) LOG.info('Updating container "%s" access to private.' % obj_id) messages.success(request, _('Successfully updated container access to ' 'private.')) except Exception: exceptions.handle(request, _('Unable to update container access.')) return shortcuts.redirect('horizon:project:containers:index')
[docs]class DeleteContainer(tables.DeleteAction): data_type_singular = _("Container") data_type_plural = _("Containers") success_url = "horizon:project:containers:index"
[docs] def delete(self, request, obj_id): api.swift.swift_delete_container(request, obj_id)
[docs] def get_success_url(self, request=None): """Returns the URL to redirect to after a successful action. """ current_container = self.table.kwargs.get("container_name", None) # If the current_container is deleted, then redirect to the default # completion url if current_container in self.success_ids: return self.success_url return request.get_full_path()
[docs]class CreateContainer(tables.LinkAction): name = "create" verbose_name = _("Create Container") url = "horizon:project:containers:create" classes = ("ajax-modal", "btn-create")
[docs]class ListObjects(tables.LinkAction): name = "list_objects" verbose_name = _("View Container") url = "horizon:project:containers:index" classes = ("btn-list",)
[docs]class CreatePseudoFolder(tables.LinkAction): name = "create_pseudo_folder" verbose_name = _("Create Pseudo-folder") url = "horizon:project:containers:create_pseudo_folder" classes = ("ajax-modal", "btn-create")
[docs] def allowed(self, request, datum=None): if self.table.kwargs.get('container_name', None): return True return False
[docs] def update(self, request, obj): # This will only be called for the row, so we can remove the button # styles meant for the table action version. self.attrs = {'class': 'ajax-modal'}
[docs]class UploadObject(tables.LinkAction): name = "upload" verbose_name = _("Upload Object") url = "horizon:project:containers:object_upload" classes = ("ajax-modal", "btn-upload")
[docs] def allowed(self, request, datum=None): if self.table.kwargs.get('container_name', None): return True return False
[docs] def update(self, request, obj): # This will only be called for the row, so we can remove the button # styles meant for the table action version. self.attrs = {'class': 'ajax-modal'}
[docs]def get_size_used(container): return filters.filesizeformat(container.bytes)
[docs]class ContainerAjaxUpdateRow(tables.Row): ajax = True
[docs] def get_data(self, request, container_name): container = api.swift.swift_get_container(request, container_name, with_data=False) return container
[docs]def get_metadata(container): # If the metadata has not been loading, display a loading image if not hasattr(container, 'is_public'): return safestring.mark_safe(LOADING_IMAGE) template_name = 'project/containers/_container_metadata.html' context = {"container": container} return template.loader.render_to_string(template_name, context)
[docs]def get_metadata_loaded(container): # Determine if metadata has been loaded if the attribute is already set. return hasattr(container, 'is_public') and container.is_public is not None
[docs]class ContainersTable(tables.DataTable): METADATA_LOADED_CHOICES = ( (False, None), (True, True), ) name = tables.Column("name", link=get_container_link, verbose_name=_("Container Name")) metadata = tables.Column(get_metadata, verbose_name=_("Container Details"), classes=('nowrap-col', ),) metadata_loaded = tables.Column(get_metadata_loaded, status=True, status_choices=METADATA_LOADED_CHOICES, hidden=True)
[docs] class Meta: name = "containers" verbose_name = _("Containers") row_class = ContainerAjaxUpdateRow status_columns = ['metadata_loaded', ] table_actions = (CreateContainer,) row_actions = (ViewContainer, MakePublicContainer, MakePrivateContainer, DeleteContainer,) browser_table = "navigation" footer = False
[docs] def get_object_id(self, container): return container.name
[docs] def get_absolute_url(self): url = super(ContainersTable, self).get_absolute_url() return http.urlquote(url)
[docs] def get_full_url(self): """Returns the encoded absolute URL path with its query string. This is used for the POST action attribute on the form element wrapping the table. We use this method to persist the pagination marker. """ url = super(ContainersTable, self).get_full_url() return http.urlquote(url)
[docs]class ViewObject(tables.LinkAction): name = "view" verbose_name = _("View Details") url = "horizon:project:containers:object_detail" classes = ("ajax-modal", "btn-view") allowed_data_types = ("objects",)
[docs]class UpdateObject(tables.LinkAction): name = "update_object" verbose_name = _("Edit") url = "horizon:project:containers:object_update" classes = ("ajax-modal", "btn-edit") allowed_data_types = ("objects",)
[docs]class DeleteObject(tables.DeleteAction): name = "delete_object" data_type_singular = _("Object") data_type_plural = _("Objects") allowed_data_types = ("objects",)
[docs] def delete(self, request, obj_id): obj = self.table.get_object_by_id(obj_id) container_name = obj.container_name api.swift.swift_delete_object(request, container_name, obj_id)
[docs] def get_success_url(self, request): url = super(DeleteObject, self).get_success_url(request) return http.urlquote(url)
[docs]class DeleteMultipleObjects(DeleteObject): name = "delete_multiple_objects" data_type_singular = _("Object") data_type_plural = _("Objects") allowed_data_types = ("objects",)
[docs]class CopyObject(tables.LinkAction): name = "copy" verbose_name = _("Copy") url = "horizon:project:containers:object_copy" classes = ("ajax-modal", "btn-copy") allowed_data_types = ("objects",)
[docs]class DownloadObject(tables.LinkAction): name = "download" verbose_name = _("Download") url = "horizon:project:containers:object_download" classes = ("btn-download",) allowed_data_types = ("objects",)
[docs] def allowed(self, request, object): return object.bytes and object.bytes > 0
[docs]class ObjectFilterAction(tables.FilterAction): def _filtered_data(self, table, filter_string): request = table.request container = self.table.kwargs['container_name'] subfolder = self.table.kwargs['subfolder_path'] prefix = wrap_delimiter(subfolder) if subfolder else '' self.filtered_data = api.swift.swift_filter_objects(request, filter_string, container, prefix=prefix) return self.filtered_data
[docs] def filter_subfolders_data(self, table, objects, filter_string): data = self._filtered_data(table, filter_string) return [datum for datum in data if datum.content_type == "application/pseudo-folder"]
[docs] def filter_objects_data(self, table, objects, filter_string): data = self._filtered_data(table, filter_string) return [datum for datum in data if datum.content_type != "application/pseudo-folder"]
[docs] def allowed(self, request, datum=None): if self.table.kwargs.get('container_name', None): return True return False
[docs]def sanitize_name(name): return name.split(swift.FOLDER_DELIMITER)[-1]
[docs]def get_size(obj): if obj.bytes is None: return _("pseudo-folder") return filters.filesizeformat(obj.bytes)
[docs]class ObjectsTable(tables.DataTable): name = tables.Column("name", link=get_link_subfolder, allowed_data_types=("subfolders",), verbose_name=_("Object Name"), filters=(sanitize_name,)) size = tables.Column(get_size, verbose_name=_('Size'))
[docs] class Meta: name = "objects" verbose_name = _("Objects") table_actions = (ObjectFilterAction, CreatePseudoFolder, UploadObject, DeleteMultipleObjects) row_actions = (DownloadObject, UpdateObject, CopyObject, ViewObject, DeleteObject) data_types = ("subfolders", "objects") browser_table = "content" footer = False
[docs] def get_absolute_url(self): url = super(ObjectsTable, self).get_absolute_url() return http.urlquote(url)
[docs] def get_full_url(self): """Returns the encoded absolute URL path with its query string. This is used for the POST action attribute on the form element wrapping the table. We use this method to persist the pagination marker. """ url = super(ObjectsTable, self).get_full_url() return http.urlquote(url)