community.general/lib/ansible/modules/cloud/vmware/vsphere_file.py

356 lines
12 KiB
Python
Raw Normal View History

#!/usr/bin/python
# -*- coding: utf-8 -*-
# Copyright: (c) 2017, Dag Wieers (@dagwieers) <dag@wieers.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 = r'''
---
module: vsphere_file
short_description: Manage files on a vCenter datastore
description:
- Manage files on a vCenter datastore.
version_added: '2.8'
author:
- Dag Wieers (@dagwieers)
options:
host:
description:
- The vCenter server on which the datastore is available.
type: str
required: true
aliases: [ hostname ]
username:
description:
- The user name to authenticate on the vCenter server.
type: str
required: true
password:
description:
- The password to authenticate on the vCenter server.
type: str
required: true
datacenter:
description:
- The datacenter on the vCenter server that holds the datastore.
type: str
required: true
datastore:
description:
- The datastore on the vCenter server to push files to.
type: str
required: true
path:
description:
- The file or directory on the datastore on the vCenter server.
type: str
required: true
aliases: [ dest ]
validate_certs:
description:
- If C(no), SSL certificates will not be validated. This should only be
set to C(no) when no other option exists.
type: bool
default: yes
timeout:
description:
- The timeout in seconds for the upload to the datastore.
type: int
default: 10
state:
description:
- The state of or the action on the provided path.
- If C(absent), the file will be removed.
- If C(directory), the directory will be created.
- If C(file), more information of the (existing) file will be returned.
- If C(touch), an empty file will be created if the path does not exist.
type: str
choices: [ absent, directory, file, touch ]
default: file
notes:
- The vSphere folder API does not allow to remove directory objects.
'''
EXAMPLES = r'''
- name: Create an empty file on a datastore
vsphere_file:
host: '{{ vhost }}'
username: '{{ vuser }}'
password: '{{ vpass }}'
datacenter: DC1 Someplace
datastore: datastore1
path: some/remote/file
state: touch
delegate_to: localhost
- name: Create a directory on a datastore
vsphere_copy:
host: '{{ vhost }}'
username: '{{ vuser }}'
password: '{{ vpass }}'
src: /other/local/file
datacenter: DC2 Someplace
datastore: datastore2
path: other/remote/file
state: directory
delegate_to: localhost
- name: Query a file on a datastore
vsphere_file:
host: '{{ vhost }}'
username: '{{ vuser }}'
password: '{{ vpass }}'
datacenter: DC1 Someplace
datastore: datastore1
path: some/remote/file
state: touch
delegate_to: localhost
ignore_errors: yes
- name: Delete a file on a datastore
vsphere_copy:
host: '{{ vhost }}'
username: '{{ vuser }}'
password: '{{ vpass }}'
datacenter: DC2 Someplace
datastore: datastore2
path: other/remote/file
state: absent
delegate_to: localhost
'''
RETURN = r'''
'''
import socket
import sys
from ansible.module_utils.basic import AnsibleModule
from ansible.module_utils.six import PY2
from ansible.module_utils.six.moves.urllib.error import HTTPError
from ansible.module_utils.six.moves.urllib.parse import quote, urlencode
from ansible.module_utils.urls import open_url
from ansible.module_utils._text import to_native
def vmware_path(datastore, datacenter, path):
''' Constructs a URL path that VSphere accepts reliably '''
path = '/folder/{path}'.format(path=quote(path.strip('/')))
# Due to a software bug in vSphere, it fails to handle ampersand in datacenter names
# The solution is to do what vSphere does (when browsing) and double-encode ampersands, maybe others ?
datacenter = datacenter.replace('&', '%26')
if not path.startswith('/'):
path = '/' + path
params = dict(dsName=datastore)
if datacenter:
params['dcPath'] = datacenter
return '{0}?{1}'.format(path, urlencode(params))
def main():
module = AnsibleModule(
argument_spec=dict(
host=dict(type='str', required=True, aliases=['hostname']),
username=dict(type='str', required=True),
password=dict(type='str', required=True, no_log=True),
datacenter=dict(type='str', required=True),
datastore=dict(type='str', required=True),
path=dict(type='str', required=True, aliases=['dest']),
state=dict(type='str', default='file', choices=['absent', 'directory', 'file', 'touch']),
timeout=dict(type='int', default=10),
validate_certs=dict(type='bool', default=True),
),
supports_check_mode=True,
)
host = module.params.get('host')
username = module.params.get('username')
password = module.params.get('password')
src = module.params.get('src')
datacenter = module.params.get('datacenter')
datastore = module.params.get('datastore')
path = module.params.get('path')
validate_certs = module.params.get('validate_certs')
timeout = module.params.get('timeout')
state = module.params.get('state')
remote_path = vmware_path(datastore, datacenter, path)
url = 'https://%s%s' % (host, remote_path)
result = dict(
path=path,
size=None,
state=state,
status=None,
url=url,
)
# Check if the file/directory exists
try:
r = open_url(url, method='HEAD', timeout=timeout,
url_username=username, url_password=password,
validate_certs=validate_certs, force_basic_auth=True)
except HTTPError as e:
r = e
except socket.error as e:
module.fail_json(msg=to_native(e), errno=e[0], reason=to_native(e), **result)
except Exception as e:
module.fail_json(msg=to_native(e), errno=dir(e), reason=to_native(e), **result)
if PY2:
sys.exc_clear() # Avoid false positive traceback in fail_json() on Python 2
status = r.getcode()
if status == 200:
exists = True
result['size'] = int(r.headers.get('content-length', None))
elif status == 404:
exists = False
else:
result['reason'] = r.msg
result['status'] = status
module.fail_json(msg="Failed to query for file '%s'" % path, errno=None, headers=dict(r.headers), **result)
if state == 'absent':
if not exists:
module.exit_json(changed=False, **result)
if module.check_mode:
result['reason'] = 'No Content'
result['status'] = 204
else:
try:
r = open_url(url, method='DELETE', timeout=timeout,
url_username=username, url_password=password,
validate_certs=validate_certs, force_basic_auth=True)
except HTTPError as e:
r = e
except socket.error as e:
module.fail_json(msg=to_native(e), errno=e[0], reason=to_native(e), **result)
except Exception as e:
module.fail_json(msg=to_native(e), errno=e[0], reason=to_native(e), **result)
if PY2:
sys.exc_clear() # Avoid false positive traceback in fail_json() on Python 2
result['reason'] = r.msg
result['status'] = r.getcode()
if result['status'] == 405:
result['state'] = 'directory'
module.fail_json(msg='Directories cannot be removed with this module', errno=None, headers=dict(r.headers), **result)
elif result['status'] != 204:
module.fail_json(msg="Failed to remove '%s'" % path, errno=None, headers=dict(r.headers), **result)
result['size'] = None
module.exit_json(changed=True, **result)
# NOTE: Creating a file in a non-existing directory, then remove the file
elif state == 'directory':
if exists:
module.exit_json(changed=False, **result)
if module.check_mode:
result['reason'] = 'Created'
result['status'] = 201
else:
# Create a temporary file in the new directory
remote_path = vmware_path(datastore, datacenter, path + '/foobar.tmp')
temp_url = 'https://%s%s' % (host, remote_path)
try:
r = open_url(temp_url, method='PUT', timeout=timeout,
url_username=username, url_password=password,
validate_certs=validate_certs, force_basic_auth=True)
except HTTPError as e:
r = e
except socket.error as e:
module.fail_json(msg=to_native(e), errno=e[0], reason=to_native(e), **result)
except Exception as e:
module.fail_json(msg=to_native(e), errno=e[0], reason=to_native(e), **result)
if PY2:
sys.exc_clear() # Avoid false positive traceback in fail_json() on Python 2
result['reason'] = r.msg
result['status'] = r.getcode()
if result['status'] != 201:
result['url'] = temp_url
module.fail_json(msg='Failed to create temporary file', errno=None, headers=dict(r.headers), **result)
try:
r = open_url(temp_url, method='DELETE', timeout=timeout,
url_username=username, url_password=password,
validate_certs=validate_certs, force_basic_auth=True)
except HTTPError as e:
r = e
except socket.error as e:
module.fail_json(msg=to_native(e), errno=e[0], reason=to_native(e), **result)
except Exception as e:
module.fail_json(msg=to_native(e), errno=e[0], reason=to_native(e), **result)
if PY2:
sys.exc_clear() # Avoid false positive traceback in fail_json() on Python 2
status = r.getcode()
if status != 204:
result['reason'] = r.msg
result['status'] = status
module.warn('Failed to remove temporary file ({reason})'.format(**result))
module.exit_json(changed=True, **result)
elif state == 'file':
if not exists:
result['state'] = 'absent'
result['status'] = status
module.fail_json(msg="File '%s' is absent, cannot continue" % path, **result)
result['status'] = status
module.exit_json(changed=False, **result)
elif state == 'touch':
if exists:
result['state'] = 'file'
module.exit_json(changed=False, **result)
if module.check_mode:
result['reason'] = 'Created'
result['status'] = 201
else:
try:
r = open_url(url, method='PUT', timeout=timeout,
url_username=username, url_password=password,
validate_certs=validate_certs, force_basic_auth=True)
except HTTPError as e:
r = e
except socket.error as e:
module.fail_json(msg=to_native(e), errno=e[0], reason=to_native(e), **result)
except Exception as e:
module.fail_json(msg=to_native(e), errno=e[0], reason=to_native(e), **result)
if PY2:
sys.exc_clear() # Avoid false positive traceback in fail_json() on Python 2
result['reason'] = r.msg
result['status'] = r.getcode()
if result['status'] != 201:
module.fail_json(msg="Failed to touch '%s'" % path, errno=None, headers=dict(r.headers), **result)
result['size'] = 0
result['state'] = 'file'
module.exit_json(changed=True, **result)
if __name__ == '__main__':
main()