Source code for cloudify_rest_client.deployment_updates

import os
import json
import shutil
import tempfile
from urllib.parse import quote as urlquote, urlparse
from urllib.request import pathname2url

from mimetypes import MimeTypes

from cloudify_rest_client import utils
from cloudify_rest_client.exceptions import CloudifyClientError
from cloudify_rest_client.responses import ListResponse


[docs] class DeploymentUpdate(dict): def __init__(self, update): self.update(update) @property def id(self): return self['id'] @property def state(self): return self['state'] @property def deployment_id(self): return self['deployment_id'] @property def old_blueprint_id(self): return self['old_blueprint_id'] @property def new_blueprint_id(self): return self['new_blueprint_id'] @property def old_inputs(self): return self['old_inputs'] @property def new_inputs(self): return self['new_inputs'] @property def steps(self): return self['steps'] @property def execution_id(self): return self['execution_id'] @property def created_at(self): return self['created_at'] @property def runtime_only_evaluation(self): return self['runtime_only_evaluation'] @property def deployment_plan(self): return self['deployment_plan']
[docs] class DeploymentUpdatesClient(object): def __init__(self, api): self.api = api
[docs] def create(self, update_id, deployment_id, **kwargs): """Create a deployment-update object. This is only useful from within the deployment-update workflow. Do not use this otherwise. """ url = '/deployment-updates/{0}'.format(update_id) data = { 'deployment_id': deployment_id, } data.update(kwargs) response = self.api.put(url, data=data) return DeploymentUpdate(response)
[docs] def set_attributes(self, update_id, **kwargs): """Update a deployment-update object with the given attributes. This is only useful from within the deployment-update workflow. Do not use this otherwise. """ url = '/deployment-updates/{0}'.format(update_id) self.api.patch(url, data=kwargs)
[docs] def list(self, _include=None, sort=None, is_descending=False, **kwargs): """List deployment updates :param _include: List of fields to include in response. :param sort: Key for sorting the list. :param is_descending: True for descending order, False for ascending. :param kwargs: Optional filter fields. for a list of available fields see the REST service's models.DeploymentUpdate.fields """ uri = '/deployment-updates' params = kwargs if sort: params['_sort'] = '-' + sort if is_descending else sort response = self.api.get(uri, params=params, _include=_include) items = [DeploymentUpdate(item) for item in response['items']] return ListResponse(items, response['metadata'])
[docs] def bulk_insert(self, updates): """Bulk insert deployment updates. For internal use only.""" uri = '/deployment-updates' self.api.post(uri, {'deployment_updates': updates}, expected_status_code=[201, 204])
def _update_from_blueprint(self, deployment_id, blueprint_path, inputs=None): """Create a deployment update transaction for blueprint app. :param deployment_id: The deployment id :param blueprint_path: the path of the blueprint to stage """ assert deployment_id tempdir = tempfile.mkdtemp() try: tar_path = utils.tar_blueprint(blueprint_path, tempdir) application_filename = os.path.basename(blueprint_path) return self._update_from_archive(deployment_id, tar_path, application_filename, inputs=inputs) finally: shutil.rmtree(tempdir) @staticmethod def _update_from_archive(deployment_id, archive_path, application_file_name=None, inputs=None): """Create a deployment update transaction for an archived app. :param archive_path: the path for the archived app. :param application_file_name: the main blueprint filename. :param deployment_id: the deployment id to update. :return: DeploymentUpdate dict :rtype: DeploymentUpdate """ assert deployment_id mime_types = MimeTypes() data_form = {} params = {} # all the inputs are passed through the query if inputs: data_form['inputs'] = ('inputs', json.dumps(inputs), 'text/plain') if application_file_name: params['application_file_name'] = urlquote(application_file_name) # For a Windows path (e.g. "C:\aaa\bbb.zip") scheme is the # drive letter and therefore the 2nd condition is present if all([urlparse(archive_path).scheme, not os.path.exists(archive_path)]): # archive location is URL params['blueprint_archive_url'] = archive_path else: data_form['blueprint_archive'] = ( os.path.basename(archive_path), open(archive_path, 'rb'), # Guess the archive mime type mime_types.guess_type(pathname2url(archive_path))) return data_form, params
[docs] def get(self, update_id, _include=None): """Get deployment update :param update_id: The update id """ uri = '/deployment-updates/{0}'.format(update_id) response = self.api.get(uri, _include=_include) return DeploymentUpdate(response)
[docs] def update_with_existing_blueprint( self, deployment_id, blueprint_id=None, inputs=None, skip_install=False, skip_uninstall=False, skip_reinstall=False, skip_drift_check=False, skip_heal=False, force_reinstall=False, workflow_id=None, force=False, ignore_failure=False, install_first=False, reinstall_list=None, preview=False, update_plugins=True, runtime_only_evaluation=None, auto_correct_types=None, reevaluate_active_statuses=None ): data = { 'workflow_id': workflow_id, 'skip_install': skip_install, 'skip_uninstall': skip_uninstall, 'skip_reinstall': skip_reinstall, 'skip_drift_check': skip_drift_check, 'skip_heal': skip_heal, 'force_reinstall': force_reinstall, 'ignore_failure': ignore_failure, 'install_first': install_first, 'preview': preview, 'blueprint_id': blueprint_id, 'update_plugins': update_plugins, 'force': force, } if inputs: data['inputs'] = inputs if reinstall_list: data['reinstall_list'] = reinstall_list if runtime_only_evaluation is not None: data['runtime_only_evaluation'] = runtime_only_evaluation if auto_correct_types is not None: data['auto_correct_types'] = auto_correct_types if reevaluate_active_statuses is not None: data['reevaluate_active_statuses'] = reevaluate_active_statuses uri = '/deployment-updates/{0}/update/initiate'.format(deployment_id) response = self.api.post(uri, data=data) return DeploymentUpdate(response)
[docs] def finalize_commit(self, update_id): """Finalize the committing process :param update_id: :return: """ assert update_id uri = '/deployment-updates/{0}/update/finalize'.format(update_id) response = self.api.post(uri) return DeploymentUpdate(response)
[docs] def dump(self, deployment_update_ids=None): """Generate deployment updates' attributes for a snapshot. :param deployment_update_ids: A list of deployment updates' identifiers, if not empty, used to select specific deployment updates to be dumped. :returns: A generator of dictionaries, which describe deployment updates' attributes. """ entities = utils.get_all( self.api.get, '/deployment-updates', params={'_get_data': True}, _include=['id', 'deployment_id', 'new_blueprint_id', 'state', 'new_inputs', 'created_at', 'created_by', 'execution_id', 'old_blueprint_id', 'runtime_only_evaluation', 'deployment_plan', 'deployment_update_node_instances', 'visibility', 'steps', 'central_plugins_to_uninstall', 'central_plugins_to_install', 'old_inputs', 'deployment_update_nodes', 'modified_entity_ids'] ) if not deployment_update_ids: return entities return (e for e in entities if e['id'] in deployment_update_ids)
[docs] def restore(self, entities, logger): """Restore deployment updates from a snapshot. :param entities: An iterable (e.g. a list) of dictionaries describing deployment updates to be restored. :param logger: A logger instance. """ for entity in entities: entity['update_id'] = entity.pop('id') entity['blueprint_id'] = entity.pop('new_blueprint_id') entity['inputs'] = entity.pop('new_inputs', None) try: self.create(**entity) except CloudifyClientError as exc: logger.error("Error restoring deployment update " f"{entity['update_id']}: {exc}")