Adding TowerCLI send module (#37843)
This commit is contained in:
parent
5671ff983d
commit
9722254618
3 changed files with 244 additions and 0 deletions
|
@ -0,0 +1,167 @@
|
|||
#!/usr/bin/python
|
||||
# coding: utf-8 -*-
|
||||
|
||||
# (c) 2017, John Westcott IV <john.westcott.iv@redhat.com>
|
||||
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||
|
||||
from __future__ import absolute_import, division, print_function
|
||||
__metaclass__ = type
|
||||
|
||||
|
||||
ANSIBLE_METADATA = {'metadata_version': '1.1',
|
||||
'status': ['preview'],
|
||||
'supported_by': 'community'}
|
||||
|
||||
|
||||
DOCUMENTATION = '''
|
||||
---
|
||||
module: tower_send
|
||||
author: "John Westcott IV (@john-westcott-iv)"
|
||||
version_added: "2.8"
|
||||
short_description: Send assets to Ansible Tower.
|
||||
description:
|
||||
- Send assets to Ansible Tower. See
|
||||
U(https://www.ansible.com/tower) for an overview.
|
||||
options:
|
||||
assets:
|
||||
description:
|
||||
- The assets to import.
|
||||
- This can be the output of tower_receive or loaded from a file
|
||||
required: False
|
||||
files:
|
||||
description:
|
||||
- List of files to import.
|
||||
required: False
|
||||
default: []
|
||||
prevent:
|
||||
description:
|
||||
- A list of asset types to prevent import for
|
||||
required: false
|
||||
default: []
|
||||
password_management:
|
||||
description:
|
||||
- The password management option to use.
|
||||
- The prompt option is not supported.
|
||||
required: false
|
||||
default: 'default'
|
||||
choices: ["default", "random"]
|
||||
|
||||
notes:
|
||||
- One of assets or files needs to be passed in
|
||||
|
||||
requirements:
|
||||
- "ansible-tower-cli >= 3.3.0"
|
||||
- six.moves.StringIO
|
||||
- sys
|
||||
|
||||
extends_documentation_fragment: tower
|
||||
'''
|
||||
|
||||
EXAMPLES = '''
|
||||
- name: Import all tower assets
|
||||
tower_send:
|
||||
assets: "{{ export_output.assets }}"
|
||||
tower_config_file: "~/tower_cli.cfg"
|
||||
'''
|
||||
|
||||
RETURN = '''
|
||||
output:
|
||||
description: The import messages
|
||||
returned: success, fail
|
||||
type: list
|
||||
sample: [ 'Message 1', 'Messag 2' ]
|
||||
'''
|
||||
|
||||
import os
|
||||
import sys
|
||||
|
||||
from ansible.module_utils.six.moves import StringIO
|
||||
from ansible.module_utils.ansible_tower import TowerModule, tower_auth_config, HAS_TOWER_CLI
|
||||
|
||||
from tempfile import mkstemp
|
||||
|
||||
try:
|
||||
from tower_cli.cli.transfer.send import Sender
|
||||
from tower_cli.utils.exceptions import TowerCLIError
|
||||
|
||||
from tower_cli.conf import settings
|
||||
TOWER_CLI_HAS_EXPORT = True
|
||||
except ImportError:
|
||||
TOWER_CLI_HAS_EXPORT = False
|
||||
|
||||
|
||||
def main():
|
||||
argument_spec = dict(
|
||||
assets=dict(required=False),
|
||||
files=dict(required=False, default=[], type='list'),
|
||||
prevent=dict(required=False, default=[], type='list'),
|
||||
password_management=dict(required=False, default='default', choices=['default', 'random']),
|
||||
)
|
||||
|
||||
module = TowerModule(argument_spec=argument_spec, supports_check_mode=False)
|
||||
|
||||
if not HAS_TOWER_CLI:
|
||||
module.fail_json(msg='ansible-tower-cli required for this module')
|
||||
|
||||
if not TOWER_CLI_HAS_EXPORT:
|
||||
module.fail_json(msg='ansible-tower-cli version does not support export')
|
||||
|
||||
assets = module.params.get('assets')
|
||||
prevent = module.params.get('prevent')
|
||||
password_management = module.params.get('password_management')
|
||||
files = module.params.get('files')
|
||||
|
||||
result = dict(
|
||||
changed=False,
|
||||
msg='',
|
||||
output='',
|
||||
)
|
||||
|
||||
if not assets and not files:
|
||||
result['msg'] = "Assets or files must be specified"
|
||||
module.fail_json(**result)
|
||||
|
||||
path = None
|
||||
if assets:
|
||||
# We got assets so we need to dump this out to a temp file and append that to files
|
||||
handle, path = mkstemp(prefix='', suffix='', dir='')
|
||||
with open(path, 'w') as f:
|
||||
f.write(assets)
|
||||
files.append(path)
|
||||
|
||||
tower_auth = tower_auth_config(module)
|
||||
failed = False
|
||||
with settings.runtime_values(**tower_auth):
|
||||
try:
|
||||
sender = Sender(no_color=False)
|
||||
old_stdout = sys.stdout
|
||||
sys.stdout = captured_stdout = StringIO()
|
||||
try:
|
||||
sender.send(files, prevent, password_management)
|
||||
except TypeError as e:
|
||||
# Newer versions of TowerCLI require 4 parameters
|
||||
sender.send(files, prevent, [], password_management)
|
||||
|
||||
if sender.error_messages > 0:
|
||||
failed = True
|
||||
result['msg'] = "Transfer Failed with %d errors" % sender.error_messages
|
||||
if sender.changed_messages > 0:
|
||||
result['changed'] = True
|
||||
except TowerCLIError as e:
|
||||
result['msg'] = e
|
||||
failed = True
|
||||
finally:
|
||||
if path is not None:
|
||||
os.remove(path)
|
||||
result['output'] = captured_stdout.getvalue().split("\n")
|
||||
sys.stdout = old_stdout
|
||||
|
||||
# Return stdout so that module returns will work
|
||||
if failed:
|
||||
module.fail_json(**result)
|
||||
else:
|
||||
module.exit_json(**result)
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
2
test/integration/targets/tower_send/aliases
Normal file
2
test/integration/targets/tower_send/aliases
Normal file
|
@ -0,0 +1,2 @@
|
|||
cloud/tower
|
||||
shippable/tower/group1
|
75
test/integration/targets/tower_send/tasks/main.yml
Normal file
75
test/integration/targets/tower_send/tasks/main.yml
Normal file
|
@ -0,0 +1,75 @@
|
|||
- name: Test no parameters
|
||||
tower_send:
|
||||
register: result
|
||||
ignore_errors: yes
|
||||
|
||||
- assert:
|
||||
that:
|
||||
- "result is failed"
|
||||
|
||||
- name: Create user json
|
||||
set_fact:
|
||||
user:
|
||||
- username: "jowestco"
|
||||
first_name: "John"
|
||||
last_name: "Westcott"
|
||||
asset_type: "user"
|
||||
email: "john.westcott.iv@redhat.com"
|
||||
|
||||
- name: Test a new import of asset
|
||||
tower_send:
|
||||
assets: "{{ user | to_json() }}"
|
||||
register: result
|
||||
|
||||
- assert:
|
||||
that:
|
||||
- "result is changed"
|
||||
|
||||
- name: Test an existing import of asset
|
||||
tower_send:
|
||||
assets: "{{ user | to_json() }}"
|
||||
register: result
|
||||
|
||||
- assert:
|
||||
that:
|
||||
- "result is successful"
|
||||
- "result is not changed"
|
||||
|
||||
- name: Change an existing asset
|
||||
tower_send:
|
||||
assets: "{{ user | combine({'last_name': 'Westcott IV'}) | to_json() }}"
|
||||
register: result
|
||||
|
||||
- assert:
|
||||
that:
|
||||
- "result is changed"
|
||||
|
||||
- name: Create organization json
|
||||
set_fact:
|
||||
organization:
|
||||
- asset_type: organization
|
||||
name: "Red Hat"
|
||||
|
||||
- name: Create temp file
|
||||
tempfile:
|
||||
state: file
|
||||
register: my_temp_file
|
||||
|
||||
- name: Drop down a file to import
|
||||
copy:
|
||||
dest: "{{ my_temp_file.path }}"
|
||||
content: "{{ organization | to_nice_json() }}"
|
||||
|
||||
- name: Create org via files
|
||||
tower_send:
|
||||
files: "{{ my_temp_file.path }}"
|
||||
register: result
|
||||
|
||||
- assert:
|
||||
that:
|
||||
- "result is changed"
|
||||
|
||||
- name: Remove Temp File
|
||||
file:
|
||||
path: "{{ my_temp_file.path }}"
|
||||
state: absent
|
Loading…
Reference in a new issue