Deploy templates: conductor and API nits
Fixes nits from https://review.openstack.org/#/c/634732 and https://review.openstack.org/#/c/631845. Change-Id: I3da177cffca19c024850be675137cbeb4de31758 Story: 1722275 Task: 28675 Task: 28677
This commit is contained in:
parent
5614d251bb
commit
e4cce6759a
@ -325,12 +325,8 @@ class ChassisController(rest.RestController):
|
|||||||
policy.authorize('baremetal:chassis:update', cdict, cdict)
|
policy.authorize('baremetal:chassis:update', cdict, cdict)
|
||||||
|
|
||||||
rpc_chassis = objects.Chassis.get_by_uuid(context, chassis_uuid)
|
rpc_chassis = objects.Chassis.get_by_uuid(context, chassis_uuid)
|
||||||
try:
|
chassis = Chassis(
|
||||||
chassis = Chassis(
|
**api_utils.apply_jsonpatch(rpc_chassis.as_dict(), patch))
|
||||||
**api_utils.apply_jsonpatch(rpc_chassis.as_dict(), patch))
|
|
||||||
|
|
||||||
except api_utils.JSONPATCH_EXCEPTIONS as e:
|
|
||||||
raise exception.PatchError(patch=patch, reason=e)
|
|
||||||
|
|
||||||
# Update only the fields that have changed
|
# Update only the fields that have changed
|
||||||
for field in objects.Chassis.fields:
|
for field in objects.Chassis.fields:
|
||||||
|
@ -116,7 +116,8 @@ class DeployTemplate(base.APIBase):
|
|||||||
|
|
||||||
# The name must also be a valid trait.
|
# The name must also be a valid trait.
|
||||||
api_utils.validate_trait(
|
api_utils.validate_trait(
|
||||||
value.name, _("Deploy template name must be a valid trait"))
|
value.name,
|
||||||
|
error_prefix=_("Deploy template name must be a valid trait"))
|
||||||
|
|
||||||
# There must be at least one step.
|
# There must be at least one step.
|
||||||
if not value.steps:
|
if not value.steps:
|
||||||
@ -393,12 +394,9 @@ class DeployTemplatesController(rest.RestController):
|
|||||||
rpc_template = api_utils.get_rpc_deploy_template_with_suffix(
|
rpc_template = api_utils.get_rpc_deploy_template_with_suffix(
|
||||||
template_ident)
|
template_ident)
|
||||||
|
|
||||||
try:
|
template_dict = rpc_template.as_dict()
|
||||||
template_dict = rpc_template.as_dict()
|
template = DeployTemplate(
|
||||||
template = DeployTemplate(
|
**api_utils.apply_jsonpatch(template_dict, patch))
|
||||||
**api_utils.apply_jsonpatch(template_dict, patch))
|
|
||||||
except api_utils.JSONPATCH_EXCEPTIONS as e:
|
|
||||||
raise exception.PatchError(patch=patch, reason=e)
|
|
||||||
template.validate(template)
|
template.validate(template)
|
||||||
self._update_changed_fields(template, rpc_template)
|
self._update_changed_fields(template, rpc_template)
|
||||||
|
|
||||||
|
@ -2149,16 +2149,14 @@ class NodesController(rest.RestController):
|
|||||||
% node_ident)
|
% node_ident)
|
||||||
error_msg += "'%(name)s'"
|
error_msg += "'%(name)s'"
|
||||||
self._check_names_acceptable(names, error_msg)
|
self._check_names_acceptable(names, error_msg)
|
||||||
try:
|
|
||||||
node_dict = rpc_node.as_dict()
|
node_dict = rpc_node.as_dict()
|
||||||
# NOTE(lucasagomes):
|
# NOTE(lucasagomes):
|
||||||
# 1) Remove chassis_id because it's an internal value and
|
# 1) Remove chassis_id because it's an internal value and
|
||||||
# not present in the API object
|
# not present in the API object
|
||||||
# 2) Add chassis_uuid
|
# 2) Add chassis_uuid
|
||||||
node_dict['chassis_uuid'] = node_dict.pop('chassis_id', None)
|
node_dict['chassis_uuid'] = node_dict.pop('chassis_id', None)
|
||||||
node = Node(**api_utils.apply_jsonpatch(node_dict, patch))
|
node = Node(**api_utils.apply_jsonpatch(node_dict, patch))
|
||||||
except api_utils.JSONPATCH_EXCEPTIONS as e:
|
|
||||||
raise exception.PatchError(patch=patch, reason=e)
|
|
||||||
self._update_changed_fields(node, rpc_node)
|
self._update_changed_fields(node, rpc_node)
|
||||||
# NOTE(deva): we calculate the rpc topic here in case node.driver
|
# NOTE(deva): we calculate the rpc topic here in case node.driver
|
||||||
# has changed, so that update is sent to the
|
# has changed, so that update is sent to the
|
||||||
|
@ -677,21 +677,18 @@ class PortsController(rest.RestController):
|
|||||||
self._check_allowed_port_fields(fields_to_check)
|
self._check_allowed_port_fields(fields_to_check)
|
||||||
|
|
||||||
rpc_port = objects.Port.get_by_uuid(context, port_uuid)
|
rpc_port = objects.Port.get_by_uuid(context, port_uuid)
|
||||||
try:
|
port_dict = rpc_port.as_dict()
|
||||||
port_dict = rpc_port.as_dict()
|
# NOTE(lucasagomes):
|
||||||
# NOTE(lucasagomes):
|
# 1) Remove node_id because it's an internal value and
|
||||||
# 1) Remove node_id because it's an internal value and
|
# not present in the API object
|
||||||
# not present in the API object
|
# 2) Add node_uuid
|
||||||
# 2) Add node_uuid
|
port_dict['node_uuid'] = port_dict.pop('node_id', None)
|
||||||
port_dict['node_uuid'] = port_dict.pop('node_id', None)
|
# NOTE(vsaienko):
|
||||||
# NOTE(vsaienko):
|
# 1) Remove portgroup_id because it's an internal value and
|
||||||
# 1) Remove portgroup_id because it's an internal value and
|
# not present in the API object
|
||||||
# not present in the API object
|
# 2) Add portgroup_uuid
|
||||||
# 2) Add portgroup_uuid
|
port_dict['portgroup_uuid'] = port_dict.pop('portgroup_id', None)
|
||||||
port_dict['portgroup_uuid'] = port_dict.pop('portgroup_id', None)
|
port = Port(**api_utils.apply_jsonpatch(port_dict, patch))
|
||||||
port = Port(**api_utils.apply_jsonpatch(port_dict, patch))
|
|
||||||
except api_utils.JSONPATCH_EXCEPTIONS as e:
|
|
||||||
raise exception.PatchError(patch=patch, reason=e)
|
|
||||||
|
|
||||||
api_utils.handle_patch_port_like_extra_vif(rpc_port, port, patch)
|
api_utils.handle_patch_port_like_extra_vif(rpc_port, port, patch)
|
||||||
|
|
||||||
|
@ -545,17 +545,14 @@ class PortgroupsController(pecan.rest.RestController):
|
|||||||
raise wsme.exc.ClientSideError(
|
raise wsme.exc.ClientSideError(
|
||||||
error_msg, status_code=http_client.BAD_REQUEST)
|
error_msg, status_code=http_client.BAD_REQUEST)
|
||||||
|
|
||||||
try:
|
portgroup_dict = rpc_portgroup.as_dict()
|
||||||
portgroup_dict = rpc_portgroup.as_dict()
|
# NOTE:
|
||||||
# NOTE:
|
# 1) Remove node_id because it's an internal value and
|
||||||
# 1) Remove node_id because it's an internal value and
|
# not present in the API object
|
||||||
# not present in the API object
|
# 2) Add node_uuid
|
||||||
# 2) Add node_uuid
|
portgroup_dict['node_uuid'] = portgroup_dict.pop('node_id', None)
|
||||||
portgroup_dict['node_uuid'] = portgroup_dict.pop('node_id', None)
|
portgroup = Portgroup(**api_utils.apply_jsonpatch(portgroup_dict,
|
||||||
portgroup = Portgroup(**api_utils.apply_jsonpatch(portgroup_dict,
|
patch))
|
||||||
patch))
|
|
||||||
except api_utils.JSONPATCH_EXCEPTIONS as e:
|
|
||||||
raise exception.PatchError(patch=patch, reason=e)
|
|
||||||
|
|
||||||
api_utils.handle_patch_port_like_extra_vif(rpc_portgroup, portgroup,
|
api_utils.handle_patch_port_like_extra_vif(rpc_portgroup, portgroup,
|
||||||
patch)
|
patch)
|
||||||
|
@ -42,10 +42,10 @@ from ironic import objects
|
|||||||
CONF = cfg.CONF
|
CONF = cfg.CONF
|
||||||
|
|
||||||
|
|
||||||
JSONPATCH_EXCEPTIONS = (jsonpatch.JsonPatchException,
|
_JSONPATCH_EXCEPTIONS = (jsonpatch.JsonPatchException,
|
||||||
jsonpatch.JsonPointerException,
|
jsonpatch.JsonPointerException,
|
||||||
KeyError,
|
KeyError,
|
||||||
IndexError)
|
IndexError)
|
||||||
|
|
||||||
|
|
||||||
# Minimum API version to use for certain verbs
|
# Minimum API version to use for certain verbs
|
||||||
@ -96,7 +96,7 @@ def validate_sort_dir(sort_dir):
|
|||||||
return sort_dir
|
return sort_dir
|
||||||
|
|
||||||
|
|
||||||
def validate_trait(trait, error_prefix='Invalid trait'):
|
def validate_trait(trait, error_prefix=_('Invalid trait')):
|
||||||
error = wsme.exc.ClientSideError(
|
error = wsme.exc.ClientSideError(
|
||||||
_('%(error_prefix)s. A valid trait must be no longer than 255 '
|
_('%(error_prefix)s. A valid trait must be no longer than 255 '
|
||||||
'characters. Standard traits are defined in the os_traits library. '
|
'characters. Standard traits are defined in the os_traits library. '
|
||||||
@ -117,13 +117,32 @@ def validate_trait(trait, error_prefix='Invalid trait'):
|
|||||||
|
|
||||||
|
|
||||||
def apply_jsonpatch(doc, patch):
|
def apply_jsonpatch(doc, patch):
|
||||||
|
"""Apply a JSON patch, one operation at a time.
|
||||||
|
|
||||||
|
If the patch fails to apply, this allows us to determine which operation
|
||||||
|
failed, making the error message a little less cryptic.
|
||||||
|
|
||||||
|
:param doc: The JSON document to patch.
|
||||||
|
:param patch: The JSON patch to apply.
|
||||||
|
:returns: The result of the patch operation.
|
||||||
|
:raises: PatchError if the patch fails to apply.
|
||||||
|
:raises: wsme.exc.ClientSideError if the patch adds a new root attribute.
|
||||||
|
"""
|
||||||
|
# Prevent removal of root attributes.
|
||||||
for p in patch:
|
for p in patch:
|
||||||
if p['op'] == 'add' and p['path'].count('/') == 1:
|
if p['op'] == 'add' and p['path'].count('/') == 1:
|
||||||
if p['path'].lstrip('/') not in doc:
|
if p['path'].lstrip('/') not in doc:
|
||||||
msg = _('Adding a new attribute (%s) to the root of '
|
msg = _('Adding a new attribute (%s) to the root of '
|
||||||
'the resource is not allowed')
|
'the resource is not allowed')
|
||||||
raise wsme.exc.ClientSideError(msg % p['path'])
|
raise wsme.exc.ClientSideError(msg % p['path'])
|
||||||
return jsonpatch.apply_patch(doc, jsonpatch.JsonPatch(patch))
|
|
||||||
|
# Apply operations one at a time, to improve error reporting.
|
||||||
|
for patch_op in patch:
|
||||||
|
try:
|
||||||
|
doc = jsonpatch.apply_patch(doc, jsonpatch.JsonPatch([patch_op]))
|
||||||
|
except _JSONPATCH_EXCEPTIONS as e:
|
||||||
|
raise exception.PatchError(patch=patch_op, reason=e)
|
||||||
|
return doc
|
||||||
|
|
||||||
|
|
||||||
def get_patch_values(patch, path):
|
def get_patch_values(patch, path):
|
||||||
|
@ -421,17 +421,14 @@ class VolumeConnectorsController(rest.RestController):
|
|||||||
|
|
||||||
rpc_connector = objects.VolumeConnector.get_by_uuid(context,
|
rpc_connector = objects.VolumeConnector.get_by_uuid(context,
|
||||||
connector_uuid)
|
connector_uuid)
|
||||||
try:
|
connector_dict = rpc_connector.as_dict()
|
||||||
connector_dict = rpc_connector.as_dict()
|
# NOTE(smoriya):
|
||||||
# NOTE(smoriya):
|
# 1) Remove node_id because it's an internal value and
|
||||||
# 1) Remove node_id because it's an internal value and
|
# not present in the API object
|
||||||
# not present in the API object
|
# 2) Add node_uuid
|
||||||
# 2) Add node_uuid
|
connector_dict['node_uuid'] = connector_dict.pop('node_id', None)
|
||||||
connector_dict['node_uuid'] = connector_dict.pop('node_id', None)
|
connector = VolumeConnector(
|
||||||
connector = VolumeConnector(
|
**api_utils.apply_jsonpatch(connector_dict, patch))
|
||||||
**api_utils.apply_jsonpatch(connector_dict, patch))
|
|
||||||
except api_utils.JSONPATCH_EXCEPTIONS as e:
|
|
||||||
raise exception.PatchError(patch=patch, reason=e)
|
|
||||||
|
|
||||||
# Update only the fields that have changed.
|
# Update only the fields that have changed.
|
||||||
for field in objects.VolumeConnector.fields:
|
for field in objects.VolumeConnector.fields:
|
||||||
|
@ -432,17 +432,14 @@ class VolumeTargetsController(rest.RestController):
|
|||||||
raise exception.InvalidUUID(message=message)
|
raise exception.InvalidUUID(message=message)
|
||||||
|
|
||||||
rpc_target = objects.VolumeTarget.get_by_uuid(context, target_uuid)
|
rpc_target = objects.VolumeTarget.get_by_uuid(context, target_uuid)
|
||||||
try:
|
target_dict = rpc_target.as_dict()
|
||||||
target_dict = rpc_target.as_dict()
|
# NOTE(smoriya):
|
||||||
# NOTE(smoriya):
|
# 1) Remove node_id because it's an internal value and
|
||||||
# 1) Remove node_id because it's an internal value and
|
# not present in the API object
|
||||||
# not present in the API object
|
# 2) Add node_uuid
|
||||||
# 2) Add node_uuid
|
target_dict['node_uuid'] = target_dict.pop('node_id', None)
|
||||||
target_dict['node_uuid'] = target_dict.pop('node_id', None)
|
target = VolumeTarget(
|
||||||
target = VolumeTarget(
|
**api_utils.apply_jsonpatch(target_dict, patch))
|
||||||
**api_utils.apply_jsonpatch(target_dict, patch))
|
|
||||||
except api_utils.JSONPATCH_EXCEPTIONS as e:
|
|
||||||
raise exception.PatchError(patch=patch, reason=e)
|
|
||||||
|
|
||||||
# Update only the fields that have changed.
|
# Update only the fields that have changed.
|
||||||
for field in objects.VolumeTarget.fields:
|
for field in objects.VolumeTarget.fields:
|
||||||
|
@ -785,25 +785,59 @@ def _get_deployment_templates(task):
|
|||||||
instance_traits)
|
instance_traits)
|
||||||
|
|
||||||
|
|
||||||
def _get_steps_from_deployment_templates(task):
|
def _get_steps_from_deployment_templates(task, templates):
|
||||||
"""Get deployment template steps for task.node.
|
"""Get deployment template steps for task.node.
|
||||||
|
|
||||||
|
Given a list of deploy template objects, return a list of all deploy steps
|
||||||
|
combined.
|
||||||
|
|
||||||
|
:param task: A TaskManager object
|
||||||
|
:param templates: a list of deploy templates
|
||||||
|
:returns: A list of deploy step dictionaries
|
||||||
|
"""
|
||||||
|
steps = []
|
||||||
|
# NOTE(mgoddard): The steps from the object include id, created_at, etc.,
|
||||||
|
# which we don't want to include when we assign them to
|
||||||
|
# node.driver_internal_info. Include only the relevant fields.
|
||||||
|
step_fields = ('interface', 'step', 'args', 'priority')
|
||||||
|
for template in templates:
|
||||||
|
steps.extend([{key: step[key] for key in step_fields}
|
||||||
|
for step in template.steps])
|
||||||
|
return steps
|
||||||
|
|
||||||
|
|
||||||
|
def _get_validated_steps_from_templates(task):
|
||||||
|
"""Return a list of validated deploy steps from deploy templates.
|
||||||
|
|
||||||
Deployment template steps are those steps defined in deployment templates
|
Deployment template steps are those steps defined in deployment templates
|
||||||
where the name of the deployment template matches one of the node's
|
where the name of the deployment template matches one of the node's
|
||||||
instance traits (the subset of the node's traits requested by the user via
|
instance traits (the subset of the node's traits requested by the user via
|
||||||
a flavor or image). There may be many such matching templates, each with a
|
a flavor or image). There may be many such matching templates, each with a
|
||||||
list of steps to execute.
|
list of steps to execute.
|
||||||
|
|
||||||
|
This method gathers the steps from all matching deploy templates for a
|
||||||
|
node, and validates those steps against the node's driver interfaces,
|
||||||
|
raising an error if validation fails.
|
||||||
|
|
||||||
:param task: A TaskManager object
|
:param task: A TaskManager object
|
||||||
:returns: A list of deploy step dictionaries
|
:raises: InvalidParameterValue if validation of steps fails.
|
||||||
|
:raises: InstanceDeployFailure if there was a problem getting the
|
||||||
|
deploy steps.
|
||||||
|
:returns: A list of validated deploy step dictionaries
|
||||||
"""
|
"""
|
||||||
|
# Gather deploy templates matching the node's instance traits.
|
||||||
templates = _get_deployment_templates(task)
|
templates = _get_deployment_templates(task)
|
||||||
steps = []
|
|
||||||
step_fields = ('interface', 'step', 'args', 'priority')
|
# Gather deploy steps from deploy templates.
|
||||||
for template in templates:
|
user_steps = _get_steps_from_deployment_templates(task, templates)
|
||||||
steps.extend([{key: step[key] for key in step_fields}
|
|
||||||
for step in template.steps])
|
# Validate the steps.
|
||||||
return steps
|
error_prefix = (_('Validation of deploy steps from deploy templates '
|
||||||
|
'matching this node\'s instance traits failed. Matching '
|
||||||
|
'deploy templates: %(templates)s. Errors: ') %
|
||||||
|
{'templates': ','.join(t.name for t in templates)})
|
||||||
|
return _validate_user_deploy_steps(task, user_steps,
|
||||||
|
error_prefix=error_prefix)
|
||||||
|
|
||||||
|
|
||||||
def _get_all_deployment_steps(task):
|
def _get_all_deployment_steps(task):
|
||||||
@ -817,8 +851,11 @@ def _get_all_deployment_steps(task):
|
|||||||
deploy steps.
|
deploy steps.
|
||||||
:returns: A list of deploy step dictionaries
|
:returns: A list of deploy step dictionaries
|
||||||
"""
|
"""
|
||||||
# Gather deploy steps from deploy templates.
|
# Gather deploy steps from deploy templates and validate.
|
||||||
user_steps = _get_steps_from_deployment_templates(task)
|
# NOTE(mgoddard): although we've probably just validated the templates in
|
||||||
|
# do_node_deploy, they may have changed in the DB since we last checked, so
|
||||||
|
# validate again.
|
||||||
|
user_steps = _get_validated_steps_from_templates(task)
|
||||||
|
|
||||||
# Gather enabled deploy steps from drivers.
|
# Gather enabled deploy steps from drivers.
|
||||||
driver_steps = _get_deployment_steps(task, enabled=True, sort=False)
|
driver_steps = _get_deployment_steps(task, enabled=True, sort=False)
|
||||||
@ -852,6 +889,17 @@ def set_node_deployment_steps(task):
|
|||||||
node.save()
|
node.save()
|
||||||
|
|
||||||
|
|
||||||
|
def _step_id(step):
|
||||||
|
"""Return the 'ID' of a deploy step.
|
||||||
|
|
||||||
|
The ID is a string, <interface>.<step>.
|
||||||
|
|
||||||
|
:param step: the step dictionary.
|
||||||
|
:return: the step's ID string.
|
||||||
|
"""
|
||||||
|
return '.'.join([step['interface'], step['step']])
|
||||||
|
|
||||||
|
|
||||||
def _validate_deploy_steps_unique(user_steps):
|
def _validate_deploy_steps_unique(user_steps):
|
||||||
"""Validate that deploy steps from deploy templates are unique.
|
"""Validate that deploy steps from deploy templates are unique.
|
||||||
|
|
||||||
@ -860,28 +908,29 @@ def _validate_deploy_steps_unique(user_steps):
|
|||||||
|
|
||||||
{ 'interface': <driver_interface>,
|
{ 'interface': <driver_interface>,
|
||||||
'step': <name_of_step>,
|
'step': <name_of_step>,
|
||||||
'args': {<arg1>: <value1>, ..., <argn>: <valuen>} }
|
'args': {<arg1>: <value1>, ..., <argn>: <valuen>},
|
||||||
|
'priority': <priority_of_step> }
|
||||||
|
|
||||||
For example::
|
For example::
|
||||||
|
|
||||||
{ 'interface': deploy',
|
{ 'interface': deploy',
|
||||||
'step': 'upgrade_firmware',
|
'step': 'upgrade_firmware',
|
||||||
'args': {'force': True} }
|
'args': {'force': True},
|
||||||
|
'priority': 10 }
|
||||||
|
|
||||||
:return: a list of validation error strings for the steps.
|
:return: a list of validation error strings for the steps.
|
||||||
"""
|
"""
|
||||||
# Check for duplicate steps. Each interface/step combination can be
|
# Check for duplicate steps. Each interface/step combination can be
|
||||||
# specified at most once.
|
# specified at most once.
|
||||||
errors = []
|
errors = []
|
||||||
counter = collections.Counter((step['interface'], step['step'])
|
counter = collections.Counter(_step_id(step) for step in user_steps)
|
||||||
for step in user_steps)
|
duplicates = {step_id for step_id, count in counter.items() if count > 1}
|
||||||
for (interface, step), count in counter.items():
|
if duplicates:
|
||||||
if count > 1:
|
err = (_('deploy steps from all deploy templates matching this '
|
||||||
err = (_('duplicate deploy steps for %(interface)s.%(step)s. '
|
'node\'s instance traits cannot have the same interface '
|
||||||
'Deploy steps from all deploy templates matching a '
|
'and step. Duplicate deploy steps for %(duplicates)s') %
|
||||||
'node\'s instance traits cannot have the same interface '
|
{'duplicates': ', '.join(duplicates)})
|
||||||
'and step') %
|
errors.append(err)
|
||||||
{'interface': interface, 'step': step})
|
|
||||||
errors.append(err)
|
|
||||||
return errors
|
return errors
|
||||||
|
|
||||||
|
|
||||||
@ -897,12 +946,14 @@ def _validate_user_step(task, user_step, driver_step, step_type):
|
|||||||
'args': {<arg1>: <value1>, ..., <argn>: <valuen>},
|
'args': {<arg1>: <value1>, ..., <argn>: <valuen>},
|
||||||
'priority': <optional_priority> }
|
'priority': <optional_priority> }
|
||||||
|
|
||||||
For example::
|
For example::
|
||||||
|
|
||||||
{ 'interface': deploy',
|
{ 'interface': deploy',
|
||||||
'step': 'upgrade_firmware',
|
'step': 'upgrade_firmware',
|
||||||
'args': {'force': True} }
|
'args': {'force': True} }
|
||||||
|
|
||||||
:param driver_step: a driver step dictionary::
|
:param driver_step: a driver step dictionary::
|
||||||
|
|
||||||
{ 'interface': <driver_interface>,
|
{ 'interface': <driver_interface>,
|
||||||
'step': <name_of_step>,
|
'step': <name_of_step>,
|
||||||
'priority': <integer>
|
'priority': <integer>
|
||||||
@ -912,9 +963,9 @@ def _validate_user_step(task, user_step, driver_step, step_type):
|
|||||||
{<arg_name>:<arg_info_dict>} entries.
|
{<arg_name>:<arg_info_dict>} entries.
|
||||||
<arg_info_dict> is a dictionary with
|
<arg_info_dict> is a dictionary with
|
||||||
{ 'description': <description>,
|
{ 'description': <description>,
|
||||||
'required': <Boolean> }
|
'required': <Boolean> } }
|
||||||
}
|
|
||||||
For example::
|
For example::
|
||||||
|
|
||||||
{ 'interface': deploy',
|
{ 'interface': deploy',
|
||||||
'step': 'upgrade_firmware',
|
'step': 'upgrade_firmware',
|
||||||
@ -923,6 +974,7 @@ def _validate_user_step(task, user_step, driver_step, step_type):
|
|||||||
'argsinfo': {
|
'argsinfo': {
|
||||||
'force': { 'description': 'Whether to force the upgrade',
|
'force': { 'description': 'Whether to force the upgrade',
|
||||||
'required': False } } }
|
'required': False } } }
|
||||||
|
|
||||||
:param step_type: either 'clean' or 'deploy'.
|
:param step_type: either 'clean' or 'deploy'.
|
||||||
:return: a list of validation error strings for the step.
|
:return: a list of validation error strings for the step.
|
||||||
"""
|
"""
|
||||||
@ -930,12 +982,12 @@ def _validate_user_step(task, user_step, driver_step, step_type):
|
|||||||
# Check that the user-specified arguments are valid
|
# Check that the user-specified arguments are valid
|
||||||
argsinfo = driver_step.get('argsinfo') or {}
|
argsinfo = driver_step.get('argsinfo') or {}
|
||||||
user_args = user_step.get('args') or {}
|
user_args = user_step.get('args') or {}
|
||||||
invalid = set(user_args) - set(argsinfo)
|
unexpected = set(user_args) - set(argsinfo)
|
||||||
if invalid:
|
if unexpected:
|
||||||
error = (_('%(type)s step %(step)s has these invalid arguments: '
|
error = (_('%(type)s step %(step)s has these unexpected arguments: '
|
||||||
'%(invalid)s') %
|
'%(unexpected)s') %
|
||||||
{'type': step_type, 'step': user_step,
|
{'type': step_type, 'step': user_step,
|
||||||
'invalid': ', '.join(invalid)})
|
'unexpected': ', '.join(unexpected)})
|
||||||
errors.append(error)
|
errors.append(error)
|
||||||
|
|
||||||
if step_type == 'clean' or user_step['priority'] > 0:
|
if step_type == 'clean' or user_step['priority'] > 0:
|
||||||
@ -949,7 +1001,7 @@ def _validate_user_step(task, user_step, driver_step, step_type):
|
|||||||
missing.append(msg)
|
missing.append(msg)
|
||||||
if missing:
|
if missing:
|
||||||
error = (_('%(type)s step %(step)s is missing these required '
|
error = (_('%(type)s step %(step)s is missing these required '
|
||||||
'keyword arguments: %(miss)s') %
|
'arguments: %(miss)s') %
|
||||||
{'type': step_type, 'step': user_step,
|
{'type': step_type, 'step': user_step,
|
||||||
'miss': ', '.join(missing)})
|
'miss': ', '.join(missing)})
|
||||||
errors.append(error)
|
errors.append(error)
|
||||||
@ -960,19 +1012,24 @@ def _validate_user_step(task, user_step, driver_step, step_type):
|
|||||||
user_step['priority'] = driver_step.get('priority', 0)
|
user_step['priority'] = driver_step.get('priority', 0)
|
||||||
elif user_step['priority'] > 0:
|
elif user_step['priority'] > 0:
|
||||||
# 'core' deploy steps can only be disabled.
|
# 'core' deploy steps can only be disabled.
|
||||||
|
|
||||||
|
# NOTE(mgoddard): we'll need something a little more sophisticated to
|
||||||
|
# track core steps once we split out the single core step.
|
||||||
is_core = (driver_step['interface'] == 'deploy' and
|
is_core = (driver_step['interface'] == 'deploy' and
|
||||||
driver_step['step'] == 'deploy')
|
driver_step['step'] == 'deploy')
|
||||||
if is_core:
|
if is_core:
|
||||||
error = (_('deploy step %(step)s is a core step and '
|
error = (_('deploy step %(step)s on interface %(interface)s is a '
|
||||||
'cannot be overridden by user steps. It may be '
|
'core step and cannot be overridden by user steps. It '
|
||||||
'disabled by setting the priority to 0')
|
'may be disabled by setting the priority to 0') %
|
||||||
% {'step': user_step})
|
{'step': user_step['step'],
|
||||||
|
'interface': user_step['interface']})
|
||||||
errors.append(error)
|
errors.append(error)
|
||||||
|
|
||||||
return errors
|
return errors
|
||||||
|
|
||||||
|
|
||||||
def _validate_user_steps(task, user_steps, driver_steps, step_type):
|
def _validate_user_steps(task, user_steps, driver_steps, step_type,
|
||||||
|
error_prefix=None):
|
||||||
"""Validate the user-specified steps.
|
"""Validate the user-specified steps.
|
||||||
|
|
||||||
:param task: A TaskManager object
|
:param task: A TaskManager object
|
||||||
@ -985,12 +1042,14 @@ def _validate_user_steps(task, user_steps, driver_steps, step_type):
|
|||||||
'args': {<arg1>: <value1>, ..., <argn>: <valuen>},
|
'args': {<arg1>: <value1>, ..., <argn>: <valuen>},
|
||||||
'priority': <optional_priority> }
|
'priority': <optional_priority> }
|
||||||
|
|
||||||
For example::
|
For example::
|
||||||
|
|
||||||
{ 'interface': deploy',
|
{ 'interface': deploy',
|
||||||
'step': 'upgrade_firmware',
|
'step': 'upgrade_firmware',
|
||||||
'args': {'force': True} }
|
'args': {'force': True} }
|
||||||
|
|
||||||
:param driver_steps: a list of driver steps::
|
:param driver_steps: a list of driver steps::
|
||||||
|
|
||||||
{ 'interface': <driver_interface>,
|
{ 'interface': <driver_interface>,
|
||||||
'step': <name_of_step>,
|
'step': <name_of_step>,
|
||||||
'priority': <integer>
|
'priority': <integer>
|
||||||
@ -1000,9 +1059,9 @@ def _validate_user_steps(task, user_steps, driver_steps, step_type):
|
|||||||
{<arg_name>:<arg_info_dict>} entries.
|
{<arg_name>:<arg_info_dict>} entries.
|
||||||
<arg_info_dict> is a dictionary with
|
<arg_info_dict> is a dictionary with
|
||||||
{ 'description': <description>,
|
{ 'description': <description>,
|
||||||
'required': <Boolean> }
|
'required': <Boolean> } }
|
||||||
}
|
|
||||||
For example::
|
For example::
|
||||||
|
|
||||||
{ 'interface': deploy',
|
{ 'interface': deploy',
|
||||||
'step': 'upgrade_firmware',
|
'step': 'upgrade_firmware',
|
||||||
@ -1011,25 +1070,25 @@ def _validate_user_steps(task, user_steps, driver_steps, step_type):
|
|||||||
'argsinfo': {
|
'argsinfo': {
|
||||||
'force': { 'description': 'Whether to force the upgrade',
|
'force': { 'description': 'Whether to force the upgrade',
|
||||||
'required': False } } }
|
'required': False } } }
|
||||||
|
|
||||||
:param step_type: either 'clean' or 'deploy'.
|
:param step_type: either 'clean' or 'deploy'.
|
||||||
|
:param error_prefix: String to use as a prefix for exception messages, or
|
||||||
|
None.
|
||||||
:raises: InvalidParameterValue if validation of steps fails.
|
:raises: InvalidParameterValue if validation of steps fails.
|
||||||
:raises: NodeCleaningFailure or InstanceDeployFailure if
|
:raises: NodeCleaningFailure or InstanceDeployFailure if
|
||||||
there was a problem getting the steps from the driver.
|
there was a problem getting the steps from the driver.
|
||||||
:return: validated steps updated with information from the driver
|
:return: validated steps updated with information from the driver
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def step_id(step):
|
|
||||||
return '.'.join([step['step'], step['interface']])
|
|
||||||
|
|
||||||
errors = []
|
errors = []
|
||||||
|
|
||||||
# Convert driver steps to a dict.
|
# Convert driver steps to a dict.
|
||||||
driver_steps = {step_id(s): s for s in driver_steps}
|
driver_steps = {_step_id(s): s for s in driver_steps}
|
||||||
|
|
||||||
for user_step in user_steps:
|
for user_step in user_steps:
|
||||||
# Check if this user_specified step isn't supported by the driver
|
# Check if this user-specified step isn't supported by the driver
|
||||||
try:
|
try:
|
||||||
driver_step = driver_steps[step_id(user_step)]
|
driver_step = driver_steps[_step_id(user_step)]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
error = (_('node does not support this %(type)s step: %(step)s')
|
error = (_('node does not support this %(type)s step: %(step)s')
|
||||||
% {'type': step_type, 'step': user_step})
|
% {'type': step_type, 'step': user_step})
|
||||||
@ -1046,9 +1105,11 @@ def _validate_user_steps(task, user_steps, driver_steps, step_type):
|
|||||||
errors.extend(dup_errors)
|
errors.extend(dup_errors)
|
||||||
|
|
||||||
if errors:
|
if errors:
|
||||||
raise exception.InvalidParameterValue('; '.join(errors))
|
err = error_prefix or ''
|
||||||
|
err += '; '.join(errors)
|
||||||
|
raise exception.InvalidParameterValue(err=err)
|
||||||
|
|
||||||
return user_steps[:]
|
return user_steps
|
||||||
|
|
||||||
|
|
||||||
def _validate_user_clean_steps(task, user_steps):
|
def _validate_user_clean_steps(task, user_steps):
|
||||||
@ -1076,7 +1137,7 @@ def _validate_user_clean_steps(task, user_steps):
|
|||||||
return _validate_user_steps(task, user_steps, driver_steps, 'clean')
|
return _validate_user_steps(task, user_steps, driver_steps, 'clean')
|
||||||
|
|
||||||
|
|
||||||
def _validate_user_deploy_steps(task, user_steps):
|
def _validate_user_deploy_steps(task, user_steps, error_prefix=None):
|
||||||
"""Validate the user-specified deploy steps.
|
"""Validate the user-specified deploy steps.
|
||||||
|
|
||||||
:param task: A TaskManager object
|
:param task: A TaskManager object
|
||||||
@ -1094,13 +1155,16 @@ def _validate_user_deploy_steps(task, user_steps):
|
|||||||
'step': 'apply_configuration',
|
'step': 'apply_configuration',
|
||||||
'args': { 'settings': [ { 'foo': 'bar' } ] },
|
'args': { 'settings': [ { 'foo': 'bar' } ] },
|
||||||
'priority': 150 }
|
'priority': 150 }
|
||||||
|
:param error_prefix: String to use as a prefix for exception messages, or
|
||||||
|
None.
|
||||||
:raises: InvalidParameterValue if validation of deploy steps fails.
|
:raises: InvalidParameterValue if validation of deploy steps fails.
|
||||||
:raises: InstanceDeployFailure if there was a problem getting the deploy
|
:raises: InstanceDeployFailure if there was a problem getting the deploy
|
||||||
steps from the driver.
|
steps from the driver.
|
||||||
:return: validated deploy steps update with information from the driver
|
:return: validated deploy steps update with information from the driver
|
||||||
"""
|
"""
|
||||||
driver_steps = _get_deployment_steps(task, enabled=False, sort=False)
|
driver_steps = _get_deployment_steps(task, enabled=False, sort=False)
|
||||||
return _validate_user_steps(task, user_steps, driver_steps, 'deploy')
|
return _validate_user_steps(task, user_steps, driver_steps, 'deploy',
|
||||||
|
error_prefix=error_prefix)
|
||||||
|
|
||||||
|
|
||||||
@task_manager.require_exclusive_lock
|
@task_manager.require_exclusive_lock
|
||||||
@ -1315,9 +1379,8 @@ def validate_deploy_templates(task):
|
|||||||
:raises: InstanceDeployFailure if there was a problem getting the deploy
|
:raises: InstanceDeployFailure if there was a problem getting the deploy
|
||||||
steps from the driver.
|
steps from the driver.
|
||||||
"""
|
"""
|
||||||
# Gather deploy steps from matching deploy templates, validate them.
|
# Gather deploy steps from matching deploy templates and validate them.
|
||||||
user_steps = _get_steps_from_deployment_templates(task)
|
_get_validated_steps_from_templates(task)
|
||||||
_validate_user_deploy_steps(task, user_steps)
|
|
||||||
|
|
||||||
|
|
||||||
def build_configdrive(node, configdrive):
|
def build_configdrive(node, configdrive):
|
||||||
|
@ -175,19 +175,16 @@ class TestListDeployTemplates(BaseDeployTemplatesAPITest):
|
|||||||
|
|
||||||
def test_detail_query_false(self):
|
def test_detail_query_false(self):
|
||||||
obj_utils.create_test_deploy_template(self.context)
|
obj_utils.create_test_deploy_template(self.context)
|
||||||
data1 = self.get_json(
|
data1 = self.get_json('/deploy_templates', headers=self.headers)
|
||||||
'/deploy_templates',
|
|
||||||
headers={api_base.Version.string: str(api_v1.max_version())})
|
|
||||||
data2 = self.get_json(
|
data2 = self.get_json(
|
||||||
'/deploy_templates?detail=False',
|
'/deploy_templates?detail=False', headers=self.headers)
|
||||||
headers={api_base.Version.string: str(api_v1.max_version())})
|
|
||||||
self.assertEqual(data1['deploy_templates'], data2['deploy_templates'])
|
self.assertEqual(data1['deploy_templates'], data2['deploy_templates'])
|
||||||
|
|
||||||
def test_detail_using_query_false_and_fields(self):
|
def test_detail_using_query_false_and_fields(self):
|
||||||
obj_utils.create_test_deploy_template(self.context)
|
obj_utils.create_test_deploy_template(self.context)
|
||||||
data = self.get_json(
|
data = self.get_json(
|
||||||
'/deploy_templates?detail=False&fields=steps',
|
'/deploy_templates?detail=False&fields=steps',
|
||||||
headers={api_base.Version.string: str(api_v1.max_version())})
|
headers=self.headers)
|
||||||
self.assertIn('steps', data['deploy_templates'][0])
|
self.assertIn('steps', data['deploy_templates'][0])
|
||||||
self.assertNotIn('uuid', data['deploy_templates'][0])
|
self.assertNotIn('uuid', data['deploy_templates'][0])
|
||||||
self.assertNotIn('extra', data['deploy_templates'][0])
|
self.assertNotIn('extra', data['deploy_templates'][0])
|
||||||
@ -195,8 +192,7 @@ class TestListDeployTemplates(BaseDeployTemplatesAPITest):
|
|||||||
def test_detail_using_query_and_fields(self):
|
def test_detail_using_query_and_fields(self):
|
||||||
obj_utils.create_test_deploy_template(self.context)
|
obj_utils.create_test_deploy_template(self.context)
|
||||||
response = self.get_json(
|
response = self.get_json(
|
||||||
'/deploy_templates?detail=True&fields=name',
|
'/deploy_templates?detail=True&fields=name', headers=self.headers,
|
||||||
headers={api_base.Version.string: str(api_v1.max_version())},
|
|
||||||
expect_errors=True)
|
expect_errors=True)
|
||||||
self.assertEqual(http_client.BAD_REQUEST, response.status_int)
|
self.assertEqual(http_client.BAD_REQUEST, response.status_int)
|
||||||
|
|
||||||
@ -397,7 +393,14 @@ class TestPatch(BaseDeployTemplatesAPITest):
|
|||||||
def test_update_name_standard_trait(self, mock_save):
|
def test_update_name_standard_trait(self, mock_save):
|
||||||
name = 'HW_CPU_X86_VMX'
|
name = 'HW_CPU_X86_VMX'
|
||||||
patch = [{'path': '/name', 'value': name, 'op': 'replace'}]
|
patch = [{'path': '/name', 'value': name, 'op': 'replace'}]
|
||||||
self._test_update_ok(mock_save, patch)
|
response = self._test_update_ok(mock_save, patch)
|
||||||
|
self.assertEqual(name, response.json['name'])
|
||||||
|
|
||||||
|
def test_update_name_custom_trait(self, mock_save):
|
||||||
|
name = 'CUSTOM_DT2'
|
||||||
|
patch = [{'path': '/name', 'value': name, 'op': 'replace'}]
|
||||||
|
response = self._test_update_ok(mock_save, patch)
|
||||||
|
self.assertEqual(name, response.json['name'])
|
||||||
|
|
||||||
def test_update_invalid_name(self, mock_save):
|
def test_update_invalid_name(self, mock_save):
|
||||||
self._test_update_bad_request(
|
self._test_update_bad_request(
|
||||||
@ -441,12 +444,6 @@ class TestPatch(BaseDeployTemplatesAPITest):
|
|||||||
self.assertTrue(response.json['error_message'])
|
self.assertTrue(response.json['error_message'])
|
||||||
self.assertFalse(mock_save.called)
|
self.assertFalse(mock_save.called)
|
||||||
|
|
||||||
def test_replace_singular(self, mock_save):
|
|
||||||
name = 'CUSTOM_DT2'
|
|
||||||
patch = [{'path': '/name', 'value': name, 'op': 'replace'}]
|
|
||||||
response = self._test_update_ok(mock_save, patch)
|
|
||||||
self.assertEqual(name, response.json['name'])
|
|
||||||
|
|
||||||
@mock.patch.object(notification_utils, '_emit_api_notification',
|
@mock.patch.object(notification_utils, '_emit_api_notification',
|
||||||
autospec=True)
|
autospec=True)
|
||||||
def test_replace_name_already_exist(self, mock_notify, mock_save):
|
def test_replace_name_already_exist(self, mock_notify, mock_save):
|
||||||
@ -582,7 +579,7 @@ class TestPatch(BaseDeployTemplatesAPITest):
|
|||||||
patch = []
|
patch = []
|
||||||
for i, step in enumerate(steps):
|
for i, step in enumerate(steps):
|
||||||
patch.append({'path': '/steps/%s' % i,
|
patch.append({'path': '/steps/%s' % i,
|
||||||
'value': steps[i],
|
'value': step,
|
||||||
'op': 'replace'})
|
'op': 'replace'})
|
||||||
response = self.patch_json('/deploy_templates/%s' % template.uuid,
|
response = self.patch_json('/deploy_templates/%s' % template.uuid,
|
||||||
patch, headers=self.headers)
|
patch, headers=self.headers)
|
||||||
|
@ -86,6 +86,36 @@ class TestApiUtils(base.TestCase):
|
|||||||
"spongebob",
|
"spongebob",
|
||||||
utils.validate_trait, "invalid", "spongebob")
|
utils.validate_trait, "invalid", "spongebob")
|
||||||
|
|
||||||
|
def test_apply_jsonpatch(self):
|
||||||
|
doc = {"foo": {"bar": "baz"}}
|
||||||
|
patch = [{"op": "add", "path": "/foo/answer", "value": 42}]
|
||||||
|
result = utils.apply_jsonpatch(doc, patch)
|
||||||
|
expected = {"foo": {"bar": "baz", "answer": 42}}
|
||||||
|
self.assertEqual(expected, result)
|
||||||
|
|
||||||
|
def test_apply_jsonpatch_no_add_root_attr(self):
|
||||||
|
doc = {}
|
||||||
|
patch = [{"op": "add", "path": "/foo", "value": 42}]
|
||||||
|
self.assertRaisesRegex(wsme.exc.ClientSideError,
|
||||||
|
"Adding a new attribute",
|
||||||
|
utils.apply_jsonpatch, doc, patch)
|
||||||
|
|
||||||
|
def test_apply_jsonpatch_remove_non_existent(self):
|
||||||
|
# Raises a KeyError.
|
||||||
|
doc = {}
|
||||||
|
patch = [{"op": "remove", "path": "/foo"}]
|
||||||
|
self.assertRaisesRegex(exception.PatchError,
|
||||||
|
"can't remove non-existent object 'foo'",
|
||||||
|
utils.apply_jsonpatch, doc, patch)
|
||||||
|
|
||||||
|
def test_apply_jsonpatch_replace_non_existent_list_item(self):
|
||||||
|
# Raises an IndexError.
|
||||||
|
doc = []
|
||||||
|
patch = [{"op": "replace", "path": "/0", "value": 42}]
|
||||||
|
self.assertRaisesRegex(exception.PatchError,
|
||||||
|
"list assignment index out of range",
|
||||||
|
utils.apply_jsonpatch, doc, patch)
|
||||||
|
|
||||||
def test_get_patch_values_no_path(self):
|
def test_get_patch_values_no_path(self):
|
||||||
patch = [{'path': '/name', 'op': 'update', 'value': 'node-0'}]
|
patch = [{'path': '/name', 'op': 'update', 'value': 'node-0'}]
|
||||||
path = '/invalid'
|
path = '/invalid'
|
||||||
|
@ -1078,7 +1078,7 @@ class NodeDeployStepsTestCase(db_base.DbTestCase):
|
|||||||
traits = ['CUSTOM_DT1', 'CUSTOM_DT2']
|
traits = ['CUSTOM_DT1', 'CUSTOM_DT2']
|
||||||
node = obj_utils.create_test_node(
|
node = obj_utils.create_test_node(
|
||||||
self.context, uuid=uuidutils.generate_uuid(),
|
self.context, uuid=uuidutils.generate_uuid(),
|
||||||
driver='fake-hardware', instance_info={'traits': traits})
|
instance_info={'traits': traits})
|
||||||
template1 = obj_utils.get_test_deploy_template(self.context)
|
template1 = obj_utils.get_test_deploy_template(self.context)
|
||||||
template2 = obj_utils.get_test_deploy_template(
|
template2 = obj_utils.get_test_deploy_template(
|
||||||
self.context, name='CUSTOM_DT2', uuid=uuidutils.generate_uuid(),
|
self.context, name='CUSTOM_DT2', uuid=uuidutils.generate_uuid(),
|
||||||
@ -1092,15 +1092,12 @@ class NodeDeployStepsTestCase(db_base.DbTestCase):
|
|||||||
self.assertEqual(expected, templates)
|
self.assertEqual(expected, templates)
|
||||||
mock_list.assert_called_once_with(task.context, traits)
|
mock_list.assert_called_once_with(task.context, traits)
|
||||||
|
|
||||||
@mock.patch.object(conductor_utils, '_get_deployment_templates',
|
def test__get_steps_from_deployment_templates(self):
|
||||||
autospec=True)
|
|
||||||
def test__get_steps_from_deployment_templates(self, mock_templates):
|
|
||||||
template1 = obj_utils.get_test_deploy_template(self.context)
|
template1 = obj_utils.get_test_deploy_template(self.context)
|
||||||
template2 = obj_utils.get_test_deploy_template(
|
template2 = obj_utils.get_test_deploy_template(
|
||||||
self.context, name='CUSTOM_DT2', uuid=uuidutils.generate_uuid(),
|
self.context, name='CUSTOM_DT2', uuid=uuidutils.generate_uuid(),
|
||||||
steps=[{'interface': 'bios', 'step': 'apply_configuration',
|
steps=[{'interface': 'bios', 'step': 'apply_configuration',
|
||||||
'args': {}, 'priority': 1}])
|
'args': {}, 'priority': 1}])
|
||||||
mock_templates.return_value = [template1, template2]
|
|
||||||
step1 = template1.steps[0]
|
step1 = template1.steps[0]
|
||||||
step2 = template2.steps[0]
|
step2 = template2.steps[0]
|
||||||
expected = [
|
expected = [
|
||||||
@ -1119,24 +1116,25 @@ class NodeDeployStepsTestCase(db_base.DbTestCase):
|
|||||||
]
|
]
|
||||||
with task_manager.acquire(
|
with task_manager.acquire(
|
||||||
self.context, self.node.uuid, shared=False) as task:
|
self.context, self.node.uuid, shared=False) as task:
|
||||||
steps = conductor_utils._get_steps_from_deployment_templates(task)
|
steps = conductor_utils._get_steps_from_deployment_templates(
|
||||||
|
task, [template1, template2])
|
||||||
self.assertEqual(expected, steps)
|
self.assertEqual(expected, steps)
|
||||||
mock_templates.assert_called_once_with(task)
|
|
||||||
|
|
||||||
@mock.patch.object(conductor_utils, '_get_steps_from_deployment_templates',
|
@mock.patch.object(conductor_utils, '_get_validated_steps_from_templates',
|
||||||
autospec=True)
|
autospec=True)
|
||||||
@mock.patch.object(conductor_utils, '_get_deployment_steps', autospec=True)
|
@mock.patch.object(conductor_utils, '_get_deployment_steps', autospec=True)
|
||||||
def _test__get_all_deployment_steps(self, user_steps, driver_steps,
|
def _test__get_all_deployment_steps(self, user_steps, driver_steps,
|
||||||
expected_steps, mock_gds, mock_gsfdt):
|
expected_steps, mock_steps,
|
||||||
mock_gsfdt.return_value = user_steps
|
mock_validated):
|
||||||
mock_gds.return_value = driver_steps
|
mock_validated.return_value = user_steps
|
||||||
|
mock_steps.return_value = driver_steps
|
||||||
|
|
||||||
with task_manager.acquire(
|
with task_manager.acquire(
|
||||||
self.context, self.node.uuid, shared=False) as task:
|
self.context, self.node.uuid, shared=False) as task:
|
||||||
steps = conductor_utils._get_all_deployment_steps(task)
|
steps = conductor_utils._get_all_deployment_steps(task)
|
||||||
self.assertEqual(expected_steps, steps)
|
self.assertEqual(expected_steps, steps)
|
||||||
mock_gsfdt.assert_called_once_with(task)
|
mock_validated.assert_called_once_with(task)
|
||||||
mock_gds.assert_called_once_with(task, enabled=True, sort=False)
|
mock_steps.assert_called_once_with(task, enabled=True, sort=False)
|
||||||
|
|
||||||
def test__get_all_deployment_steps_no_steps(self):
|
def test__get_all_deployment_steps_no_steps(self):
|
||||||
# Nothing in -> nothing out.
|
# Nothing in -> nothing out.
|
||||||
@ -1206,6 +1204,19 @@ class NodeDeployStepsTestCase(db_base.DbTestCase):
|
|||||||
self._test__get_all_deployment_steps(user_steps, driver_steps,
|
self._test__get_all_deployment_steps(user_steps, driver_steps,
|
||||||
expected_steps)
|
expected_steps)
|
||||||
|
|
||||||
|
@mock.patch.object(conductor_utils, '_get_validated_steps_from_templates',
|
||||||
|
autospec=True)
|
||||||
|
@mock.patch.object(conductor_utils, '_get_deployment_steps', autospec=True)
|
||||||
|
def test__get_all_deployment_steps_error(self, mock_steps, mock_validated):
|
||||||
|
mock_validated.side_effect = exception.InvalidParameterValue('foo')
|
||||||
|
|
||||||
|
with task_manager.acquire(
|
||||||
|
self.context, self.node.uuid, shared=False) as task:
|
||||||
|
self.assertRaises(exception.InvalidParameterValue,
|
||||||
|
conductor_utils._get_all_deployment_steps, task)
|
||||||
|
mock_validated.assert_called_once_with(task)
|
||||||
|
self.assertFalse(mock_steps.called)
|
||||||
|
|
||||||
@mock.patch.object(conductor_utils, '_get_all_deployment_steps',
|
@mock.patch.object(conductor_utils, '_get_all_deployment_steps',
|
||||||
autospec=True)
|
autospec=True)
|
||||||
def test_set_node_deployment_steps(self, mock_steps):
|
def test_set_node_deployment_steps(self, mock_steps):
|
||||||
@ -1280,7 +1291,7 @@ class NodeDeployStepsTestCase(db_base.DbTestCase):
|
|||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
||||||
self.assertRaisesRegex(exception.InvalidParameterValue,
|
self.assertRaisesRegex(exception.InvalidParameterValue,
|
||||||
"power_one.*invalid.*arg1",
|
"power_one.*unexpected.*arg1",
|
||||||
conductor_utils._validate_user_deploy_steps,
|
conductor_utils._validate_user_deploy_steps,
|
||||||
task, user_steps)
|
task, user_steps)
|
||||||
mock_steps.assert_called_once_with(task, enabled=False, sort=False)
|
mock_steps.assert_called_once_with(task, enabled=False, sort=False)
|
||||||
@ -1356,7 +1367,7 @@ class NodeDeployStepsTestCase(db_base.DbTestCase):
|
|||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
||||||
self.assertRaisesRegex(exception.InvalidParameterValue,
|
self.assertRaisesRegex(exception.InvalidParameterValue,
|
||||||
"duplicate deploy steps for "
|
"Duplicate deploy steps for "
|
||||||
"power.power_one",
|
"power.power_one",
|
||||||
conductor_utils._validate_user_deploy_steps,
|
conductor_utils._validate_user_deploy_steps,
|
||||||
task, user_steps)
|
task, user_steps)
|
||||||
@ -1560,7 +1571,7 @@ class NodeCleaningStepsTestCase(db_base.DbTestCase):
|
|||||||
|
|
||||||
with task_manager.acquire(self.context, node.uuid) as task:
|
with task_manager.acquire(self.context, node.uuid) as task:
|
||||||
self.assertRaisesRegex(exception.InvalidParameterValue,
|
self.assertRaisesRegex(exception.InvalidParameterValue,
|
||||||
"update_firmware.*invalid.*arg1",
|
"update_firmware.*unexpected.*arg1",
|
||||||
conductor_utils._validate_user_clean_steps,
|
conductor_utils._validate_user_clean_steps,
|
||||||
task, user_steps)
|
task, user_steps)
|
||||||
mock_steps.assert_called_once_with(task, enabled=False, sort=False)
|
mock_steps.assert_called_once_with(task, enabled=False, sort=False)
|
||||||
@ -2455,10 +2466,56 @@ class ValidateInstanceInfoTraitsTestCase(tests_base.TestCase):
|
|||||||
self.node)
|
self.node)
|
||||||
|
|
||||||
|
|
||||||
|
@mock.patch.object(conductor_utils, '_get_deployment_templates',
|
||||||
|
autospec=True)
|
||||||
@mock.patch.object(conductor_utils, '_get_steps_from_deployment_templates',
|
@mock.patch.object(conductor_utils, '_get_steps_from_deployment_templates',
|
||||||
autospec=True)
|
autospec=True)
|
||||||
@mock.patch.object(conductor_utils, '_validate_user_deploy_steps',
|
@mock.patch.object(conductor_utils, '_validate_user_deploy_steps',
|
||||||
autospec=True)
|
autospec=True)
|
||||||
|
class GetValidatedStepsFromTemplatesTestCase(db_base.DbTestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(GetValidatedStepsFromTemplatesTestCase, self).setUp()
|
||||||
|
self.node = obj_utils.create_test_node(self.context,
|
||||||
|
driver='fake-hardware')
|
||||||
|
self.template = obj_utils.get_test_deploy_template(self.context)
|
||||||
|
|
||||||
|
def test_ok(self, mock_validate, mock_steps, mock_templates):
|
||||||
|
mock_templates.return_value = [self.template]
|
||||||
|
steps = [db_utils.get_test_deploy_template_step()]
|
||||||
|
mock_steps.return_value = steps
|
||||||
|
mock_validate.return_value = steps
|
||||||
|
with task_manager.acquire(
|
||||||
|
self.context, self.node.uuid, shared=False) as task:
|
||||||
|
result = conductor_utils._get_validated_steps_from_templates(task)
|
||||||
|
self.assertEqual(steps, result)
|
||||||
|
mock_templates.assert_called_once_with(task)
|
||||||
|
mock_steps.assert_called_once_with(task, [self.template])
|
||||||
|
mock_validate.assert_called_once_with(task, steps, mock.ANY)
|
||||||
|
|
||||||
|
def test_invalid_parameter_value(self, mock_validate, mock_steps,
|
||||||
|
mock_templates):
|
||||||
|
mock_templates.return_value = [self.template]
|
||||||
|
mock_validate.side_effect = exception.InvalidParameterValue('fake')
|
||||||
|
with task_manager.acquire(
|
||||||
|
self.context, self.node.uuid, shared=False) as task:
|
||||||
|
self.assertRaises(
|
||||||
|
exception.InvalidParameterValue,
|
||||||
|
conductor_utils._get_validated_steps_from_templates, task)
|
||||||
|
|
||||||
|
def test_instance_deploy_failure(self, mock_validate, mock_steps,
|
||||||
|
mock_templates):
|
||||||
|
mock_templates.return_value = [self.template]
|
||||||
|
mock_validate.side_effect = exception.InstanceDeployFailure('foo')
|
||||||
|
with task_manager.acquire(
|
||||||
|
self.context, self.node.uuid, shared=False) as task:
|
||||||
|
self.assertRaises(
|
||||||
|
exception.InstanceDeployFailure,
|
||||||
|
conductor_utils._get_validated_steps_from_templates, task)
|
||||||
|
|
||||||
|
|
||||||
|
@mock.patch.object(conductor_utils, '_get_validated_steps_from_templates',
|
||||||
|
autospec=True)
|
||||||
class ValidateDeployTemplatesTestCase(db_base.DbTestCase):
|
class ValidateDeployTemplatesTestCase(db_base.DbTestCase):
|
||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
@ -2466,26 +2523,17 @@ class ValidateDeployTemplatesTestCase(db_base.DbTestCase):
|
|||||||
self.node = obj_utils.create_test_node(self.context,
|
self.node = obj_utils.create_test_node(self.context,
|
||||||
driver='fake-hardware')
|
driver='fake-hardware')
|
||||||
|
|
||||||
def test_validate_deploy_templates(self, mock_validate, mock_get):
|
def test_ok(self, mock_validated):
|
||||||
steps = [db_utils.get_test_deploy_template_step()]
|
|
||||||
mock_get.return_value = steps
|
|
||||||
with task_manager.acquire(
|
with task_manager.acquire(
|
||||||
self.context, self.node.uuid, shared=False) as task:
|
self.context, self.node.uuid, shared=False) as task:
|
||||||
conductor_utils.validate_deploy_templates(task)
|
result = conductor_utils.validate_deploy_templates(task)
|
||||||
mock_validate.assert_called_once_with(task, steps)
|
self.assertIsNone(result)
|
||||||
|
mock_validated.assert_called_once_with(task)
|
||||||
|
|
||||||
def test_validate_deploy_templates_invalid_parameter_value(
|
def test_error(self, mock_validated):
|
||||||
self, mock_validate, mock_get):
|
|
||||||
mock_validate.side_effect = exception.InvalidParameterValue('fake')
|
|
||||||
with task_manager.acquire(
|
with task_manager.acquire(
|
||||||
self.context, self.node.uuid, shared=False) as task:
|
self.context, self.node.uuid, shared=False) as task:
|
||||||
|
mock_validated.side_effect = exception.InvalidParameterValue('foo')
|
||||||
self.assertRaises(exception.InvalidParameterValue,
|
self.assertRaises(exception.InvalidParameterValue,
|
||||||
conductor_utils.validate_deploy_templates, task)
|
conductor_utils.validate_deploy_templates, task)
|
||||||
|
mock_validated.assert_called_once_with(task)
|
||||||
def test_validate_deploy_templates_instance_deploy_failure(
|
|
||||||
self, mock_validate, mock_get):
|
|
||||||
mock_validate.side_effect = exception.InstanceDeployFailure('foo')
|
|
||||||
with task_manager.acquire(
|
|
||||||
self.context, self.node.uuid, shared=False) as task:
|
|
||||||
self.assertRaises(exception.InstanceDeployFailure,
|
|
||||||
conductor_utils.validate_deploy_templates, task)
|
|
||||||
|
@ -5,8 +5,8 @@ features:
|
|||||||
the node deployment process, each specifying a list of deploy steps to
|
the node deployment process, each specifying a list of deploy steps to
|
||||||
execute with configurable priority and arguments.
|
execute with configurable priority and arguments.
|
||||||
|
|
||||||
Introduces the following new API endpoints, available from Bare Metal REST
|
Introduces the following new API endpoints, available from Bare Metal API
|
||||||
API version 1.55:
|
version 1.55:
|
||||||
|
|
||||||
* ``GET /v1/deploy_templates``
|
* ``GET /v1/deploy_templates``
|
||||||
* ``GET /v1/deploy_templates/<deploy template identifier>``
|
* ``GET /v1/deploy_templates/<deploy template identifier>``
|
||||||
|
Loading…
x
Reference in New Issue
Block a user