
This commit adds the command "subcloud deploy abort" to dcmanager. It allows the user to abort the on-going execution of a playbook against the subcloud. Any task will be aborted immediately if the unabortable flag is not set by the playbook [1]. If current operation is install, a shutdown command will also be issued. Test Plan: Success cases: - PASS: Trigger an abort during installation and verify that the playbook execution was aborted immediately, the subcloud was shut dows after it and the RVMC pod and job were terminated. - PASS: Trigger an abort during config and verify that the playbook execution was aborted immediately. - PASS: Trigger an abort during bootstrap without the presence of unabortable flag and verify that the playbook execution was aborted immidiately. - PASS: Trigger an abort during bootstrap with the presence of unabortable flag and verify that the playbook execution was aborted only after the flag was deleted. - PASS: Trigger an abort directly calling the API (using CURL instead of using the CLI. Failure cases: - PASS: Verify that the abort request is rejected if deploy state is not 'installing', 'bootstrapping' or 'configuring'. - PASS: Abort when an unabortable task is running and then force an external error during this task restarting dcmanager-manager service, verify that deploy state is set to a failed state e.g. bootstrap-failed. - PASS: Abort when an unabortable task is running and then force an internal error during this task using ansible.builtin.fail module, verify that deploy state is set to a failed state e.g. bootstrap-failed. - PASS: Abort when an unabortable task is running and then force the playbook to halt the execution during this task using ansible.builtin.pause module, verify that deploy state is set to a failed state after 10 minutes e.g. bootstrap-failed. Story: 2010756 Task: 48102 Co-Authored-By: Gustavo Herzmann <gustavo.herzmann@windriver.com> Signed-off-by: Victor Romano <victor.gluzromano@windriver.com> Change-Id: Ic5311324a76bf7ce1215692e934d5577ff82868e
126 lines
3.6 KiB
Python
126 lines
3.6 KiB
Python
# Copyright 2015 Huawei Technologies Co., Ltd.
|
|
# Copyright 2015 Ericsson AB.
|
|
# All Rights Reserved.
|
|
#
|
|
# 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.
|
|
#
|
|
# Copyright (c) 2020 Wind River Systems, Inc.
|
|
#
|
|
|
|
"""
|
|
DC Orchestrator base exception handling.
|
|
"""
|
|
import six
|
|
|
|
from oslo_utils import encodeutils
|
|
from oslo_utils import excutils
|
|
|
|
from dcorch.common.i18n import _
|
|
|
|
|
|
class DCCommonException(Exception):
|
|
"""Base Commond Driver Exception.
|
|
|
|
To correctly use this class, inherit from it and define
|
|
a 'message' property. That message will get printf'd
|
|
with the keyword arguments provided to the constructor.
|
|
"""
|
|
|
|
message = _("An unknown exception occurred.")
|
|
|
|
def __init__(self, **kwargs):
|
|
try:
|
|
super(DCCommonException, self).__init__(self.message % kwargs) # pylint: disable=W1645
|
|
self.msg = self.message % kwargs # pylint: disable=W1645
|
|
except Exception:
|
|
with excutils.save_and_reraise_exception() as ctxt:
|
|
if not self.use_fatal_exceptions():
|
|
ctxt.reraise = False
|
|
# at least get the core message out if something happened
|
|
super(DCCommonException, self).__init__(self.message) # pylint: disable=W1645
|
|
|
|
if six.PY2:
|
|
def __unicode__(self):
|
|
return encodeutils.exception_to_unicode(self.msg)
|
|
|
|
def use_fatal_exceptions(self):
|
|
return False
|
|
|
|
|
|
class NotFound(DCCommonException):
|
|
pass
|
|
|
|
|
|
class Forbidden(DCCommonException):
|
|
message = _("Requested API is forbidden")
|
|
|
|
|
|
class Conflict(DCCommonException):
|
|
pass
|
|
|
|
|
|
class ServiceUnavailable(DCCommonException):
|
|
message = _("The service is unavailable")
|
|
|
|
|
|
class InvalidInputError(DCCommonException):
|
|
message = _("An invalid value was provided")
|
|
|
|
|
|
class InternalError(DCCommonException):
|
|
message = _("Error when performing operation")
|
|
|
|
|
|
class ProjectNotFound(NotFound):
|
|
message = _("Project %(project_id)s doesn't exist")
|
|
|
|
|
|
class OAMAddressesNotFound(NotFound):
|
|
message = _("OAM Addresses Not Found")
|
|
|
|
|
|
class CertificateNotFound(NotFound):
|
|
message = _("Certificate in region=%(region_name)s with signature "
|
|
"%(signature)s not found")
|
|
|
|
|
|
class LoadNotFound(NotFound):
|
|
message = _("Load in region=%(region_name)s with id %(load_id)s not found")
|
|
|
|
|
|
class LoadNotInVault(NotFound):
|
|
message = _("Load at path %(path)s not found")
|
|
|
|
|
|
class LoadMaxReached(Conflict):
|
|
message = _("Load in region=%(region_name)s at maximum number of loads")
|
|
|
|
|
|
class PlaybookExecutionFailed(DCCommonException):
|
|
message = _("Playbook execution failed, command=%(playbook_cmd)s")
|
|
|
|
|
|
class PlaybookExecutionTimeout(PlaybookExecutionFailed):
|
|
message = _("Playbook execution failed [TIMEOUT (%(timeout)s)], "
|
|
"command=%(playbook_cmd)s")
|
|
|
|
|
|
class ImageNotInLocalRegistry(NotFound):
|
|
message = _("Image %(image_name)s:%(image_tag)s not found in the local registry. "
|
|
"Please check with command: system registry-image-list or "
|
|
"system registry-image-tags %(image_name)s")
|
|
|
|
|
|
class SubcloudShutdownError(PlaybookExecutionFailed):
|
|
message = _("Subcloud %(subcloud_name)s could not be shut down.")
|