mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
Initial commit for Pure Storage Ansible module (#25386)
* Initial commit for Pure Storage Ansible module * Initial commit for Pure Storage Ansible module * Initial commit for Pure Storage Ansible module * Fix import issues as required by post-2.2 * Move last import to top * Follow suggestions and only implement one module per PR Fix documentation changes requested * Documentation and formatting changes
This commit is contained in:
parent
1267567556
commit
937d7993cc
6 changed files with 330 additions and 0 deletions
|
@ -203,6 +203,11 @@ Ansible Changes By Release
|
||||||
* gcp_healthcheck
|
* gcp_healthcheck
|
||||||
* gcp_target_proxy
|
* gcp_target_proxy
|
||||||
* gcp_url_map
|
* gcp_url_map
|
||||||
|
- purestorage
|
||||||
|
* purefa_host
|
||||||
|
* purefa_volume
|
||||||
|
* purefa_hg
|
||||||
|
* purefa_pg
|
||||||
- rundeck
|
- rundeck
|
||||||
* rundeck_acl_policy
|
* rundeck_acl_policy
|
||||||
* rundeck_project
|
* rundeck_project
|
||||||
|
|
|
@ -37,6 +37,7 @@ The following is a list of module_utils files and a general description. The mod
|
||||||
- openstack.py - Utilities for modules that work with Openstack instances.
|
- openstack.py - Utilities for modules that work with Openstack instances.
|
||||||
- openswitch.py - Definitions and helper functions for modules that manage OpenSwitch devices
|
- openswitch.py - Definitions and helper functions for modules that manage OpenSwitch devices
|
||||||
- powershell.ps1 - Utilities for working with Microsoft Windows clients
|
- powershell.ps1 - Utilities for working with Microsoft Windows clients
|
||||||
|
- pure.py - Functions and utilities for modules that work with the Pure Storage storage platforms.
|
||||||
- pycompat24.py - Exception workaround for Python 2.4.
|
- pycompat24.py - Exception workaround for Python 2.4.
|
||||||
- rax.py - Definitions and helper functions for modules that work with Rackspace resources.
|
- rax.py - Definitions and helper functions for modules that work with Rackspace resources.
|
||||||
- redhat.py - Functions for modules that manage Red Hat Network registration and subscriptions
|
- redhat.py - Functions for modules that manage Red Hat Network registration and subscriptions
|
||||||
|
|
77
lib/ansible/module_utils/pure.py
Normal file
77
lib/ansible/module_utils/pure.py
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# This code is part of Ansible, but is an independent component.
|
||||||
|
# This particular file snippet, and this file snippet only, is BSD licensed.
|
||||||
|
# Modules you write using this snippet, which is embedded dynamically by Ansible
|
||||||
|
# still belong to the author of the module, and may assign their own license
|
||||||
|
# to the complete work.
|
||||||
|
#
|
||||||
|
# Copyright (c), Simon Dodsley <simon@purestorage.com>,2017
|
||||||
|
# All rights reserved.
|
||||||
|
#
|
||||||
|
# Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
# are permitted provided that the following conditions are met:
|
||||||
|
#
|
||||||
|
# * Redistributions of source code must retain the above copyright
|
||||||
|
# notice, this list of conditions and the following disclaimer.
|
||||||
|
# * Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
# this list of conditions and the following disclaimer in the documentation
|
||||||
|
# and/or other materials provided with the distribution.
|
||||||
|
#
|
||||||
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||||
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
||||||
|
# IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
|
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||||||
|
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||||
|
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
|
||||||
|
# USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
HAS_PURESTORAGE = True
|
||||||
|
try:
|
||||||
|
from purestorage import purestorage
|
||||||
|
except ImportError:
|
||||||
|
HAS_PURESTORAGE = False
|
||||||
|
|
||||||
|
from functools import wraps
|
||||||
|
from os import environ
|
||||||
|
from os import path
|
||||||
|
import platform
|
||||||
|
|
||||||
|
VERSION = 1.0
|
||||||
|
USER_AGENT_BASE = 'Ansible'
|
||||||
|
|
||||||
|
|
||||||
|
def get_system(module):
|
||||||
|
"""Return System Object or Fail"""
|
||||||
|
user_agent = '%(base)s %(class)s/%(version)s (%(platform)s)' % {
|
||||||
|
'base': USER_AGENT_BASE,
|
||||||
|
'class': __name__,
|
||||||
|
'version': VERSION,
|
||||||
|
'platform': platform.platform()
|
||||||
|
}
|
||||||
|
array_name = module.params['fa_url']
|
||||||
|
api = module.params['api_token']
|
||||||
|
|
||||||
|
if array_name and api:
|
||||||
|
system = purestorage.FlashArray(array_name, api_token=api, user_agent=user_agent)
|
||||||
|
elif environ.get('PUREFA_URL') and environ.get('PUREFA_API'):
|
||||||
|
system = purestorage.FlashArray(environ.get('PUREFA_URL'), api_token=(environ.get('PUREFA_API')), user_agent=user_agent)
|
||||||
|
else:
|
||||||
|
module.fail_json(msg="You must set PUREFA_URL and PUREFA_API environment variables or the fa_url and api_token module arguments")
|
||||||
|
|
||||||
|
try:
|
||||||
|
system.get()
|
||||||
|
except Exception:
|
||||||
|
module.fail_json(msg="Pure Storage FlashArray authentication failed. Check your credentials")
|
||||||
|
return system
|
||||||
|
|
||||||
|
|
||||||
|
def purefa_argument_spec():
|
||||||
|
"""Return standard base dictionary used for the argument_spec argument in AnsibleModule"""
|
||||||
|
|
||||||
|
return dict(
|
||||||
|
fa_url=dict(),
|
||||||
|
api_token=dict(no_log=True),
|
||||||
|
)
|
0
lib/ansible/modules/storage/purestorage/__init__.py
Normal file
0
lib/ansible/modules/storage/purestorage/__init__.py
Normal file
207
lib/ansible/modules/storage/purestorage/purefa_host.py
Normal file
207
lib/ansible/modules/storage/purestorage/purefa_host.py
Normal file
|
@ -0,0 +1,207 @@
|
||||||
|
#!/usr/bin/python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# (c) 2017, Simon Dodsley (simon@purestorage.com)
|
||||||
|
#
|
||||||
|
# This file is part of Ansible
|
||||||
|
#
|
||||||
|
# Ansible is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# Ansible is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
ANSIBLE_METADATA = {'metadata_version': '1.0',
|
||||||
|
'status': ['preview'],
|
||||||
|
'supported_by': 'community'}
|
||||||
|
|
||||||
|
|
||||||
|
DOCUMENTATION = '''
|
||||||
|
---
|
||||||
|
module: purefa_host
|
||||||
|
version_added: "2.4"
|
||||||
|
short_description: Create, Delete and Modify Hosts on Pure Storage FlashArray
|
||||||
|
description:
|
||||||
|
- This module creates, deletes or modifies hosts on Pure Storage FlashArray.
|
||||||
|
author: Simon Dodsley (@simondodsley)
|
||||||
|
options:
|
||||||
|
host:
|
||||||
|
description:
|
||||||
|
- Host Name
|
||||||
|
required: true
|
||||||
|
state:
|
||||||
|
description:
|
||||||
|
- Creates host.
|
||||||
|
- When removing host all connected volumes will be disconnected.
|
||||||
|
required: false
|
||||||
|
default: present
|
||||||
|
choices: [ "present", "absent" ]
|
||||||
|
protocol:
|
||||||
|
description:
|
||||||
|
- Defines the host connection protocol for volumes.
|
||||||
|
required: false
|
||||||
|
default: iscsi
|
||||||
|
choices: [ "iscsi", "fc" ]
|
||||||
|
wwns:
|
||||||
|
description:
|
||||||
|
- List of wwns of the host if protocol is fc
|
||||||
|
required: false
|
||||||
|
iqn:
|
||||||
|
description:
|
||||||
|
- List of IQNs of the host if protocol is iscsi
|
||||||
|
required: false
|
||||||
|
volume:
|
||||||
|
description:
|
||||||
|
- Volume name to map to the host
|
||||||
|
required: false
|
||||||
|
extends_documentation_fragment:
|
||||||
|
- purestorage
|
||||||
|
'''
|
||||||
|
|
||||||
|
EXAMPLES = '''
|
||||||
|
- name: Create new new host
|
||||||
|
purefa_host:
|
||||||
|
host: foo
|
||||||
|
fa_url: 10.10.10.2
|
||||||
|
api_token: e31060a7-21fc-e277-6240-25983c6c4592
|
||||||
|
|
||||||
|
- name: Delete host
|
||||||
|
purefa_host:
|
||||||
|
host: foo
|
||||||
|
state: absent
|
||||||
|
fa_url: 10.10.10.2
|
||||||
|
api_token: e31060a7-21fc-e277-6240-25983c6c4592
|
||||||
|
|
||||||
|
- name: Make sure host bar is available with wwn ports
|
||||||
|
purefa_host:
|
||||||
|
host: bar
|
||||||
|
protocol: fc
|
||||||
|
wwns:
|
||||||
|
- "00:00:00:00:00:00:00"
|
||||||
|
- "11:11:11:11:11:11:11"
|
||||||
|
fa_url: 10.10.10.2
|
||||||
|
api_token: e31060a7-21fc-e277-6240-25983c6c4592
|
||||||
|
|
||||||
|
- name: Make sure host bar is available with iSCSI ports
|
||||||
|
purefa_host:
|
||||||
|
host: bar
|
||||||
|
protocol: iscsi
|
||||||
|
iqn:
|
||||||
|
- "iqn.1994-05.com.redhat:7d366003913"
|
||||||
|
fa_url: 10.10.10.2
|
||||||
|
api_token: e31060a7-21fc-e277-6240-25983c6c4592
|
||||||
|
|
||||||
|
- name: Map host foo to volume bar
|
||||||
|
purefa_host:
|
||||||
|
host: foo
|
||||||
|
volume: bar
|
||||||
|
fa_url: 10.10.10.2
|
||||||
|
api_token: e31060a7-21fc-e277-6240-25983c6c4592
|
||||||
|
'''
|
||||||
|
|
||||||
|
RETURN = '''
|
||||||
|
'''
|
||||||
|
|
||||||
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
from ansible.module_utils.pure import get_system, purefa_argument_spec
|
||||||
|
|
||||||
|
|
||||||
|
HAS_PURESTORAGE = True
|
||||||
|
try:
|
||||||
|
from purestorage import purestorage
|
||||||
|
except ImportError:
|
||||||
|
HAS_PURESTORAGE = False
|
||||||
|
|
||||||
|
|
||||||
|
def get_host(module, array):
|
||||||
|
|
||||||
|
host = None
|
||||||
|
|
||||||
|
for h in array.list_hosts():
|
||||||
|
if h["name"] == module.params['host']:
|
||||||
|
host = h
|
||||||
|
break
|
||||||
|
|
||||||
|
return host
|
||||||
|
|
||||||
|
|
||||||
|
def make_host(module, array):
|
||||||
|
|
||||||
|
changed = True
|
||||||
|
|
||||||
|
if not module.check_mode:
|
||||||
|
host = array.create_host(module.params['host'])
|
||||||
|
if module.params['protocol'] == 'iscsi':
|
||||||
|
if module.params['iqn']:
|
||||||
|
array.set_host(module.params['host'], addiqnlist=module.params['iqn'])
|
||||||
|
if module.params['protocol'] == 'fc':
|
||||||
|
if module.params['wwns']:
|
||||||
|
array.set_host(module.params['host'], addwwnlist=module.params['wwns'])
|
||||||
|
if module.params['volume']:
|
||||||
|
array.connect_host(module.params['host'], module.params['volume'])
|
||||||
|
module.exit_json(changed=changed)
|
||||||
|
|
||||||
|
|
||||||
|
def update_host(module, array):
|
||||||
|
changed = False
|
||||||
|
host = module.params['host']
|
||||||
|
module.exit_json(changed=changed)
|
||||||
|
|
||||||
|
|
||||||
|
def delete_host(module, array):
|
||||||
|
changed = True
|
||||||
|
if not module.check_mode:
|
||||||
|
for vol in array.list_host_connections(module.params['host']):
|
||||||
|
array.disconnect_host(module.params['host'], vol["vol"])
|
||||||
|
array.delete_host(module.params['host'])
|
||||||
|
module.exit_json(changed=changed)
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
argument_spec = purefa_argument_spec()
|
||||||
|
argument_spec.update(
|
||||||
|
dict(
|
||||||
|
host=dict(required=True),
|
||||||
|
state=dict(default='present', choices=['present', 'absent']),
|
||||||
|
protocol=dict(default='iscsi', choices=['iscsi', 'fc']),
|
||||||
|
iqn=dict(type='list'),
|
||||||
|
wwns=dict(type='list'),
|
||||||
|
volume=dict()
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
module = AnsibleModule(argument_spec, supports_check_mode=True)
|
||||||
|
|
||||||
|
if not HAS_PURESTORAGE:
|
||||||
|
module.fail_json(msg='purestorage sdk is required for this module in host')
|
||||||
|
|
||||||
|
state = module.params['state']
|
||||||
|
protocol = module.params['protocol']
|
||||||
|
array = get_system(module)
|
||||||
|
host = get_host(module, array)
|
||||||
|
|
||||||
|
if module.params['volume']:
|
||||||
|
try:
|
||||||
|
array.get_volume(module.params['volume'])
|
||||||
|
except:
|
||||||
|
module.fail_json(msg='Volume {} not found'.format(module.params['volume']))
|
||||||
|
|
||||||
|
if host and state == 'present':
|
||||||
|
update_host(module, array)
|
||||||
|
elif host and state == 'absent':
|
||||||
|
delete_host(module, array)
|
||||||
|
elif host is None and state == 'absent':
|
||||||
|
module.exit_json(changed=False)
|
||||||
|
else:
|
||||||
|
make_host(module, array)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
40
lib/ansible/utils/module_docs_fragments/purestorage.py
Normal file
40
lib/ansible/utils/module_docs_fragments/purestorage.py
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
#
|
||||||
|
# (c) 2017, Simon Dodsley <simon@purestorage.com>
|
||||||
|
#
|
||||||
|
# This file is part of Ansible
|
||||||
|
#
|
||||||
|
# Ansible is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# Ansible is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
|
class ModuleDocFragment(object):
|
||||||
|
|
||||||
|
# Standard Pure Storage documentation fragment
|
||||||
|
DOCUMENTATION = '''
|
||||||
|
options:
|
||||||
|
fa_url:
|
||||||
|
description:
|
||||||
|
- FlashArray management IPv4 address or Hostname.
|
||||||
|
required: true
|
||||||
|
api_token:
|
||||||
|
description:
|
||||||
|
- FlashArray API token for admin privilaged user.
|
||||||
|
required: true
|
||||||
|
notes:
|
||||||
|
- This module requires purestorage python library
|
||||||
|
- You must set C(PUREFA_URL) and C(PUREFA_API) environment variables
|
||||||
|
if I(url) and I(api_token) arguments are not passed to the module directly
|
||||||
|
requirements:
|
||||||
|
- "python >= 2.7"
|
||||||
|
- purestorage
|
||||||
|
'''
|
Loading…
Reference in a new issue