# Copyright 2012 United States Government as represented by the # Administrator of the National Aeronautics and Space Administration. # All Rights Reserved. # # 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. from datetime import datetime from urllib import parse import functools import swiftclient from django.conf import settings from django.utils.translation import gettext_lazy as _ from horizon import exceptions from openstack_dashboard.api import base from openstack_dashboard.contrib.developer.profiler import api as profiler FOLDER_DELIMITER = "/" CHUNK_SIZE = settings.SWIFT_FILE_TRANSFER_CHUNK_SIZE # Swift ACL GLOBAL_READ_ACL = ".r:*" LIST_CONTENTS_ACL = ".rlistings" def safe_swift_exception(function): @functools.wraps(function) def wrapper(*args, **kwargs): try: return function(*args, **kwargs) except swiftclient.client.ClientException as e: e.http_scheme = e.http_host = e.http_port = '' raise e return wrapper class Container(base.APIDictWrapper): pass class StorageObject(base.APIDictWrapper): def __init__(self, apidict, container_name, orig_name=None, data=None): super().__init__(apidict) self.container_name = container_name self.orig_name = orig_name self.data = data @property def id(self): return self.name class PseudoFolder(base.APIDictWrapper): def __init__(self, apidict, container_name): super().__init__(apidict) self.container_name = container_name @property def id(self): return '%s/%s' % (self.container_name, self.name) @property def name(self): return self.subdir.rstrip(FOLDER_DELIMITER) @property def bytes(self): return 0 @property def content_type(self): return "application/pseudo-folder" def _objectify(items, container_name): """Splits a listing of objects into their appropriate wrapper classes.""" objects = [] # Deal with objects and object pseudo-folders first, save subdirs for later for item in items: if item.get("subdir", None) is not None: object_cls = PseudoFolder else: object_cls = StorageObject objects.append(object_cls(item, container_name)) return objects def get_storage_policy_display_name(name): """Gets the user friendly display name for a storage policy""" display_names = settings.SWIFT_STORAGE_POLICY_DISPLAY_NAMES return display_names.get(name) def _metadata_to_header(metadata): headers = {} public = metadata.get('is_public') if public is True: public_container_acls = [GLOBAL_READ_ACL, LIST_CONTENTS_ACL] headers['x-container-read'] = ",".join(public_container_acls) elif public is False: headers['x-container-read'] = "" storage_policy = metadata.get("storage_policy") if storage_policy: headers["x-storage-policy"] = storage_policy return headers def swift_api(request): endpoint = base.url_for(request, 'object-store') cacert = settings.OPENSTACK_SSL_CACERT insecure = settings.OPENSTACK_SSL_NO_VERIFY return swiftclient.client.Connection(None, request.user.username, None, preauthtoken=request.user.token.id, preauthurl=endpoint, cacert=cacert, insecure=insecure, auth_version="3") @profiler.trace def swift_container_exists(request, container_name): try: swift_api(request).head_container(container_name) return True except swiftclient.client.ClientException: return False @profiler.trace def swift_object_exists(request, container_name, object_name): try: swift_api(request).head_object(container_name, object_name) return True except swiftclient.client.ClientException: return False @profiler.trace @safe_swift_exception def swift_get_containers(request, marker=None, prefix=None): limit = settings.API_RESULT_LIMIT headers, containers = swift_api(request).get_account(limit=limit + 1, marker=marker, prefix=prefix, full_listing=True) container_objs = [Container(c) for c in containers] if(len(container_objs) > limit): return (container_objs[0:-1], True) return (container_objs, False) @profiler.trace @safe_swift_exception def swift_get_container(request, container_name, with_data=False): if with_data: headers, data = swift_api(request).get_object(container_name, "") else: data = None headers = swift_api(request).head_container(container_name) timestamp = None is_public = False public_url = None storage_policy = headers.get("x-storage-policy") storage_policy_display_name = \ get_storage_policy_display_name(storage_policy) try: is_public = GLOBAL_READ_ACL in headers.get('x-container-read', '') if is_public: swift_endpoint = base.url_for(request, 'object-store', endpoint_type='publicURL') parameters = parse.quote(container_name.encode('utf8')) public_url = swift_endpoint + '/' + parameters ts_float = float(headers.get('x-timestamp')) timestamp = datetime.utcfromtimestamp(ts_float).isoformat() except Exception: pass container_info = { 'name': container_name, 'container_object_count': headers.get('x-container-object-count'), 'container_bytes_used': headers.get('x-container-bytes-used'), 'timestamp': timestamp, 'data': data, 'is_public': is_public, 'storage_policy': { "name": storage_policy, }, 'public_url': public_url, } if storage_policy_display_name: container_info['storage_policy']['display_name'] = \ get_storage_policy_display_name(storage_policy) return Container(container_info) @profiler.trace @safe_swift_exception def swift_create_container(request, name, metadata=None): if swift_container_exists(request, name): raise exceptions.AlreadyExists(name, 'container') headers = _metadata_to_header(metadata or {}) swift_api(request).put_container(name, headers=headers) return Container({'name': name}) @profiler.trace @safe_swift_exception def swift_update_container(request, name, metadata=None): headers = _metadata_to_header(metadata or {}) swift_api(request).post_container(name, headers=headers) return Container({'name': name}) @profiler.trace @safe_swift_exception def swift_delete_container(request, name): # It cannot be deleted if it's not empty. The batch remove of objects # be done in swiftclient instead of Horizon. objects, more = swift_get_objects(request, name) if objects: error_msg = _("The container cannot be deleted " "since it is not empty.") exc = exceptions.Conflict(error_msg) raise exc swift_api(request).delete_container(name) return True @profiler.trace @safe_swift_exception def swift_get_objects(request, container_name, prefix=None, marker=None, limit=None): limit = limit or settings.API_RESULT_LIMIT kwargs = dict(prefix=prefix, marker=marker, limit=limit + 1, delimiter=FOLDER_DELIMITER, full_listing=True) headers, objects = swift_api(request).get_container(container_name, **kwargs) object_objs = _objectify(objects, container_name) if(len(object_objs) > limit): return (object_objs[0:-1], True) return (object_objs, False) @profiler.trace @safe_swift_exception def swift_filter_objects(request, filter_string, container_name, prefix=None, marker=None): # FIXME(kewu): Swift currently has no real filtering API, thus the marker # parameter here won't actually help the pagination. For now I am just # getting the largest number of objects from a container and filtering # based on those objects. limit = 9999 objects = swift_get_objects(request, container_name, prefix=prefix, marker=marker, limit=limit) filter_string_list = filter_string.lower().strip().split(' ') def matches_filter(obj): for q in filter_string_list: return wildcard_search(obj.name.lower(), q) return filter(matches_filter, objects[0]) def wildcard_search(string, q): q_list = q.split('*') if all(map(lambda x: x == '', q_list)): return True if q_list[0] not in string: return False if q_list[0] == '': tail = string else: head, delimiter, tail = string.partition(q_list[0]) return wildcard_search(tail, '*'.join(q_list[1:])) @profiler.trace @safe_swift_exception def swift_copy_object(request, orig_container_name, orig_object_name, new_container_name, new_object_name): if swift_object_exists(request, new_container_name, new_object_name): raise exceptions.AlreadyExists(new_object_name, 'object') headers = {"X-Copy-From": FOLDER_DELIMITER.join([orig_container_name, orig_object_name])} etag = swift_api(request).put_object(new_container_name, new_object_name, None, headers=headers) obj_info = {'name': new_object_name, 'etag': etag} return StorageObject(obj_info, new_container_name) @profiler.trace @safe_swift_exception def swift_upload_object(request, container_name, object_name, object_file=None): headers = {} size = 0 if object_file: headers['X-Object-Meta-Orig-Filename'] = object_file.name size = object_file.size etag = swift_api(request).put_object(container_name, object_name, object_file, content_length=size, headers=headers) obj_info = {'name': object_name, 'bytes': size, 'etag': etag} return StorageObject(obj_info, container_name) @profiler.trace @safe_swift_exception def swift_create_pseudo_folder(request, container_name, pseudo_folder_name): # Make sure the folder name doesn't already exist. if swift_object_exists(request, container_name, pseudo_folder_name): name = pseudo_folder_name.strip('/') raise exceptions.AlreadyExists(name, 'pseudo-folder') headers = {} etag = swift_api(request).put_object(container_name, pseudo_folder_name, None, headers=headers) obj_info = { 'name': pseudo_folder_name, 'etag': etag } return PseudoFolder(obj_info, container_name) @profiler.trace @safe_swift_exception def swift_delete_object(request, container_name, object_name): swift_api(request).delete_object(container_name, object_name) return True @profiler.trace @safe_swift_exception def swift_delete_folder(request, container_name, object_name): objects, more = swift_get_objects(request, container_name, prefix=object_name) # In case the given object is pseudo folder, # it can be deleted only if it is empty. # swift_get_objects will return at least # one object (i.e container_name) even if the # given pseudo folder is empty. So if swift_get_objects # returns more than one object then only it will be # considered as non empty folder. if len(objects) > 1: error_msg = _("The pseudo folder cannot be deleted " "since it is not empty.") exc = exceptions.Conflict(error_msg) raise exc swift_api(request).delete_object(container_name, object_name) return True @profiler.trace @safe_swift_exception def swift_get_object(request, container_name, object_name, with_data=True, resp_chunk_size=CHUNK_SIZE): if with_data: headers, data = swift_api(request).get_object( container_name, object_name, resp_chunk_size=resp_chunk_size) else: data = None headers = swift_api(request).head_object(container_name, object_name) orig_name = headers.get("x-object-meta-orig-filename") timestamp = None try: ts_float = float(headers.get('x-timestamp')) timestamp = datetime.utcfromtimestamp(ts_float).isoformat() except Exception: pass obj_info = { 'name': object_name, 'bytes': headers.get('content-length'), 'content_type': headers.get('content-type'), 'etag': headers.get('etag'), 'timestamp': timestamp, } return StorageObject(obj_info, container_name, orig_name=orig_name, data=data) @profiler.trace def swift_get_capabilities(request): try: return swift_api(request).get_capabilities() # NOTE(tsufiev): Ceph backend currently does not support '/info', even # some Swift installations do not support it (see `expose_info` docs). except swiftclient.exceptions.ClientException: return {}