From 70fd1ec1309910926f40a7cd733dbe11ac7ecf2b Mon Sep 17 00:00:00 2001 From: Michael Price Date: Tue, 28 Aug 2018 12:38:43 -0500 Subject: [PATCH] Define module for NetApp E-Series iSCSI targets (#40632) Create a new module for managing E-Series iSCSI targets. --- .../storage/netapp/netapp_e_iscsi_target.py | 295 ++++++++++++++++++ .../netapp_eseries_iscsi_target/aliases | 10 + .../tasks/main.yml | 1 + .../netapp_eseries_iscsi_target/tasks/run.yml | 68 ++++ .../netapp/test_netapp_e_iscsi_target.py | 134 ++++++++ 5 files changed, 508 insertions(+) create mode 100644 lib/ansible/modules/storage/netapp/netapp_e_iscsi_target.py create mode 100644 test/integration/targets/netapp_eseries_iscsi_target/aliases create mode 100644 test/integration/targets/netapp_eseries_iscsi_target/tasks/main.yml create mode 100644 test/integration/targets/netapp_eseries_iscsi_target/tasks/run.yml create mode 100644 test/units/modules/storage/netapp/test_netapp_e_iscsi_target.py diff --git a/lib/ansible/modules/storage/netapp/netapp_e_iscsi_target.py b/lib/ansible/modules/storage/netapp/netapp_e_iscsi_target.py new file mode 100644 index 0000000000..ad4975e5eb --- /dev/null +++ b/lib/ansible/modules/storage/netapp/netapp_e_iscsi_target.py @@ -0,0 +1,295 @@ +#!/usr/bin/python + +# (c) 2018, NetApp, Inc +# 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: netapp_e_iscsi_target +short_description: NetApp E-Series manage iSCSI target configuration +description: + - Configure the settings of an E-Series iSCSI target +version_added: '2.7' +author: Michael Price (@lmprice) +extends_documentation_fragment: + - netapp.eseries +options: + name: + description: + - The name/alias to assign to the iSCSI target. + - This alias is often used by the initiator software in order to make an iSCSI target easier to identify. + aliases: + - alias + ping: + description: + - Enable ICMP ping responses from the configured iSCSI ports. + type: bool + default: yes + chap_secret: + description: + - Enable Challenge-Handshake Authentication Protocol (CHAP), utilizing this value as the password. + - When this value is specified, we will always trigger an update (changed=True). We have no way of verifying + whether or not the password has changed. + - The chap secret may only use ascii characters with values between 32 and 126 decimal. + - The chap secret must be no less than 12 characters, but no more than 16 characters in length. + aliases: + - chap + - password + unnamed_discovery: + description: + - When an initiator initiates a discovery session to an initiator port, it is considered an unnamed + discovery session if the iSCSI target iqn is not specified in the request. + - This option may be disabled to increase security if desired. + type: bool + default: yes + log_path: + description: + - A local path (on the Ansible controller), to a file to be used for debug logging. + required: no +notes: + - Check mode is supported. + - Some of the settings are dependent on the settings applied to the iSCSI interfaces. These can be configured using + M(netapp_e_iscsi_interface). + - This module requires a Web Services API version of >= 1.3. +""" + +EXAMPLES = """ + - name: Enable ping responses and unnamed discovery sessions for all iSCSI ports + netapp_e_iscsi_target: + api_url: "https://localhost:8443/devmgr/v2" + api_username: admin + api_password: myPassword + ssid: "1" + validate_certs: no + name: myTarget + ping: yes + unnamed_discovery: yes + + - name: Set the target alias and the CHAP secret + netapp_e_iscsi_target: + ssid: "{{ ssid }}" + api_url: "{{ netapp_api_url }}" + api_username: "{{ netapp_api_username }}" + api_password: "{{ netapp_api_password }}" + name: myTarget + chap: password1234 +""" + +RETURN = """ +msg: + description: Success message + returned: on success + type: string + sample: The iSCSI target settings have been updated. +alias: + description: + - The alias assigned to the iSCSI target. + returned: on success + sample: myArray + type: string +iqn: + description: + - The iqn (iSCSI Qualified Name), assigned to the iSCSI target. + returned: on success + sample: iqn.1992-08.com.netapp:2800.000a132000b006d2000000005a0e8f45 + type: string +""" +import json +import logging +from pprint import pformat + +from ansible.module_utils.basic import AnsibleModule +from ansible.module_utils.netapp import request, eseries_host_argument_spec +from ansible.module_utils._text import to_native + +HEADERS = { + "Content-Type": "application/json", + "Accept": "application/json", +} + + +class IscsiTarget(object): + def __init__(self): + argument_spec = eseries_host_argument_spec() + argument_spec.update(dict( + name=dict(type='str', required=False, aliases=['alias']), + ping=dict(type='bool', required=False, default=True), + chap_secret=dict(type='str', required=False, aliases=['chap', 'password'], no_log=True), + unnamed_discovery=dict(type='bool', required=False, default=True), + log_path=dict(type='str', required=False), + )) + + self.module = AnsibleModule(argument_spec=argument_spec, supports_check_mode=True, ) + args = self.module.params + + self.name = args['name'] + self.ping = args['ping'] + self.chap_secret = args['chap_secret'] + self.unnamed_discovery = args['unnamed_discovery'] + + self.ssid = args['ssid'] + self.url = args['api_url'] + self.creds = dict(url_password=args['api_password'], + validate_certs=args['validate_certs'], + url_username=args['api_username'], ) + + self.check_mode = self.module.check_mode + self.post_body = dict() + self.controllers = list() + + log_path = args['log_path'] + + # logging setup + self._logger = logging.getLogger(self.__class__.__name__) + + if log_path: + logging.basicConfig( + level=logging.DEBUG, filename=log_path, filemode='w', + format='%(relativeCreated)dms %(levelname)s %(module)s.%(funcName)s:%(lineno)d\n %(message)s') + + if not self.url.endswith('/'): + self.url += '/' + self._logger.info(self.chap_secret) + if self.chap_secret is not None: + if len(self.chap_secret) < 12 or len(self.chap_secret) > 16: + self.module.fail_json(msg="The provided CHAP secret is not valid, it must be between 12 and 16" + " characters in length.") + + for c in self.chap_secret: + ordinal = ord(c) + if ordinal < 32 or ordinal > 126: + self.module.fail_json(msg="The provided CHAP secret is not valid, it may only utilize ascii" + " characters with decimal values between 32 and 126.") + + @property + def target(self): + """Provide information on the iSCSI Target configuration + + Sample: + { + 'alias': 'myCustomName', + 'ping': True, + 'unnamed_discovery': True, + 'chap': False, + 'iqn': 'iqn.1992-08.com.netapp:2800.000a132000b006d2000000005a0e8f45', + } + """ + target = dict() + try: + (rc, data) = request(self.url + 'storage-systems/%s/graph/xpath-filter?query=/storagePoolBundle/target' + % self.ssid, headers=HEADERS, **self.creds) + # This likely isn't an iSCSI-enabled system + if not data: + self.module.fail_json( + msg="This storage-system doesn't appear to have iSCSI interfaces. Array Id [%s]." % (self.ssid)) + + data = data[0] + + chap = any( + [auth for auth in data['configuredAuthMethods']['authMethodData'] if auth['authMethod'] == 'chap']) + + target.update(dict(alias=data['alias']['iscsiAlias'], + iqn=data['nodeName']['iscsiNodeName'], + chap=chap)) + + (rc, data) = request(self.url + 'storage-systems/%s/graph/xpath-filter?query=/sa/iscsiEntityData' + % self.ssid, headers=HEADERS, **self.creds) + + data = data[0] + target.update(dict(ping=data['icmpPingResponseEnabled'], + unnamed_discovery=data['unnamedDiscoverySessionsEnabled'])) + + except Exception as err: + self.module.fail_json( + msg="Failed to retrieve the iSCSI target information. Array Id [%s]. Error [%s]." + % (self.ssid, to_native(err))) + + return target + + def apply_iscsi_settings(self): + """Update the iSCSI target alias and CHAP settings""" + update = False + target = self.target + + body = dict() + + if self.name is not None and self.name != target['alias']: + update = True + body['alias'] = self.name + + # If the CHAP secret was provided, we trigger an update. + if self.chap_secret is not None: + update = True + body.update(dict(enableChapAuthentication=True, + chapSecret=self.chap_secret)) + # If no secret was provided, then we disable chap + elif target['chap']: + update = True + body.update(dict(enableChapAuthentication=False)) + + self._logger.info(pformat(body)) + + if update and not self.check_mode: + try: + request(self.url + 'storage-systems/%s/iscsi/target-settings' % self.ssid, method='POST', + data=json.dumps(body), headers=HEADERS, **self.creds) + except Exception as err: + self.module.fail_json( + msg="Failed to update the iSCSI target settings. Array Id [%s]. Error [%s]." + % (self.ssid, to_native(err))) + + return update + + def apply_target_changes(self): + update = False + target = self.target + + body = dict() + + if self.ping != target['ping']: + update = True + body['icmpPingResponseEnabled'] = self.ping + + if self.unnamed_discovery != target['unnamed_discovery']: + update = True + body['unnamedDiscoverySessionsEnabled'] = self.unnamed_discovery + + self._logger.info(pformat(body)) + if update and not self.check_mode: + try: + request(self.url + 'storage-systems/%s/iscsi/entity' % self.ssid, method='POST', + data=json.dumps(body), timeout=60, headers=HEADERS, **self.creds) + except Exception as err: + self.module.fail_json( + msg="Failed to update the iSCSI target settings. Array Id [%s]. Error [%s]." + % (self.ssid, to_native(err))) + return update + + def update(self): + update = self.apply_iscsi_settings() + update = self.apply_target_changes() or update + + target = self.target + data = dict((key, target[key]) for key in target if key in ['iqn', 'alias']) + + self.module.exit_json(msg="The interface settings have been updated.", changed=update, **data) + + def __call__(self, *args, **kwargs): + self.update() + + +def main(): + iface = IscsiTarget() + iface() + + +if __name__ == '__main__': + main() diff --git a/test/integration/targets/netapp_eseries_iscsi_target/aliases b/test/integration/targets/netapp_eseries_iscsi_target/aliases new file mode 100644 index 0000000000..d314d14a74 --- /dev/null +++ b/test/integration/targets/netapp_eseries_iscsi_target/aliases @@ -0,0 +1,10 @@ +# This test is not enabled by default, but can be utilized by defining required variables in integration_config.yml +# Example integration_config.yml: +# --- +#netapp_e_api_host: 10.113.1.111:8443 +#netapp_e_api_username: admin +#netapp_e_api_password: myPass +#netapp_e_ssid: 1 + +unsupported +netapp/eseries diff --git a/test/integration/targets/netapp_eseries_iscsi_target/tasks/main.yml b/test/integration/targets/netapp_eseries_iscsi_target/tasks/main.yml new file mode 100644 index 0000000000..996354c886 --- /dev/null +++ b/test/integration/targets/netapp_eseries_iscsi_target/tasks/main.yml @@ -0,0 +1 @@ +- include_tasks: run.yml diff --git a/test/integration/targets/netapp_eseries_iscsi_target/tasks/run.yml b/test/integration/targets/netapp_eseries_iscsi_target/tasks/run.yml new file mode 100644 index 0000000000..b519241d1b --- /dev/null +++ b/test/integration/targets/netapp_eseries_iscsi_target/tasks/run.yml @@ -0,0 +1,68 @@ +# Test code for the netapp_e_iscsi_interface module +# (c) 2018, NetApp, Inc +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +- name: NetApp Test iSCSI Target module + fail: + msg: 'Please define netapp_e_api_username, netapp_e_api_password, netapp_e_api_host, and netapp_e_ssid.' + when: netapp_e_api_username is undefined or netapp_e_api_password is undefined + or netapp_e_api_host is undefined or netapp_e_ssid is undefined + vars: &vars + credentials: &creds + api_url: "https://{{ netapp_e_api_host }}/devmgr/v2" + api_username: "{{ netapp_e_api_username }}" + api_password: "{{ netapp_e_api_password }}" + ssid: "{{ netapp_e_ssid }}" + validate_certs: no + secrets: &secrets + # 12 characters + - 012345678912 + # 16 characters + - 0123456789123456 + +- name: set credentials + set_fact: + credentials: *creds + +- name: Show some debug information + debug: + msg: "Using user={{ credentials.api_username }} on server={{ credentials.api_url }}." + +- name: Ensure we can set the chap secret + netapp_e_iscsi_target: + <<: *creds + name: myTarget + chap_secret: "{{ item }}" + loop: *secrets + +- name: Turn off all of the options + netapp_e_iscsi_target: + <<: *creds + name: abc + ping: no + unnamed_discovery: no + +- name: Ensure we can set the ping option + netapp_e_iscsi_target: + <<: *creds + name: myTarget + ping: yes + unnamed_discovery: yes + register: result + +- name: Ensure we received a change + assert: + that: result.changed + +- name: Run the ping change in check-mode + netapp_e_iscsi_target: + <<: *creds + name: myTarget + ping: yes + unnamed_discovery: yes + check: yes + register: result + +- name: Ensure no change resulted + assert: + that: not result.changed diff --git a/test/units/modules/storage/netapp/test_netapp_e_iscsi_target.py b/test/units/modules/storage/netapp/test_netapp_e_iscsi_target.py new file mode 100644 index 0000000000..d73555fadc --- /dev/null +++ b/test/units/modules/storage/netapp/test_netapp_e_iscsi_target.py @@ -0,0 +1,134 @@ +# coding=utf-8 +# (c) 2018, NetApp Inc. +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +from mock import MagicMock + +from ansible.modules.storage.netapp.netapp_e_iscsi_target import IscsiTarget +from units.modules.utils import AnsibleExitJson, AnsibleFailJson, ModuleTestCase, set_module_args + +__metaclass__ = type + +import mock + +from ansible.compat.tests.mock import PropertyMock + + +class IscsiTargetTest(ModuleTestCase): + REQUIRED_PARAMS = { + 'api_username': 'rw', + 'api_password': 'password', + 'api_url': 'http://localhost', + 'ssid': '1', + 'name': 'abc', + } + + CHAP_SAMPLE = 'a' * 14 + + REQ_FUNC = 'ansible.modules.storage.netapp.netapp_e_iscsi_target.request' + + def _set_args(self, args=None): + module_args = self.REQUIRED_PARAMS.copy() + if args is not None: + module_args.update(args) + set_module_args(module_args) + + def test_validate_params(self): + """Ensure we can pass valid parameters to the module""" + for i in range(12, 16): + secret = 'a' * i + self._set_args(dict(chap=secret)) + tgt = IscsiTarget() + + def test_invalid_chap_secret(self): + for secret in [11 * 'a', 17 * 'a', u'©' * 12]: + with self.assertRaisesRegexp(AnsibleFailJson, r'.*?CHAP secret is not valid.*') as result: + self._set_args(dict(chap=secret)) + tgt = IscsiTarget() + + def test_apply_iscsi_settings(self): + """Ensure that the presence of CHAP always triggers an update.""" + self._set_args(dict(chap=self.CHAP_SAMPLE)) + tgt = IscsiTarget() + + # CHAP is enabled + fake = dict(alias=self.REQUIRED_PARAMS.get('name'), chap=True) + + # We don't care about the return here + with mock.patch(self.REQ_FUNC, return_value=(200, "")) as request: + with mock.patch.object(IscsiTarget, 'target', new_callable=PropertyMock) as call: + call.return_value = fake + self.assertTrue(tgt.apply_iscsi_settings()) + self.assertTrue(request.called, msg="An update was expected!") + + # Retest with check_mode enabled + tgt.check_mode = True + request.reset_mock() + self.assertTrue(tgt.apply_iscsi_settings()) + self.assertFalse(request.called, msg="No update was expected in check_mode!") + + def test_apply_iscsi_settings_no_change(self): + """Ensure that we don't make unnecessary requests or updates""" + name = 'abc' + self._set_args(dict(alias=name)) + fake = dict(alias=name, chap=False) + with mock.patch(self.REQ_FUNC, return_value=(200, "")) as request: + with mock.patch.object(IscsiTarget, 'target', new_callable=PropertyMock) as call: + call.return_value = fake + tgt = IscsiTarget() + self.assertFalse(tgt.apply_iscsi_settings()) + self.assertFalse(request.called, msg="No update was expected!") + + def test_apply_iscsi_settings_fail(self): + """Ensure we handle request failures cleanly""" + self._set_args() + fake = dict(alias='', chap=True) + with self.assertRaisesRegexp(AnsibleFailJson, r".*?update.*"): + with mock.patch(self.REQ_FUNC, side_effect=Exception) as request: + with mock.patch.object(IscsiTarget, 'target', new_callable=PropertyMock) as call: + call.return_value = fake + tgt = IscsiTarget() + tgt.apply_iscsi_settings() + + def test_apply_target_changes(self): + """Ensure that changes trigger an update.""" + self._set_args(dict(ping=True, unnamed_discovery=True)) + tgt = IscsiTarget() + + # CHAP is enabled + fake = dict(ping=False, unnamed_discovery=False) + + # We don't care about the return here + with mock.patch(self.REQ_FUNC, return_value=(200, "")) as request: + with mock.patch.object(IscsiTarget, 'target', new_callable=PropertyMock) as call: + call.return_value = fake + self.assertTrue(tgt.apply_target_changes()) + self.assertTrue(request.called, msg="An update was expected!") + + # Retest with check_mode enabled + tgt.check_mode = True + request.reset_mock() + self.assertTrue(tgt.apply_target_changes()) + self.assertFalse(request.called, msg="No update was expected in check_mode!") + + def test_apply_target_changes_no_change(self): + """Ensure that we don't make unnecessary requests or updates""" + self._set_args(dict(ping=True, unnamed_discovery=True)) + fake = dict(ping=True, unnamed_discovery=True) + with mock.patch(self.REQ_FUNC, return_value=(200, "")) as request: + with mock.patch.object(IscsiTarget, 'target', new_callable=PropertyMock) as call: + call.return_value = fake + tgt = IscsiTarget() + self.assertFalse(tgt.apply_target_changes()) + self.assertFalse(request.called, msg="No update was expected!") + + def test_apply_target_changes_fail(self): + """Ensure we handle request failures cleanly""" + self._set_args() + fake = dict(ping=False, unnamed_discovery=False) + with self.assertRaisesRegexp(AnsibleFailJson, r".*?update.*"): + with mock.patch(self.REQ_FUNC, side_effect=Exception) as request: + with mock.patch.object(IscsiTarget, 'target', new_callable=PropertyMock) as call: + call.return_value = fake + tgt = IscsiTarget() + tgt.apply_target_changes()