Updating os_ironic_node module

Updating the os_ironic_node module to the most recent version
including support for power and maintenance states.
This commit is contained in:
Julia Kreger 2015-05-01 09:51:59 -04:00 committed by Matt Clay
parent 51149b9643
commit d67ba5e25b

View file

@ -1,7 +1,7 @@
#!/usr/bin/python #!/usr/bin/python
# coding: utf-8 -*- # coding: utf-8 -*-
# (c) 2014, Hewlett-Packard Development Company, L.P. # (c) 2015, Hewlett-Packard Development Company, L.P.
# #
# This module is free software: you can redistribute it and/or modify # This module is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by # it under the terms of the GNU General Public License as published by
@ -25,7 +25,6 @@ except ImportError:
DOCUMENTATION = ''' DOCUMENTATION = '''
--- ---
module: os_ironic_node module: os_ironic_node
version_added: "1.10"
short_description: Activate/Deactivate Bare Metal Resources from OpenStack short_description: Activate/Deactivate Bare Metal Resources from OpenStack
extends_documentation_fragment: openstack extends_documentation_fragment: openstack
description: description:
@ -36,6 +35,13 @@ options:
- Indicates desired state of the resource - Indicates desired state of the resource
choices: ['present', 'absent'] choices: ['present', 'absent']
default: present default: present
deploy:
description:
- Indicates if the resource should be deployed. Allows for deployment
logic to be disengaged and control of the node power or maintenance
state to be changed.
choices: ['true', 'false']
default: true
uuid: uuid:
description: description:
- globally unique identifier (UUID) to be given to the resource. - globally unique identifier (UUID) to be given to the resource.
@ -44,7 +50,7 @@ options:
ironic_url: ironic_url:
description: description:
- If noauth mode is utilized, this is required to be set to the - If noauth mode is utilized, this is required to be set to the
endpoint URL for the Ironic API. Use with "auth" and "auth_plugin" endpoint URL for the Ironic API. Use with "auth" and "auth_type"
settings set to None. settings set to None.
required: false required: false
default: None default: None
@ -57,7 +63,8 @@ options:
instance_info: instance_info:
description: description:
- Definition of the instance information which is used to deploy - Definition of the instance information which is used to deploy
the node. the node. This information is only required when an instance is
set to present.
image_source: image_source:
description: description:
- An HTTP(S) URL where the image can be retrieved from. - An HTTP(S) URL where the image can be retrieved from.
@ -67,6 +74,26 @@ options:
image_disk_format: image_disk_format:
description: description:
- The type of image that has been requested to be deployed. - The type of image that has been requested to be deployed.
power:
description:
- A setting to allow power state to be asserted allowing nodes
that are not yet deployed to be powered on, and nodes that
are deployed to be powered off.
choices: ['present', 'absent']
default: present
maintenance:
description:
- A setting to allow the direct control if a node is in
maintenance mode.
required: false
default: false
maintenance_reason:
description:
- A string expression regarding the reason a node is in a
maintenance mode.
required: false
default: None
requirements: ["shade"] requirements: ["shade"]
''' '''
@ -76,6 +103,9 @@ os_ironic_node:
cloud: "openstack" cloud: "openstack"
uuid: "d44666e1-35b3-4f6b-acb0-88ab7052da69" uuid: "d44666e1-35b3-4f6b-acb0-88ab7052da69"
state: present state: present
power: present
deploy: True
maintenance: False
config_drive: "http://192.168.1.1/host-configdrive.iso" config_drive: "http://192.168.1.1/host-configdrive.iso"
instance_info: instance_info:
image_source: "http://192.168.1.1/deploy_image.img" image_source: "http://192.168.1.1/deploy_image.img"
@ -85,6 +115,16 @@ os_ironic_node:
''' '''
def _choose_id_value(module):
if module.params['uuid']:
return module.params['uuid']
if module.params['name']:
return module.params['name']
return None
# TODO(TheJulia): Change this over to use the machine patch method
# in shade once it is available.
def _prepare_instance_info_patch(instance_info): def _prepare_instance_info_patch(instance_info):
patch = [] patch = []
patch.append({ patch.append({
@ -95,56 +135,193 @@ def _prepare_instance_info_patch(instance_info):
return patch return patch
def _is_true(value):
true_values = [True, 'yes', 'Yes', 'True', 'true', 'present', 'on']
if value in true_values:
return True
return False
def _is_false(value):
false_values = [False, None, 'no', 'No', 'False', 'false', 'absent', 'off']
if value in false_values:
return True
return False
def _check_set_maintenance(module, cloud, node):
if _is_true(module.params['maintenance']):
if _is_false(node['maintenance']):
cloud.set_machine_maintenance_state(
node['uuid'],
True,
reason=module.params['maintenance_reason'])
module.exit_json(changed=True, msg="Node has been set into "
"maintenance mode")
else:
# User has requested maintenance state, node is already in the
# desired state, checking to see if the reason has changed.
if (str(node['maintenance_reason']) not in
str(module.params['maintenance_reason'])):
cloud.set_machine_maintenance_state(
node['uuid'],
True,
reason=module.params['maintenance_reason'])
module.exit_json(changed=True, msg="Node maintenance reason "
"updated, cannot take any "
"additional action.")
elif _is_false(module.params['maintenance']):
if node['maintenance'] is True:
cloud.remove_machine_from_maintenance(node['uuid'])
return True
else:
module.fail_json(msg="maintenance parameter was set but a valid "
"the value was not recognized.")
return False
def _check_set_power_state(module, cloud, node):
if 'power on' in str(node['power_state']):
if _is_false(module.params['power']):
# User has requested the node be powered off.
cloud.set_machine_power_off(node['uuid'])
module.exit_json(changed=True, msg="Power requested off")
if 'power off' in str(node['power_state']):
if (_is_false(module.params['power']) and
_is_false(module.params['state'])):
return False
if (_is_false(module.params['power']) and
_is_false(module.params['state'])):
module.exit_json(
changed=False,
msg="Power for node is %s, node must be reactivated "
"OR set to state absent"
)
# In the event the power has been toggled on and
# deployment has been requested, we need to skip this
# step.
if (_is_true(module.params['power']) and
_is_false(module.params['deploy'])):
# Node is powered down when it is not awaiting to be provisioned
cloud.set_machine_power_on(node['uuid'])
return True
# Default False if no action has been taken.
return False
def main(): def main():
argument_spec = openstack_full_argument_spec( argument_spec = openstack_full_argument_spec(
uuid=dict(required=True), uuid=dict(required=False),
instance_info=dict(type='dict', required=True), name=dict(required=False),
instance_info=dict(type='dict', required=False),
config_drive=dict(required=False), config_drive=dict(required=False),
ironic_url=dict(required=False), ironic_url=dict(required=False),
state=dict(required=False, default='present'),
maintenance=dict(required=False),
maintenance_reason=dict(required=False),
power=dict(required=False, default='present'),
deploy=dict(required=False, default=True),
) )
module_kwargs = openstack_module_kwargs() module_kwargs = openstack_module_kwargs()
module = AnsibleModule(argument_spec, **module_kwargs) module = AnsibleModule(argument_spec, **module_kwargs)
if not HAS_SHADE: if not HAS_SHADE:
module.fail_json(msg='shade is required for this module') module.fail_json(msg='shade is required for this module')
if (module.params['auth_plugin'] == 'None' and if (module.params['auth_type'] in [None, 'None'] and
module.params['ironic_url'] is None): module.params['ironic_url'] is None):
module.fail_json(msg="Authentication appears disabled, Please " module.fail_json(msg="Authentication appears disabled, Please "
"define an ironic_url parameter") "define an ironic_url parameter")
if module.params['ironic_url'] and module.params['auth_plugin'] == 'None': if (module.params['ironic_url'] and
module.params['auth'] = dict(endpoint=module.params['ironic_url']) module.params['auth_type'] in [None, 'None']):
module.params['auth'] = dict(
endpoint=module.params['ironic_url']
)
node_id = _choose_id_value(module)
if not node_id:
module.fail_json(msg="A uuid or name value must be defined "
"to use this module.")
try: try:
cloud = shade.operator_cloud(**module.params) cloud = shade.operator_cloud(**module.params)
server = cloud.get_machine_by_uuid(module.params['uuid']) node = cloud.get_machine(node_id)
if node is None:
module.fail_json(msg="node not found")
uuid = node['uuid']
instance_info = module.params['instance_info'] instance_info = module.params['instance_info']
uuid = module.params['uuid'] changed = False
if module.params['state'] == 'present':
if server is None:
module.fail_json(msg="node not found")
else:
# TODO: compare properties here and update if necessary
# ... but the interface for that is terrible!
if server.provision_state is "active":
module.exit_json(
changed=False,
result="Node already in an active state"
)
patch = _prepare_instance_info_patch(instance_info) # User has reqeusted desired state to be in maintenance state.
cloud.set_node_instance_info(uuid, patch) if module.params['state'] is 'maintenance':
cloud.validate_node(uuid) module.params['maintenance'] = True
cloud.activate_node(uuid, module.params['config_drive'])
# TODO: Add more error checking and a wait option.
module.exit_json(changed=False, result="node activated")
if module.params['state'] == 'absent': if node['provision_state'] in [
if server.provision_state is not "deleted": 'cleaning',
'deleting',
'wait call-back']:
module.fail_json(msg="Node is in %s state, cannot act upon the "
"request as the node is in a transition "
"state" % node['provision_state'])
# TODO(TheJulia) This is in-development code, that requires
# code in the shade library that is still in development.
if _check_set_maintenance(module, cloud, node):
if node['provision_state'] in 'active':
module.exit_json(changed=True,
result="Maintenance state changed")
changed = True
node = cloud.get_machine(node_id)
if _check_set_power_state(module, cloud, node):
changed = True
node = cloud.get_machine(node_id)
if _is_true(module.params['state']):
if _is_false(module.params['deploy']):
module.exit_json(
changed=changed,
result="User request has explicitly disabled "
"deployment logic"
)
if 'active' in node['provision_state']:
module.exit_json(
changed=changed,
result="Node already in an active state."
)
if instance_info is None:
module.fail_json(
changed=changed,
msg="When setting an instance to present, "
"instance_info is a required variable.")
# TODO(TheJulia): Update instance info, however info is
# deployment specific. Perhaps consider adding rebuild
# support, although there is a known desire to remove
# rebuild support from Ironic at some point in the future.
patch = _prepare_instance_info_patch(instance_info)
cloud.set_node_instance_info(uuid, patch)
cloud.validate_node(uuid)
cloud.activate_node(uuid, module.params['config_drive'])
# TODO(TheJulia): Add more error checking and a wait option.
# We will need to loop, or just add the logic to shade,
# although this could be a very long running process as
# baremetal deployments are not a "quick" task.
module.exit_json(changed=changed, result="node activated")
elif _is_false(module.params['state']):
if node['provision_state'] not in "deleted":
cloud.purge_node_instance_info(uuid) cloud.purge_node_instance_info(uuid)
cloud.deactivate_node(uuid) cloud.deactivate_node(uuid)
module.exit_json(changed=True, result="deleted") module.exit_json(changed=True, result="deleted")
else: else:
module.exit_json(changed=False, result="node not found") module.exit_json(changed=False, result="node not found")
else:
module.fail_json(msg="State must be present, absent, "
"maintenance, off")
except shade.OpenStackCloudException as e: except shade.OpenStackCloudException as e:
module.fail_json(msg=e.message) module.fail_json(msg=e.message)