From 92492cf50461e214b777c707148886a8e87f340d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Gr=C3=A9goire=20Unbekandt?= <gregoire.unbekandt@gmail.com> Date: Mon, 24 Jun 2019 16:39:46 +0200 Subject: [PATCH] Add ability to import image into multi-stores The import image api now supports a list of stores to import data into. This list can be specified through a new "stores" field that has been added to the request body. During import stage, Glance iterates overs this list and send the data to each store one by one. If an invalid backend is requested by the user, an exception is raised. If an errors occurs during verify, already pushed data is removed and image state is unchanged. Change-Id: Id3ac19488c0a693d7042be4a3c83f3b9f12313d0 Implements: blueprint import-multi-stores --- api-ref/source/v2/images-import.inc | 37 +- api-ref/source/v2/images-parameters.yaml | 31 ++ .../v2/samples/image-import-g-d-request.json | 4 +- .../v2/samples/image-import-w-d-request.json | 4 +- .../admin/interoperable-image-import.rst | 105 ++++++ glance/api/v2/images.py | 21 +- glance/async_/flows/api_image_import.py | 107 +++++- glance/common/scripts/image_import/main.py | 4 +- glance/common/utils.py | 33 ++ glance/domain/__init__.py | 2 +- glance/domain/proxy.py | 4 +- glance/location.py | 108 ++++-- glance/notifier.py | 27 +- glance/quota/__init__.py | 5 +- glance/tests/functional/v2/test_images.py | 326 ++++++++++++++++++ .../async_/flows/test_api_image_import.py | 78 +++++ glance/tests/unit/common/test_utils.py | 103 ++++++ glance/tests/unit/test_cache_middleware.py | 1 + glance/tests/unit/test_notifier.py | 65 +++- glance/tests/unit/test_quota.py | 2 +- glance/tests/unit/test_store_image.py | 131 ++++++- .../tests/unit/v2/test_image_data_resource.py | 2 +- .../import-multi-stores-3e781f2878b3134d.yaml | 31 ++ 23 files changed, 1131 insertions(+), 100 deletions(-) create mode 100644 releasenotes/notes/import-multi-stores-3e781f2878b3134d.yaml diff --git a/api-ref/source/v2/images-import.inc b/api-ref/source/v2/images-import.inc index 0bee77fd89..761a5f0dc7 100644 --- a/api-ref/source/v2/images-import.inc +++ b/api-ref/source/v2/images-import.inc @@ -171,12 +171,12 @@ call. In the ``web-download`` workflow, the data is made available to the Image service by being posted to an accessible location with a URL that you know. -Beginning with API version 2.8, an optional ``X-Image-Meta-Store`` -header may be added to the request. When present, the image data will be -placed into the backing store whose identifier is the value of this -header. If the store identifier specified is not recognized, a 409 (Conflict) -response is returned. When the header is not present, the image -data is placed into the default backing store. +Beginning with API version 2.8, an optional ``stores`` parameter may be added +to the body request. When present, it contains the list of backing store +identifiers to import the image binary data to. If at least one store +identifier specified is not recognized, a 409 (Conflict) response is returned. +When the parameter is not present, the image data is placed into the default +backing store. * Store identifiers are site-specific. Use the :ref:`Store Discovery <store-discovery-call>` call to determine what @@ -184,13 +184,27 @@ data is placed into the default backing store. * The default store may be determined from the :ref:`Store Discovery <store-discovery-call>` response. * A default store is always defined, so if you do not have a need - to use a particular store, simply omit this header and the default store + to use particular stores, simply omit this parameter and the default store will be used. -* For API versions before version 2.8, this header is silently +* For API versions before version 2.8, this parameter is silently ignored. -Example call: ``curl -i -X POST -H "X-Image-Meta-Store: {store_identifier}" --H "X-Auth-Token: $token" $image_url/v2/images/{image_id}/import`` +For backwards compatibility, if the ``stores`` parameter is not specified, the +header 'X-Image-Meta-Store' is evaluated. + +To import the data into the entire set of stores you may consume from this +particular deployment of Glance without specifying each one of them, you can +use the optional boolean body parameter ``all_stores``. +Note that this can't be used simultaneously with the ``stores`` parameter. + +To set the behavior of the import workflow in case of error, you can use the +optional boolean body parameter ``all_stores_must_succeed``. +When set to True, if an error occurs during the upload in at least one store, +the worfklow fails, the data is deleted from stores where copying is done and +the state of the image remains unchanged. +When set to False (default), the workflow will fail only if the upload fails +on all stores specified. In case of a partial success, the locations added to +the image will be the stores where the data has been correctly uploaded. The JSON request body specifies what import method you wish to use for this image request. @@ -259,6 +273,9 @@ Request - X-Image-Meta-Store: store-header - image_id: image_id-in-path - method: method-in-request + - all_stores: all-stores-in-request + - all_stores_must_succeed: all-stores-succeed-in-request + - stores: stores-in-request Request Example - glance-direct import method --------------------------------------------- diff --git a/api-ref/source/v2/images-parameters.yaml b/api-ref/source/v2/images-parameters.yaml index 28535e0b42..8860da3b96 100644 --- a/api-ref/source/v2/images-parameters.yaml +++ b/api-ref/source/v2/images-parameters.yaml @@ -286,6 +286,30 @@ visibility-in-query: type: string # variables in body +all-stores-in-request: + description: | + When set to True the data will be imported to the set of stores you may + consume from this particular deployment of Glance (ie: the same set of + stores returned to a call to /v2/info/stores on the glance-api the request + hits). + This can't be used simultaneously with the ``stores`` parameter. + in: body + required: false + type: boolean +all-stores-succeed-in-request: + description: | + A boolean parameter indicating the behavior of the import workflow when an + error occurs. + When set to True, if an error occurs during the upload in at least one + store, the worfklow fails, the data is deleted from stores where copying + is done (not staging), and the state of the image is unchanged. + When set to False, the workflow will fail (data deleted from stores, ...) + only if the import fails on all stores specified by the user. In case of a + partial success, the locations added to the image will be the stores where + the data has been correctly uploaded. + in: body + required: false + type: boolean checksum: description: | Hash that is used over the image data. The Image @@ -606,6 +630,13 @@ status: in: body required: true type: string +stores-in-request: + description: | + If present contains the list of store id to import the image binary data + to. + in: body + required: false + type: array tags: description: | List of tags for this image, possibly an empty list. diff --git a/api-ref/source/v2/samples/image-import-g-d-request.json b/api-ref/source/v2/samples/image-import-g-d-request.json index 64589bf46a..1e2d9ba58d 100644 --- a/api-ref/source/v2/samples/image-import-g-d-request.json +++ b/api-ref/source/v2/samples/image-import-g-d-request.json @@ -1,5 +1,7 @@ { "method": { "name": "glance-direct" - } + }, + "stores": ["common", "cheap", "fast", "reliable"], + "all_stores_must_succeed": false } diff --git a/api-ref/source/v2/samples/image-import-w-d-request.json b/api-ref/source/v2/samples/image-import-w-d-request.json index 1713ebd04c..5656b53190 100644 --- a/api-ref/source/v2/samples/image-import-w-d-request.json +++ b/api-ref/source/v2/samples/image-import-w-d-request.json @@ -2,5 +2,7 @@ "method": { "name": "web-download", "uri": "https://download.cirros-cloud.net/0.4.0/cirros-0.4.0-ppc64le-disk.img" - } + }, + "all_stores": true, + "all_stores_must_succeed": true } diff --git a/doc/source/admin/interoperable-image-import.rst b/doc/source/admin/interoperable-image-import.rst index b69c9eb15b..15c21f0018 100644 --- a/doc/source/admin/interoperable-image-import.rst +++ b/doc/source/admin/interoperable-image-import.rst @@ -223,6 +223,111 @@ be either 80 or 443.) .. _iir_plugins: +Importing in multiple stores +---------------------------- + +Starting with Ussuri, it is possible to import data into multiple stores +using interoperable image import workflow. + +The status of the image is set to ``active`` according to the value of +``all_stores_must_succeed`` parameter. + +* If set to False: the image will be available as soon as an import to + one store has succeeded. + +* If set to True (default): the status is set to ``active`` only when all + stores have been successfully treated. + +Check progress +~~~~~~~~~~~~~~ + +As each store is treated sequentially, it can take quite some time for the +workflow to complete depending on the size of the image and the number of +stores to import data to. +It is possible to follow task progress by looking at 2 reserved image +properties: + +* ``os_glance_importing_to_stores``: This property contains a list of stores + that has not yet been processed. At the beginning of the import flow, it is + filled with the stores provided in the request. Each time a store is fully + handled, it is removed from the list. + +* ``os_glance_failed_import``: Each time an import in a store fails, it is + added to this list. This property is emptied at the beginning of the import + flow. + +These 2 properties are also available in the notifications sent during the +workflow: + +.. note:: Example + + An operator calls the import image api with the following parameters:: + + curl -i -X POST -H "X-Auth-Token: $token" + -H "Content-Type: application/json" + -d '{"method": {"name":"glance-direct"}, + "stores": ["ceph1", "ceph2"], + "all_stores_must_succeed": false}' + $image_url/v2/images/{image_id}/import + + The upload fails for 'ceph2' but succeed on 'ceph1'. Since the parameter + ``all_stores_must_succeed`` has been set to 'false', the task ends + successfully and the image is now active. + + Notifications sent by glance looks like (payload is truncated for + clarity):: + + { + "priority": "INFO", + "event_type": "image.prepare", + "timestamp": "2019-08-27 16:10:30.066867", + "payload": {"status": "importing", + "name": "example", + "backend": "ceph1", + "os_glance_importing_to_stores": ["ceph1", "ceph2"], + "os_glance_failed_import": [], + ...}, + "message_id": "1c8993ad-e47c-4af7-9f75-fa49596eeb10", + ... + } + + { + "priority": "INFO", + "event_type": "image.upload", + "timestamp": "2019-08-27 16:10:32.058812", + "payload": {"status": "active", + "name": "example", + "backend": "ceph1", + "os_glance_importing_to_stores": ["ceph2"], + "os_glance_failed_import": [], + ...}, + "message_id": "8b8993ad-e47c-4af7-9f75-fa49596eeb11", + ... + } + + { + "priority": "INFO", + "event_type": "image.prepare", + "timestamp": "2019-08-27 16:10:33.066867", + "payload": {"status": "active", + "name": "example", + "backend": "ceph2", + "os_glance_importing_to_stores": ["ceph2"], + "os_glance_failed_import": [], + ...}, + "message_id": "1c8993ad-e47c-4af7-9f75-fa49596eeb18", + ... + } + + { + "priority": "ERROR", + "event_type": "image.upload", + "timestamp": "2019-08-27 16:10:34.058812", + "payload": "Error Message", + "message_id": "8b8993ad-e47c-4af7-9f75-fa49596eeb11", + ... + } + Customizing the image import process ------------------------------------ diff --git a/glance/api/v2/images.py b/glance/api/v2/images.py index 2259802767..8c80ba5a49 100644 --- a/glance/api/v2/images.py +++ b/glance/api/v2/images.py @@ -105,6 +105,7 @@ class ImagesController(object): task_repo = self.gateway.get_task_repo(req.context) import_method = body.get('method').get('name') uri = body.get('method').get('uri') + all_stores_must_succeed = body.get('all_stores_must_succeed', True) try: image = image_repo.get(image_id) @@ -127,24 +128,26 @@ class ImagesController(object): msg = _("'disk_format' needs to be set before import") raise exception.Conflict(msg) - backend = None + stores = [None] if CONF.enabled_backends: - backend = req.headers.get('x-image-meta-store', - CONF.glance_store.default_backend) try: - glance_store.get_store_from_store_identifier(backend) - except glance_store.UnknownScheme: - msg = _("Store for scheme %s not found") % backend - LOG.warn(msg) - raise exception.Conflict(msg) + stores = utils.get_stores_from_request(req, body) + except glance_store.UnknownScheme as exc: + LOG.warn(exc.msg) + raise exception.Conflict(exc.msg) except exception.Conflict as e: raise webob.exc.HTTPConflict(explanation=e.msg) except exception.NotFound as e: raise webob.exc.HTTPNotFound(explanation=e.msg) + if (not all_stores_must_succeed) and (not CONF.enabled_backends): + msg = (_("All_stores_must_succeed can only be set with " + "enabled_backends %s") % uri) + raise webob.exc.HTTPBadRequest(explanation=msg) + task_input = {'image_id': image_id, 'import_req': body, - 'backend': backend} + 'backend': stores} if (import_method == 'web-download' and not utils.validate_import_uri(uri)): diff --git a/glance/async_/flows/api_image_import.py b/glance/async_/flows/api_image_import.py index 7a28e7ff0f..a4828185f5 100644 --- a/glance/async_/flows/api_image_import.py +++ b/glance/async_/flows/api_image_import.py @@ -16,6 +16,7 @@ import os import glance_store as store_api from glance_store import backend +from glance_store import exceptions as store_exceptions from oslo_config import cfg from oslo_log import log as logging from oslo_utils import encodeutils @@ -188,13 +189,16 @@ class _VerifyStaging(task.Task): class _ImportToStore(task.Task): - def __init__(self, task_id, task_type, image_repo, uri, image_id, backend): + def __init__(self, task_id, task_type, image_repo, uri, image_id, backend, + all_stores_must_succeed, set_active): self.task_id = task_id self.task_type = task_type self.image_repo = image_repo self.uri = uri self.image_id = image_id self.backend = backend + self.all_stores_must_succeed = all_stores_must_succeed + self.set_active = set_active super(_ImportToStore, self).__init__( name='%s-ImportToStore-%s' % (task_type, task_id)) @@ -245,13 +249,74 @@ class _ImportToStore(task.Task): # will need the file path anyways for our delete workflow for now. # For future proofing keeping this as is. image = self.image_repo.get(self.image_id) - image_import.set_image_data(image, file_path or self.uri, self.task_id, - backend=self.backend) - - # NOTE(flaper87): We need to save the image again after the locations - # have been set in the image. + if image.status == "deleted": + raise exception.ImportTaskError("Image has been deleted, aborting" + " import.") + try: + image_import.set_image_data(image, file_path or self.uri, + self.task_id, backend=self.backend, + set_active=self.set_active) + # NOTE(yebinama): set_image_data catches Exception and raises from + # them. Can't be more specific on exceptions catched. + except Exception: + if self.all_stores_must_succeed: + raise + msg = (_("%(task_id)s of %(task_type)s failed but since " + "all_stores_must_succeed is set to false, continue.") % + {'task_id': self.task_id, 'task_type': self.task_type}) + LOG.warning(msg) + if self.backend is not None: + failed_import = image.extra_properties.get( + 'os_glance_failed_import', '').split(',') + failed_import.append(self.backend) + image.extra_properties['os_glance_failed_import'] = ','.join( + failed_import).lstrip(',') + if self.backend is not None: + importing = image.extra_properties.get( + 'os_glance_importing_to_stores', '').split(',') + try: + importing.remove(self.backend) + image.extra_properties[ + 'os_glance_importing_to_stores'] = ','.join( + importing).lstrip(',') + except ValueError: + LOG.debug("Store %s not found in property " + "os_glance_importing_to_stores.", self.backend) + # NOTE(flaper87): We need to save the image again after + # the locations have been set in the image. self.image_repo.save(image) + def revert(self, result, **kwargs): + """ + Remove location from image in case of failure + + :param result: taskflow result object + """ + image = self.image_repo.get(self.image_id) + for i, location in enumerate(image.locations): + if location.get('metadata', {}).get('store') == self.backend: + try: + image.locations.pop(i) + except (store_exceptions.NotFound, + store_exceptions.Forbidden): + msg = (_("Error deleting from store %{store}s when " + "reverting.") % {'store': self.backend}) + LOG.warning(msg) + # NOTE(yebinama): Some store drivers doesn't document which + # exceptions they throw. + except Exception: + msg = (_("Unexpected exception when deleting from store" + "%{store}s.") % {'store': self.backend}) + LOG.warning(msg) + else: + if len(image.locations) == 0: + image.checksum = None + image.os_hash_algo = None + image.os_hash_value = None + image.size = None + self.image_repo.save(image) + break + class _SaveImage(task.Task): @@ -337,7 +402,9 @@ def get_flow(**kwargs): image_id = kwargs.get('image_id') import_method = kwargs.get('import_req')['method']['name'] uri = kwargs.get('import_req')['method'].get('uri') - backend = kwargs.get('backend') + stores = kwargs.get('backend', [None]) + all_stores_must_succeed = kwargs.get('import_req').get( + 'all_stores_must_succeed', True) separator = '' if not CONF.enabled_backends and not CONF.node_staging_uri.endswith('/'): @@ -368,13 +435,20 @@ def get_flow(**kwargs): for plugin in import_plugins.get_import_plugins(**kwargs): flow.add(plugin) - import_to_store = _ImportToStore(task_id, - task_type, - image_repo, - file_uri, - image_id, - backend) - flow.add(import_to_store) + for idx, store in enumerate(stores, 1): + set_active = (not all_stores_must_succeed) or (idx == len(stores)) + task_name = task_type + "-" + (store or "") + import_task = lf.Flow(task_name) + import_to_store = _ImportToStore(task_id, + task_name, + image_repo, + file_uri, + image_id, + store, + all_stores_must_succeed, + set_active) + import_task.add(import_to_store) + flow.add(import_task) delete_task = lf.Flow(task_type).add(_DeleteFromFS(task_id, task_type)) flow.add(delete_task) @@ -394,6 +468,11 @@ def get_flow(**kwargs): image = image_repo.get(image_id) from_state = image.status image.status = 'importing' + image.extra_properties[ + 'os_glance_importing_to_stores'] = ','.join((store for store in + stores if + store is not None)) + image.extra_properties['os_glance_failed_import'] = '' image_repo.save(image, from_state=from_state) return flow diff --git a/glance/common/scripts/image_import/main.py b/glance/common/scripts/image_import/main.py index 9900a595ad..25116fc77c 100644 --- a/glance/common/scripts/image_import/main.py +++ b/glance/common/scripts/image_import/main.py @@ -137,13 +137,13 @@ def create_image(image_repo, image_factory, image_properties, task_id): return image -def set_image_data(image, uri, task_id, backend=None): +def set_image_data(image, uri, task_id, backend=None, set_active=True): data_iter = None try: LOG.info(_LI("Task %(task_id)s: Got image data uri %(data_uri)s to be " "imported"), {"data_uri": uri, "task_id": task_id}) data_iter = script_utils.get_image_data_iter(uri) - image.set_data(data_iter, backend=backend) + image.set_data(data_iter, backend=backend, set_active=set_active) except Exception as e: with excutils.save_and_reraise_exception(): LOG.warn(_LW("Task %(task_id)s failed with exception %(error)s") % diff --git a/glance/common/utils.py b/glance/common/utils.py index b4144f1823..b7cf164eb3 100644 --- a/glance/common/utils.py +++ b/glance/common/utils.py @@ -29,6 +29,7 @@ except ImportError: from eventlet.green import socket import functools +import glance_store import os import re @@ -668,3 +669,35 @@ def evaluate_filter_op(value, operator, threshold): msg = _("Unable to filter on a unknown operator.") raise exception.InvalidFilterOperatorValue(msg) + + +def get_stores_from_request(req, body): + """Processes a supplied request and extract stores from it + + :param req: request to process + :param body: request body + + :raises glance_store.UnknownScheme: if a store is not valid + :return: a list of stores + """ + if body.get('all_stores', False): + if 'stores' in body or 'x-image-meta-store' in req.headers: + msg = _("All_stores parameter can't be used with " + "x-image-meta-store header or stores parameter") + raise exc.HTTPBadRequest(explanation=msg) + stores = list(CONF.enabled_backends) + else: + try: + stores = body['stores'] + except KeyError: + stores = [req.headers.get('x-image-meta-store', + CONF.glance_store.default_backend)] + else: + if 'x-image-meta-store' in req.headers: + msg = _("Stores parameter and x-image-meta-store header can't " + "be both specified") + raise exc.HTTPBadRequest(explanation=msg) + # Validate each store + for store in stores: + glance_store.get_store_from_store_identifier(store) + return stores diff --git a/glance/domain/__init__.py b/glance/domain/__init__.py index 6222d70388..88fc162476 100644 --- a/glance/domain/__init__.py +++ b/glance/domain/__init__.py @@ -289,7 +289,7 @@ class Image(object): def get_data(self, *args, **kwargs): raise NotImplementedError() - def set_data(self, data, size=None, backend=None): + def set_data(self, data, size=None, backend=None, set_active=True): raise NotImplementedError() diff --git a/glance/domain/proxy.py b/glance/domain/proxy.py index 60c54a9246..391df84de5 100644 --- a/glance/domain/proxy.py +++ b/glance/domain/proxy.py @@ -194,8 +194,8 @@ class Image(object): def reactivate(self): self.base.reactivate() - def set_data(self, data, size=None, backend=None): - self.base.set_data(data, size, backend=backend) + def set_data(self, data, size=None, backend=None, set_active=True): + self.base.set_data(data, size, backend=backend, set_active=set_active) def get_data(self, *args, **kwargs): return self.base.get_data(*args, **kwargs) diff --git a/glance/location.py b/glance/location.py index e2ecd901e6..d6991e5c9f 100644 --- a/glance/location.py +++ b/glance/location.py @@ -436,31 +436,17 @@ class ImageProxy(glance.domain.proxy.Image): self.image.image_id, location) - def set_data(self, data, size=None, backend=None): - if size is None: - size = 0 # NOTE(markwash): zero -> unknown size + def _upload_to_store(self, data, verifier, store=None, size=None): + """ + Upload data to store - # Create the verifier for signature verification (if correct properties - # are present) - extra_props = self.image.extra_properties - if (signature_utils.should_create_verifier(extra_props)): - # NOTE(bpoulos): if creating verifier fails, exception will be - # raised - img_signature = extra_props[signature_utils.SIGNATURE] - hash_method = extra_props[signature_utils.HASH_METHOD] - key_type = extra_props[signature_utils.KEY_TYPE] - cert_uuid = extra_props[signature_utils.CERT_UUID] - verifier = signature_utils.get_verifier( - context=self.context, - img_signature_certificate_uuid=cert_uuid, - img_signature_hash_method=hash_method, - img_signature=img_signature, - img_signature_key_type=key_type - ) - else: - verifier = None - - hashing_algo = CONF['hashing_algorithm'] + :param data: data to upload to store + :param verifier: for signature verification + :param store: store to upload data to + :param size: data size + :return: + """ + hashing_algo = self.image.os_hash_algo or CONF['hashing_algorithm'] if CONF.enabled_backends: (location, size, checksum, multihash, loc_meta) = self.store_api.add_with_multihash( @@ -469,7 +455,7 @@ class ImageProxy(glance.domain.proxy.Image): utils.LimitingReader(utils.CooperativeReader(data), CONF.image_size_cap), size, - backend, + store, hashing_algo, context=self.context, verifier=verifier) @@ -487,16 +473,33 @@ class ImageProxy(glance.domain.proxy.Image): hashing_algo, context=self.context, verifier=verifier) + self._verify_signature(verifier, location, loc_meta) + for attr, data in {"size": size, "os_hash_value": multihash, + "checksum": checksum}.items(): + self._verify_uploaded_data(data, attr) + self.image.locations.append({'url': location, 'metadata': loc_meta, + 'status': 'active'}) + self.image.checksum = checksum + self.image.os_hash_value = multihash + self.image.size = size + self.image.os_hash_algo = hashing_algo + def _verify_signature(self, verifier, location, loc_meta): + """ + Verify signature of uploaded data. + + :param verifier: for signature verification + """ # NOTE(bpoulos): if verification fails, exception will be raised - if verifier: + if verifier is not None: try: verifier.verify() - LOG.info(_LI("Successfully verified signature for image %s"), - self.image.image_id) + msg = _LI("Successfully verified signature for image %s") + LOG.info(msg, self.image.image_id) except crypto_exception.InvalidSignature: if CONF.enabled_backends: - self.store_api.delete(location, loc_meta.get('store'), + self.store_api.delete(location, + loc_meta.get('store'), context=self.context) else: self.store_api.delete_from_backend(location, @@ -505,13 +508,46 @@ class ImageProxy(glance.domain.proxy.Image): _('Signature verification failed') ) - self.image.locations = [{'url': location, 'metadata': loc_meta, - 'status': 'active'}] - self.image.size = size - self.image.checksum = checksum - self.image.os_hash_value = multihash - self.image.os_hash_algo = hashing_algo - self.image.status = 'active' + def _verify_uploaded_data(self, value, attribute_name): + """ + Verify value of attribute_name uploaded data + + :param value: value to compare + :param attribute_name: attribute name of the image to compare with + """ + image_value = getattr(self.image, attribute_name) + if image_value is not None and value != image_value: + msg = _("%s of uploaded data is different from current " + "value set on the image.") + LOG.error(msg, attribute_name) + raise exception.UploadException(msg % attribute_name) + + def set_data(self, data, size=None, backend=None, set_active=True): + if size is None: + size = 0 # NOTE(markwash): zero -> unknown size + + # Create the verifier for signature verification (if correct properties + # are present) + extra_props = self.image.extra_properties + verifier = None + if signature_utils.should_create_verifier(extra_props): + # NOTE(bpoulos): if creating verifier fails, exception will be + # raised + img_signature = extra_props[signature_utils.SIGNATURE] + hash_method = extra_props[signature_utils.HASH_METHOD] + key_type = extra_props[signature_utils.KEY_TYPE] + cert_uuid = extra_props[signature_utils.CERT_UUID] + verifier = signature_utils.get_verifier( + context=self.context, + img_signature_certificate_uuid=cert_uuid, + img_signature_hash_method=hash_method, + img_signature=img_signature, + img_signature_key_type=key_type + ) + + self._upload_to_store(data, verifier, backend, size) + if set_active and self.image.status != 'active': + self.image.status = 'active' def get_data(self, offset=0, chunk_size=None): if not self.image.locations: diff --git a/glance/notifier.py b/glance/notifier.py index b08e183ba7..5bc8955020 100644 --- a/glance/notifier.py +++ b/glance/notifier.py @@ -400,6 +400,17 @@ class ImageProxy(NotificationProxy, domain_proxy.Image): 'receiver_user_id': self.context.user_id, } + def _format_import_properties(self): + importing = self.repo.extra_properties.get( + 'os_glance_importing_to_stores') + importing = importing.split(',') if importing else [] + failed = self.repo.extra_properties.get('os_glance_failed_import') + failed = failed.split(',') if failed else [] + return { + 'os_glance_importing_to_stores': importing, + 'os_glance_failed_import': failed + } + def _get_chunk_data_iterator(self, data, chunk_size=None): sent = 0 for chunk in data: @@ -426,12 +437,15 @@ class ImageProxy(NotificationProxy, domain_proxy.Image): data = self.repo.get_data(offset=offset, chunk_size=chunk_size) return self._get_chunk_data_iterator(data, chunk_size=chunk_size) - def set_data(self, data, size=None, backend=None): - self.send_notification('image.prepare', self.repo, backend=backend) + def set_data(self, data, size=None, backend=None, set_active=True): + self.send_notification('image.prepare', self.repo, backend=backend, + extra_payload=self._format_import_properties()) notify_error = self.notifier.error + status = self.repo.status try: - self.repo.set_data(data, size, backend=backend) + self.repo.set_data(data, size, backend=backend, + set_active=set_active) except glance_store.StorageFull as e: msg = (_("Image storage media is full: %s") % encodeutils.exception_to_unicode(e)) @@ -486,8 +500,11 @@ class ImageProxy(NotificationProxy, domain_proxy.Image): 'error': encodeutils.exception_to_unicode(e)}) _send_notification(notify_error, 'image.upload', msg) else: - self.send_notification('image.upload', self.repo) - self.send_notification('image.activate', self.repo) + extra_payload = self._format_import_properties() + self.send_notification('image.upload', self.repo, + extra_payload=extra_payload) + if set_active and status != 'active': + self.send_notification('image.activate', self.repo) class ImageMemberProxy(NotificationProxy, domain_proxy.ImageMember): diff --git a/glance/quota/__init__.py b/glance/quota/__init__.py index 9d8af363d4..1ac97162b7 100644 --- a/glance/quota/__init__.py +++ b/glance/quota/__init__.py @@ -306,7 +306,7 @@ class ImageProxy(glance.domain.proxy.Image): super(ImageProxy, self).__init__(image) self.orig_props = set(image.extra_properties.keys()) - def set_data(self, data, size=None, backend=None): + def set_data(self, data, size=None, backend=None, set_active=True): remaining = glance.api.common.check_quota( self.context, size, self.db_api, image_id=self.image.image_id) if remaining is not None: @@ -315,7 +315,8 @@ class ImageProxy(glance.domain.proxy.Image): data = utils.LimitingReader( data, remaining, exception_class=exception.StorageQuotaFull) - self.image.set_data(data, size=size, backend=backend) + self.image.set_data(data, size=size, backend=backend, + set_active=set_active) # NOTE(jbresnah) If two uploads happen at the same time and neither # properly sets the size attribute[1] then there is a race condition diff --git a/glance/tests/functional/v2/test_images.py b/glance/tests/functional/v2/test_images.py index c3d7d34157..4b87adfc95 100644 --- a/glance/tests/functional/v2/test_images.py +++ b/glance/tests/functional/v2/test_images.py @@ -5134,6 +5134,332 @@ class TestImagesMultipleBackend(functional.MultipleBackendFunctionalTest): self.stop_servers() + def test_image_import_multi_stores(self): + self.config(node_staging_uri="file:///tmp/staging/") + self.start_servers(**self.__dict__.copy()) + + # Image list should be empty + path = self._url('/v2/images') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + images = jsonutils.loads(response.text)['images'] + self.assertEqual(0, len(images)) + + # web-download should be available in discovery response + path = self._url('/v2/info/import') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + discovery_calls = jsonutils.loads( + response.text)['import-methods']['value'] + self.assertIn("web-download", discovery_calls) + + # file1 and file2 should be available in discovery response + available_stores = ['file1', 'file2'] + path = self._url('/v2/info/stores') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + discovery_calls = jsonutils.loads( + response.text)['stores'] + # os_glance_staging_store should not be available in discovery response + for stores in discovery_calls: + self.assertIn('id', stores) + self.assertIn(stores['id'], available_stores) + self.assertFalse(stores["id"].startswith("os_glance_")) + + # Create an image + path = self._url('/v2/images') + headers = self._headers({'content-type': 'application/json'}) + data = jsonutils.dumps({'name': 'image-1', 'type': 'kernel', + 'disk_format': 'aki', + 'container_format': 'aki'}) + response = requests.post(path, headers=headers, data=data) + self.assertEqual(http.CREATED, response.status_code) + + # Check 'OpenStack-image-store-ids' header present in response + self.assertIn('OpenStack-image-store-ids', response.headers) + for store in available_stores: + self.assertIn(store, response.headers['OpenStack-image-store-ids']) + + # Returned image entity should have a generated id and status + image = jsonutils.loads(response.text) + image_id = image['id'] + checked_keys = set([ + u'status', + u'name', + u'tags', + u'created_at', + u'updated_at', + u'visibility', + u'self', + u'protected', + u'id', + u'file', + u'min_disk', + u'type', + u'min_ram', + u'schema', + u'disk_format', + u'container_format', + u'owner', + u'checksum', + u'size', + u'virtual_size', + u'os_hidden', + u'os_hash_algo', + u'os_hash_value' + ]) + self.assertEqual(checked_keys, set(image.keys())) + expected_image = { + 'status': 'queued', + 'name': 'image-1', + 'tags': [], + 'visibility': 'shared', + 'self': '/v2/images/%s' % image_id, + 'protected': False, + 'file': '/v2/images/%s/file' % image_id, + 'min_disk': 0, + 'type': 'kernel', + 'min_ram': 0, + 'schema': '/v2/schemas/image', + } + for key, value in expected_image.items(): + self.assertEqual(value, image[key], key) + + # Image list should now have one entry + path = self._url('/v2/images') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + images = jsonutils.loads(response.text)['images'] + self.assertEqual(1, len(images)) + self.assertEqual(image_id, images[0]['id']) + + # Verify image is in queued state and checksum is None + func_utils.verify_image_hashes_and_status(self, image_id, + status='queued') + # Import image to multiple stores + path = self._url('/v2/images/%s/import' % image_id) + headers = self._headers({ + 'content-type': 'application/json', + 'X-Roles': 'admin' + }) + + # Start http server locally + thread, httpd, port = test_utils.start_standalone_http_server() + + image_data_uri = 'http://localhost:%s/' % port + data = jsonutils.dumps( + {'method': {'name': 'web-download', 'uri': image_data_uri}, + 'stores': ['file1', 'file2']}) + response = requests.post(path, headers=headers, data=data) + self.assertEqual(http.ACCEPTED, response.status_code) + + # Verify image is in active state and checksum is set + # NOTE(abhishekk): As import is a async call we need to provide + # some timelap to complete the call. + path = self._url('/v2/images/%s' % image_id) + func_utils.wait_for_status(request_path=path, + request_headers=self._headers(), + status='active', + max_sec=40, + delay_sec=0.2, + start_delay_sec=1) + with requests.get(image_data_uri) as r: + expect_c = six.text_type(hashlib.md5(r.content).hexdigest()) + expect_h = six.text_type(hashlib.sha512(r.content).hexdigest()) + func_utils.verify_image_hashes_and_status(self, + image_id, + checksum=expect_c, + os_hash_value=expect_h, + status='active') + + # kill the local http server + httpd.shutdown() + httpd.server_close() + + # Ensure image is created in the two stores + path = self._url('/v2/images/%s' % image_id) + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + self.assertIn('file2', jsonutils.loads(response.text)['stores']) + self.assertIn('file1', jsonutils.loads(response.text)['stores']) + + # Deleting image should work + path = self._url('/v2/images/%s' % image_id) + response = requests.delete(path, headers=self._headers()) + self.assertEqual(http.NO_CONTENT, response.status_code) + + # Image list should now be empty + path = self._url('/v2/images') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + images = jsonutils.loads(response.text)['images'] + self.assertEqual(0, len(images)) + + self.stop_servers() + + def test_image_import_multi_stores_specifying_all_stores(self): + self.config(node_staging_uri="file:///tmp/staging/") + self.start_servers(**self.__dict__.copy()) + + # Image list should be empty + path = self._url('/v2/images') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + images = jsonutils.loads(response.text)['images'] + self.assertEqual(0, len(images)) + + # web-download should be available in discovery response + path = self._url('/v2/info/import') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + discovery_calls = jsonutils.loads( + response.text)['import-methods']['value'] + self.assertIn("web-download", discovery_calls) + + # file1 and file2 should be available in discovery response + available_stores = ['file1', 'file2'] + path = self._url('/v2/info/stores') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + discovery_calls = jsonutils.loads( + response.text)['stores'] + # os_glance_staging_store should not be available in discovery response + for stores in discovery_calls: + self.assertIn('id', stores) + self.assertIn(stores['id'], available_stores) + self.assertFalse(stores["id"].startswith("os_glance_")) + + # Create an image + path = self._url('/v2/images') + headers = self._headers({'content-type': 'application/json'}) + data = jsonutils.dumps({'name': 'image-1', 'type': 'kernel', + 'disk_format': 'aki', + 'container_format': 'aki'}) + response = requests.post(path, headers=headers, data=data) + self.assertEqual(http.CREATED, response.status_code) + + # Check 'OpenStack-image-store-ids' header present in response + self.assertIn('OpenStack-image-store-ids', response.headers) + for store in available_stores: + self.assertIn(store, response.headers['OpenStack-image-store-ids']) + + # Returned image entity should have a generated id and status + image = jsonutils.loads(response.text) + image_id = image['id'] + checked_keys = set([ + u'status', + u'name', + u'tags', + u'created_at', + u'updated_at', + u'visibility', + u'self', + u'protected', + u'id', + u'file', + u'min_disk', + u'type', + u'min_ram', + u'schema', + u'disk_format', + u'container_format', + u'owner', + u'checksum', + u'size', + u'virtual_size', + u'os_hidden', + u'os_hash_algo', + u'os_hash_value' + ]) + self.assertEqual(checked_keys, set(image.keys())) + expected_image = { + 'status': 'queued', + 'name': 'image-1', + 'tags': [], + 'visibility': 'shared', + 'self': '/v2/images/%s' % image_id, + 'protected': False, + 'file': '/v2/images/%s/file' % image_id, + 'min_disk': 0, + 'type': 'kernel', + 'min_ram': 0, + 'schema': '/v2/schemas/image', + } + for key, value in expected_image.items(): + self.assertEqual(value, image[key], key) + + # Image list should now have one entry + path = self._url('/v2/images') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + images = jsonutils.loads(response.text)['images'] + self.assertEqual(1, len(images)) + self.assertEqual(image_id, images[0]['id']) + + # Verify image is in queued state and checksum is None + func_utils.verify_image_hashes_and_status(self, image_id, + status='queued') + # Import image to multiple stores + path = self._url('/v2/images/%s/import' % image_id) + headers = self._headers({ + 'content-type': 'application/json', + 'X-Roles': 'admin' + }) + + # Start http server locally + thread, httpd, port = test_utils.start_standalone_http_server() + + image_data_uri = 'http://localhost:%s/' % port + data = jsonutils.dumps( + {'method': {'name': 'web-download', 'uri': image_data_uri}, + 'all_stores': True}) + response = requests.post(path, headers=headers, data=data) + self.assertEqual(http.ACCEPTED, response.status_code) + + # Verify image is in active state and checksum is set + # NOTE(abhishekk): As import is a async call we need to provide + # some timelap to complete the call. + path = self._url('/v2/images/%s' % image_id) + func_utils.wait_for_status(request_path=path, + request_headers=self._headers(), + status='active', + max_sec=40, + delay_sec=0.2, + start_delay_sec=1) + with requests.get(image_data_uri) as r: + expect_c = six.text_type(hashlib.md5(r.content).hexdigest()) + expect_h = six.text_type(hashlib.sha512(r.content).hexdigest()) + func_utils.verify_image_hashes_and_status(self, + image_id, + checksum=expect_c, + os_hash_value=expect_h, + status='active') + + # kill the local http server + httpd.shutdown() + httpd.server_close() + + # Ensure image is created in the two stores + path = self._url('/v2/images/%s' % image_id) + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + self.assertIn('file2', jsonutils.loads(response.text)['stores']) + self.assertIn('file1', jsonutils.loads(response.text)['stores']) + + # Deleting image should work + path = self._url('/v2/images/%s' % image_id) + response = requests.delete(path, headers=self._headers()) + self.assertEqual(http.NO_CONTENT, response.status_code) + + # Image list should now be empty + path = self._url('/v2/images') + response = requests.get(path, headers=self._headers()) + self.assertEqual(http.OK, response.status_code) + images = jsonutils.loads(response.text)['images'] + self.assertEqual(0, len(images)) + + self.stop_servers() + def test_image_lifecycle(self): # Image list should be empty self.start_servers(**self.__dict__.copy()) diff --git a/glance/tests/unit/async_/flows/test_api_image_import.py b/glance/tests/unit/async_/flows/test_api_image_import.py index 5519027891..53f8c9bafb 100644 --- a/glance/tests/unit/async_/flows/test_api_image_import.py +++ b/glance/tests/unit/async_/flows/test_api_image_import.py @@ -18,13 +18,20 @@ import mock from oslo_config import cfg import glance.async_.flows.api_image_import as import_flow +from glance.common.exception import ImportTaskError +from glance import context +from glance import gateway import glance.tests.utils as test_utils +from cursive import exception as cursive_exception + CONF = cfg.CONF TASK_TYPE = 'api_image_import' TASK_ID1 = 'dbbe7231-020f-4311-87e1-5aaa6da56c02' IMAGE_ID1 = '41f5b3b0-f54c-4cef-bd45-ce3e376a142f' +UUID1 = 'c80a1a6c-bd1f-41c5-90ee-81afedb1d58d' +TENANT1 = '6838eb7b-6ded-434a-882c-b344c77fe8df' class TestApiImageImportTask(test_utils.BaseTestCase): @@ -88,3 +95,74 @@ class TestApiImageImportTask(test_utils.BaseTestCase): import_req=self.wd_task_input['import_req']) self._pass_uri(uri=test_uri, file_uri=expected_uri, import_req=self.gd_task_input['import_req']) + + +class TestImportToStoreTask(test_utils.BaseTestCase): + + def setUp(self): + super(TestImportToStoreTask, self).setUp() + self.gateway = gateway.Gateway() + self.context = context.RequestContext(user_id=TENANT1, + project_id=TENANT1, + overwrite=False) + self.img_factory = self.gateway.get_image_factory(self.context) + + def test_raises_when_image_deleted(self): + img_repo = mock.MagicMock() + image_import = import_flow._ImportToStore(TASK_ID1, TASK_TYPE, + img_repo, "http://url", + IMAGE_ID1, "store1", False, + True) + image = self.img_factory.new_image(image_id=UUID1) + image.status = "deleted" + img_repo.get.return_value = image + self.assertRaises(ImportTaskError, image_import.execute) + + @mock.patch("glance.async_.flows.api_image_import.image_import") + def test_remove_store_from_property(self, mock_import): + img_repo = mock.MagicMock() + image_import = import_flow._ImportToStore(TASK_ID1, TASK_TYPE, + img_repo, "http://url", + IMAGE_ID1, "store1", True, + True) + extra_properties = {"os_glance_importing_to_stores": "store1,store2"} + image = self.img_factory.new_image(image_id=UUID1, + extra_properties=extra_properties) + img_repo.get.return_value = image + image_import.execute() + self.assertEqual( + image.extra_properties['os_glance_importing_to_stores'], "store2") + + @mock.patch("glance.async_.flows.api_image_import.image_import") + def test_raises_when_all_stores_must_succeed(self, mock_import): + img_repo = mock.MagicMock() + image_import = import_flow._ImportToStore(TASK_ID1, TASK_TYPE, + img_repo, "http://url", + IMAGE_ID1, "store1", True, + True) + image = self.img_factory.new_image(image_id=UUID1) + img_repo.get.return_value = image + mock_import.set_image_data.side_effect = \ + cursive_exception.SignatureVerificationError( + "Signature verification failed") + self.assertRaises(cursive_exception.SignatureVerificationError, + image_import.execute) + + @mock.patch("glance.async_.flows.api_image_import.image_import") + def test_doesnt_raise_when_not_all_stores_must_succeed(self, mock_import): + img_repo = mock.MagicMock() + image_import = import_flow._ImportToStore(TASK_ID1, TASK_TYPE, + img_repo, "http://url", + IMAGE_ID1, "store1", False, + True) + image = self.img_factory.new_image(image_id=UUID1) + img_repo.get.return_value = image + mock_import.set_image_data.side_effect = \ + cursive_exception.SignatureVerificationError( + "Signature verification failed") + try: + image_import.execute() + self.assertEqual(image.extra_properties['os_glance_failed_import'], + "store1") + except cursive_exception.SignatureVerificationError: + self.fail("Exception shouldn't be raised") diff --git a/glance/tests/unit/common/test_utils.py b/glance/tests/unit/common/test_utils.py index a19855d9bf..0dddf725b6 100644 --- a/glance/tests/unit/common/test_utils.py +++ b/glance/tests/unit/common/test_utils.py @@ -14,9 +14,11 @@ # License for the specific language governing permissions and limitations # under the License. +import glance_store as store import mock import tempfile +from oslo_config import cfg from oslo_log import log as logging import six import webob @@ -27,6 +29,9 @@ from glance.common import utils from glance.tests import utils as test_utils +CONF = cfg.CONF + + class TestStoreUtils(test_utils.BaseTestCase): """Test glance.common.store_utils module""" @@ -408,6 +413,104 @@ class TestUtils(test_utils.BaseTestCase): utils.parse_valid_host_port, pair) + def test_get_stores_from_request_returns_default(self): + enabled_backends = { + "ceph1": "rbd", + "ceph2": "rbd" + } + self.config(enabled_backends=enabled_backends) + store.register_store_opts(CONF) + self.config(default_backend="ceph1", group="glance_store") + + req = webob.Request.blank('/some_request') + mp = "glance.common.utils.glance_store.get_store_from_store_identifier" + with mock.patch(mp) as mock_get_store: + result = utils.get_stores_from_request(req, {}) + self.assertEqual(["ceph1"], result) + mock_get_store.assert_called_once_with("ceph1") + + def test_get_stores_from_request_returns_stores_from_body(self): + enabled_backends = { + "ceph1": "rbd", + "ceph2": "rbd" + } + self.config(enabled_backends=enabled_backends) + store.register_store_opts(CONF) + self.config(default_backend="ceph1", group="glance_store") + + body = {"stores": ["ceph1", "ceph2"]} + req = webob.Request.blank("/some_request") + mp = "glance.common.utils.glance_store.get_store_from_store_identifier" + with mock.patch(mp) as mock_get_store: + result = utils.get_stores_from_request(req, body) + self.assertEqual(["ceph1", "ceph2"], result) + mock_get_store.assert_any_call("ceph1") + mock_get_store.assert_any_call("ceph2") + self.assertEqual(mock_get_store.call_count, 2) + + def test_get_stores_from_request_returns_store_from_headers(self): + enabled_backends = { + "ceph1": "rbd", + "ceph2": "rbd" + } + self.config(enabled_backends=enabled_backends) + store.register_store_opts(CONF) + self.config(default_backend="ceph1", group="glance_store") + + headers = {"x-image-meta-store": "ceph2"} + req = webob.Request.blank("/some_request", headers=headers) + mp = "glance.common.utils.glance_store.get_store_from_store_identifier" + with mock.patch(mp) as mock_get_store: + result = utils.get_stores_from_request(req, {}) + self.assertEqual(["ceph2"], result) + mock_get_store.assert_called_once_with("ceph2") + + def test_get_stores_from_request_raises_bad_request(self): + enabled_backends = { + "ceph1": "rbd", + "ceph2": "rbd" + } + self.config(enabled_backends=enabled_backends) + store.register_store_opts(CONF) + self.config(default_backend="ceph1", group="glance_store") + headers = {"x-image-meta-store": "ceph2"} + body = {"stores": ["ceph1"]} + req = webob.Request.blank("/some_request", headers=headers) + self.assertRaises(webob.exc.HTTPBadRequest, + utils.get_stores_from_request, req, body) + + def test_get_stores_from_request_returns_all_stores(self): + enabled_backends = { + "ceph1": "rbd", + "ceph2": "rbd" + } + self.config(enabled_backends=enabled_backends) + store.register_store_opts(CONF) + self.config(default_backend="ceph1", group="glance_store") + body = {"all_stores": True} + req = webob.Request.blank("/some_request") + mp = "glance.common.utils.glance_store.get_store_from_store_identifier" + with mock.patch(mp) as mock_get_store: + result = sorted(utils.get_stores_from_request(req, body)) + self.assertEqual(["ceph1", "ceph2"], result) + mock_get_store.assert_any_call("ceph1") + mock_get_store.assert_any_call("ceph2") + self.assertEqual(mock_get_store.call_count, 2) + + def test_get_stores_from_request_raises_bad_request_with_all_stores(self): + enabled_backends = { + "ceph1": "rbd", + "ceph2": "rbd" + } + self.config(enabled_backends=enabled_backends) + store.register_store_opts(CONF) + self.config(default_backend="ceph1", group="glance_store") + headers = {"x-image-meta-store": "ceph2"} + body = {"stores": ["ceph1"], "all_stores": True} + req = webob.Request.blank("/some_request", headers=headers) + self.assertRaises(webob.exc.HTTPBadRequest, + utils.get_stores_from_request, req, body) + class SplitFilterOpTestCase(test_utils.BaseTestCase): diff --git a/glance/tests/unit/test_cache_middleware.py b/glance/tests/unit/test_cache_middleware.py index 58f4591d70..6d03552230 100644 --- a/glance/tests/unit/test_cache_middleware.py +++ b/glance/tests/unit/test_cache_middleware.py @@ -38,6 +38,7 @@ class ImageStub(object): self.extra_properties = extra_properties self.checksum = 'c1234' self.size = 123456789 + self.os_hash_algo = None class TestCacheMiddlewareURLMatching(testtools.TestCase): diff --git a/glance/tests/unit/test_notifier.py b/glance/tests/unit/test_notifier.py index 346ef5a634..c75ce24888 100644 --- a/glance/tests/unit/test_notifier.py +++ b/glance/tests/unit/test_notifier.py @@ -44,7 +44,7 @@ class ImageStub(glance.domain.Image): def get_data(self, offset=0, chunk_size=None): return ['01234', '56789'] - def set_data(self, data, size, backend=None): + def set_data(self, data, size, backend=None, set_active=True): for chunk in data: pass @@ -272,10 +272,17 @@ class TestImageNotifications(utils.BaseTestCase): self.assertEqual('INFO', output_log['notification_type']) self.assertEqual('image.prepare', output_log['event_type']) self.assertEqual(self.image.image_id, output_log['payload']['id']) + self.assertEqual(['store1', 'store2'], output_log['payload'][ + 'os_glance_importing_to_stores']) + self.assertEqual([], + output_log['payload']['os_glance_failed_import']) yield 'abcd' yield 'efgh' insurance['called'] = True + self.image_proxy.extra_properties[ + 'os_glance_importing_to_stores'] = 'store1,store2' + self.image_proxy.extra_properties['os_glance_failed_import'] = '' self.image_proxy.set_data(data_iterator(), 8) self.assertTrue(insurance['called']) @@ -294,39 +301,83 @@ class TestImageNotifications(utils.BaseTestCase): self.assertTrue(insurance['called']) def test_image_set_data_upload_and_activate_notification(self): + image = ImageStub(image_id=UUID1, name='image-1', status='queued', + created_at=DATETIME, updated_at=DATETIME, + owner=TENANT1, visibility='public') + context = glance.context.RequestContext(tenant=TENANT2, user=USER1) + fake_notifier = unit_test_utils.FakeNotifier() + image_proxy = glance.notifier.ImageProxy(image, context, fake_notifier) + def data_iterator(): - self.notifier.log = [] + fake_notifier.log = [] yield 'abcde' yield 'fghij' + image_proxy.extra_properties[ + 'os_glance_importing_to_stores'] = 'store2' - self.image_proxy.set_data(data_iterator(), 10) + image_proxy.extra_properties[ + 'os_glance_importing_to_stores'] = 'store1,store2' + image_proxy.extra_properties['os_glance_failed_import'] = '' + image_proxy.set_data(data_iterator(), 10) - output_logs = self.notifier.get_logs() + output_logs = fake_notifier.get_logs() self.assertEqual(2, len(output_logs)) output_log = output_logs[0] self.assertEqual('INFO', output_log['notification_type']) self.assertEqual('image.upload', output_log['event_type']) self.assertEqual(self.image.image_id, output_log['payload']['id']) + self.assertEqual(['store2'], output_log['payload'][ + 'os_glance_importing_to_stores']) + self.assertEqual([], + output_log['payload']['os_glance_failed_import']) output_log = output_logs[1] self.assertEqual('INFO', output_log['notification_type']) self.assertEqual('image.activate', output_log['event_type']) self.assertEqual(self.image.image_id, output_log['payload']['id']) - def test_image_set_data_upload_and_activate_notification_disabled(self): + def test_image_set_data_upload_and_not_activate_notification(self): insurance = {'called': False} def data_iterator(): self.notifier.log = [] yield 'abcde' yield 'fghij' + self.image_proxy.extra_properties[ + 'os_glance_importing_to_stores'] = 'store2' + insurance['called'] = True + + self.image_proxy.set_data(data_iterator(), 10) + + output_logs = self.notifier.get_logs() + self.assertEqual(1, len(output_logs)) + + output_log = output_logs[0] + self.assertEqual('INFO', output_log['notification_type']) + self.assertEqual('image.upload', output_log['event_type']) + self.assertEqual(self.image.image_id, output_log['payload']['id']) + self.assertTrue(insurance['called']) + + def test_image_set_data_upload_and_activate_notification_disabled(self): + insurance = {'called': False} + image = ImageStub(image_id=UUID1, name='image-1', status='queued', + created_at=DATETIME, updated_at=DATETIME, + owner=TENANT1, visibility='public') + context = glance.context.RequestContext(tenant=TENANT2, user=USER1) + fake_notifier = unit_test_utils.FakeNotifier() + image_proxy = glance.notifier.ImageProxy(image, context, fake_notifier) + + def data_iterator(): + fake_notifier.log = [] + yield 'abcde' + yield 'fghij' insurance['called'] = True self.config(disabled_notifications=['image.activate', 'image.upload']) - self.image_proxy.set_data(data_iterator(), 10) + image_proxy.set_data(data_iterator(), 10) self.assertTrue(insurance['called']) - output_logs = self.notifier.get_logs() + output_logs = fake_notifier.get_logs() self.assertEqual(0, len(output_logs)) def test_image_set_data_storage_full(self): diff --git a/glance/tests/unit/test_quota.py b/glance/tests/unit/test_quota.py index 51d0295e10..3930272d0a 100644 --- a/glance/tests/unit/test_quota.py +++ b/glance/tests/unit/test_quota.py @@ -43,7 +43,7 @@ class FakeImage(object): locations = [{'url': 'file:///not/a/path', 'metadata': {}}] tags = set([]) - def set_data(self, data, size=None, backend=None): + def set_data(self, data, size=None, backend=None, set_active=True): self.size = 0 for d in data: self.size += len(d) diff --git a/glance/tests/unit/test_store_image.py b/glance/tests/unit/test_store_image.py index 55d734461c..b351d63504 100644 --- a/glance/tests/unit/test_store_image.py +++ b/glance/tests/unit/test_store_image.py @@ -12,6 +12,7 @@ # 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 cryptography import exceptions as crypto_exception from cursive import exception as cursive_exception from cursive import signature_utils import glance_store @@ -48,8 +49,11 @@ class ImageStub(object): self.status = status self.locations = locations or [] self.visibility = visibility - self.size = 1 + self.size = None self.extra_properties = extra_properties or {} + self.os_hash_algo = None + self.os_hash_value = None + self.checksum = None def delete(self): self.status = 'deleted' @@ -84,6 +88,104 @@ class FakeMemberRepo(object): self.tenants.remove(member.member_id) +class TestStoreMultiBackends(utils.BaseTestCase): + def setUp(self): + self.store_api = unit_test_utils.FakeStoreAPI() + self.store_utils = unit_test_utils.FakeStoreUtils(self.store_api) + self.enabled_backends = { + "ceph1": "rbd", + "ceph2": "rbd" + } + super(TestStoreMultiBackends, self).setUp() + self.config(enabled_backends=self.enabled_backends) + + @mock.patch("glance.location.signature_utils.get_verifier") + def test_set_data_calls_upload_to_store(self, msig): + context = glance.context.RequestContext(user=USER1) + extra_properties = { + 'img_signature_certificate_uuid': 'UUID', + 'img_signature_hash_method': 'METHOD', + 'img_signature_key_type': 'TYPE', + 'img_signature': 'VALID' + } + image_stub = ImageStub(UUID2, status='queued', locations=[], + extra_properties=extra_properties) + image = glance.location.ImageProxy(image_stub, context, + self.store_api, self.store_utils) + with mock.patch.object(image, "_upload_to_store") as mloc: + image.set_data('YYYY', 4, backend='ceph1') + msig.assert_called_once_with(context=context, + img_signature_certificate_uuid='UUID', + img_signature_hash_method='METHOD', + img_signature='VALID', + img_signature_key_type='TYPE') + mloc.assert_called_once_with('YYYY', msig.return_value, 'ceph1', 4) + + self.assertEqual('active', image.status) + + def test_image_set_data(self): + store_api = mock.MagicMock() + store_api.add_with_multihash.return_value = ( + "rbd://ceph1", 4, "Z", "MH", {"backend": "ceph1"}) + context = glance.context.RequestContext(user=USER1) + image_stub = ImageStub(UUID2, status='queued', locations=[]) + image = glance.location.ImageProxy(image_stub, context, + store_api, self.store_utils) + image.set_data('YYYY', 4, backend='ceph1') + self.assertEqual(4, image.size) + + # NOTE(markwash): FakeStore returns image_id for location + self.assertEqual("rbd://ceph1", image.locations[0]['url']) + self.assertEqual({"backend": "ceph1"}, image.locations[0]['metadata']) + self.assertEqual('Z', image.checksum) + self.assertEqual('active', image.status) + + @mock.patch('glance.location.LOG') + def test_image_set_data_valid_signature(self, mock_log): + store_api = mock.MagicMock() + store_api.add_with_multihash.return_value = ( + "rbd://ceph1", 4, "Z", "MH", {"backend": "ceph1"}) + context = glance.context.RequestContext(user=USER1) + extra_properties = { + 'img_signature_certificate_uuid': 'UUID', + 'img_signature_hash_method': 'METHOD', + 'img_signature_key_type': 'TYPE', + 'img_signature': 'VALID' + } + image_stub = ImageStub(UUID2, status='queued', + extra_properties=extra_properties) + self.mock_object(signature_utils, 'get_verifier', + unit_test_utils.fake_get_verifier) + image = glance.location.ImageProxy(image_stub, context, + store_api, self.store_utils) + image.set_data('YYYY', 4, backend='ceph1') + self.assertEqual('active', image.status) + call = mock.call(u'Successfully verified signature for image %s', + UUID2) + mock_log.info.assert_has_calls([call]) + + @mock.patch("glance.location.signature_utils.get_verifier") + def test_image_set_data_invalid_signature(self, msig): + msig.return_value.verify.side_effect = \ + crypto_exception.InvalidSignature + store_api = mock.MagicMock() + store_api.add_with_multihash.return_value = ( + "rbd://ceph1", 4, "Z", "MH", {"backend": "ceph1"}) + context = glance.context.RequestContext(user=USER1) + extra_properties = { + 'img_signature_certificate_uuid': 'UUID', + 'img_signature_hash_method': 'METHOD', + 'img_signature_key_type': 'TYPE', + 'img_signature': 'INVALID' + } + image_stub = ImageStub(UUID2, status='queued', + extra_properties=extra_properties) + image = glance.location.ImageProxy(image_stub, context, + store_api, self.store_utils) + self.assertRaises(cursive_exception.SignatureVerificationError, + image.set_data, 'YYYY', 4, backend='ceph1') + + class TestStoreImage(utils.BaseTestCase): def setUp(self): locations = [{'url': '%s/%s' % (BASE_URI, UUID1), @@ -124,7 +226,11 @@ class TestStoreImage(utils.BaseTestCase): context = glance.context.RequestContext(user=USER1) (image2, image_stub2) = self._add_image(context, UUID2, 'ZZZ', 3) location_data = image2.locations[0] - image1.locations.append(location_data) + + with mock.patch("glance.location.store") as mock_store: + mock_store.get_size_from_uri_and_backend.return_value = 3 + image1.locations.append(location_data) + self.assertEqual(2, len(image1.locations)) self.assertEqual(UUID2, location_data['url']) @@ -603,8 +709,9 @@ class TestStoreImage(utils.BaseTestCase): location2 = {'url': UUID2, 'metadata': {}} location3 = {'url': UUID3, 'metadata': {}} - - image3.locations += [location2, location3] + with mock.patch("glance.location.store") as mock_store: + mock_store.get_size_from_uri_and_backend.return_value = 4 + image3.locations += [location2, location3] self.assertEqual([location2, location3], image_stub3.locations) self.assertEqual([location2, location3], image3.locations) @@ -633,7 +740,9 @@ class TestStoreImage(utils.BaseTestCase): location2 = {'url': UUID2, 'metadata': {}} location3 = {'url': UUID3, 'metadata': {}} - image3.locations += [location2, location3] + with mock.patch("glance.location.store") as mock_store: + mock_store.get_size_from_uri_and_backend.return_value = 4 + image3.locations += [location2, location3] self.assertEqual(1, image_stub3.locations.index(location3)) @@ -660,7 +769,9 @@ class TestStoreImage(utils.BaseTestCase): location2 = {'url': UUID2, 'metadata': {}} location3 = {'url': UUID3, 'metadata': {}} - image3.locations += [location2, location3] + with mock.patch("glance.location.store") as mock_store: + mock_store.get_size_from_uri_and_backend.return_value = 4 + image3.locations += [location2, location3] self.assertEqual(1, image_stub3.locations.index(location3)) self.assertEqual(location2, image_stub3.locations[0]) @@ -690,7 +801,9 @@ class TestStoreImage(utils.BaseTestCase): location3 = {'url': UUID3, 'metadata': {}} location_bad = {'url': 'unknown://location', 'metadata': {}} - image3.locations += [location2, location3] + with mock.patch("glance.location.store") as mock_store: + mock_store.get_size_from_uri_and_backend.return_value = 4 + image3.locations += [location2, location3] self.assertIn(location3, image_stub3.locations) self.assertNotIn(location_bad, image_stub3.locations) @@ -718,7 +831,9 @@ class TestStoreImage(utils.BaseTestCase): image_stub3 = ImageStub('fake_image_id', status='queued', locations=[]) image3 = glance.location.ImageProxy(image_stub3, context, self.store_api, self.store_utils) - image3.locations += [location2, location3] + with mock.patch("glance.location.store") as mock_store: + mock_store.get_size_from_uri_and_backend.return_value = 4 + image3.locations += [location2, location3] image_stub3.locations.reverse() diff --git a/glance/tests/unit/v2/test_image_data_resource.py b/glance/tests/unit/v2/test_image_data_resource.py index 8ce7f9f413..44ab4aa129 100644 --- a/glance/tests/unit/v2/test_image_data_resource.py +++ b/glance/tests/unit/v2/test_image_data_resource.py @@ -71,7 +71,7 @@ class FakeImage(object): return self.data[offset:offset + chunk_size] return self.data[offset:] - def set_data(self, data, size=None, backend=None): + def set_data(self, data, size=None, backend=None, set_active=True): self.data = ''.join(data) self.size = size self.status = 'modified-by-fake' diff --git a/releasenotes/notes/import-multi-stores-3e781f2878b3134d.yaml b/releasenotes/notes/import-multi-stores-3e781f2878b3134d.yaml new file mode 100644 index 0000000000..e8ff3702a7 --- /dev/null +++ b/releasenotes/notes/import-multi-stores-3e781f2878b3134d.yaml @@ -0,0 +1,31 @@ +--- +features: + - | + Add ability to import image into multiple stores during `interoperable + image import process`_. + +upgrade: + - | + Add ability to import image into multiple stores during `interoperable + image import process`_. + This feature will only work if multiple stores are enabled in the + deployment. + It introduces 3 new optional body fields to the `import API path`: + + - ``stores``: List containing the stores id to import the image binary data + to. + + - ``all_stores``: To import the data in all configured stores. + + - ``all_stores_must_succeed``: Control wether the import have to succeed in + all stores. + + Users can follow workflow execution with 2 new reserved properties: + + - ``os_glance_importing_to_stores``: list of stores that has not yet been + processed. + + - ``os_glance_failed_import``: Each time an import in a store fails, it is + added to this list. + + .. _`interoperable image import process`: https://developer.openstack.org/api-ref/image/v2/#interoperable-image-import