From f39a4cab7bd7ba5b93d17996843a25895781021c Mon Sep 17 00:00:00 2001 From: Ondra Machacek Date: Fri, 17 May 2024 13:41:57 +0200 Subject: [PATCH] Add vcsa_settings module vcsa_settings module is used to Configure settings of VMware appliance. Signed-off-by: Ondra Machacek --- .../fragments/10_vcsa_settings_new_module.yml | 3 + plugins/module_utils/vmware_rest_client.py | 37 ++ plugins/modules/vcsa_settings.py | 576 ++++++++++++++++++ .../targets/vcsa_settings/mock.json | 167 +++++ .../integration/targets/vcsa_settings/run.yml | 12 + .../targets/vcsa_settings/runme.sh | 3 + .../targets/vcsa_settings/tasks/main.yml | 59 ++ .../targets/vcsa_settings/vars.yml | 6 + 8 files changed, 863 insertions(+) create mode 100644 changelogs/fragments/10_vcsa_settings_new_module.yml create mode 100644 plugins/modules/vcsa_settings.py create mode 100644 tests/integration/targets/vcsa_settings/mock.json create mode 100644 tests/integration/targets/vcsa_settings/run.yml create mode 100755 tests/integration/targets/vcsa_settings/runme.sh create mode 100644 tests/integration/targets/vcsa_settings/tasks/main.yml create mode 100644 tests/integration/targets/vcsa_settings/vars.yml diff --git a/changelogs/fragments/10_vcsa_settings_new_module.yml b/changelogs/fragments/10_vcsa_settings_new_module.yml new file mode 100644 index 00000000..2550115f --- /dev/null +++ b/changelogs/fragments/10_vcsa_settings_new_module.yml @@ -0,0 +1,3 @@ +--- +minor_changes: + - vcsa_settings - Add new module to configure VCSA settings diff --git a/plugins/module_utils/vmware_rest_client.py b/plugins/module_utils/vmware_rest_client.py index 4295d5d7..be8e1a3b 100644 --- a/plugins/module_utils/vmware_rest_client.py +++ b/plugins/module_utils/vmware_rest_client.py @@ -556,3 +556,40 @@ def get_tag_by_category_name(self, tag_name=None, category_name=None): category_id = category_obj.id return self.get_tag_by_category_id(tag_name=tag_name, category_id=category_id) + + def obj_to_dict(self, vmware_obj, r): + """ + Tranform VMware SDK object to dictionary. + Args: + vmware_obj: Object to transform. + r: Dictionary to fill with object data. + """ + for k, v in vars(vmware_obj).items(): + if not k.startswith('_'): + if hasattr(v, '__dict__') and not isinstance(v, str): + self.obj_to_dict(v, r[k]) + elif isinstance(v, int): + r[k] = int(v) + else: + r[k] = str(v) + + def set_param(self, param, cmp_fn, set_fn): + """ + Since most of the check is similar to do. This method implement + generic call for most of the parameters. It checks if parameter + specified is different to one which is currently set and if yes, + it will update it. + + param: AnsibleModule parameter name + cmp_fn: function that compares the parameter value to any API call + set_fn: function that is called if the cmd_fn is true + """ + generic_param = self.params.get(param) + if generic_param is None: + return + + if cmp_fn(generic_param): + self.changed = True + if not self.module.check_mode: + set_fn(generic_param) + self.info[param] = generic_param diff --git a/plugins/modules/vcsa_settings.py b/plugins/modules/vcsa_settings.py new file mode 100644 index 00000000..14e6d70a --- /dev/null +++ b/plugins/modules/vcsa_settings.py @@ -0,0 +1,576 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- + +# Copyright: (c) 2023, Ansible Cloud Team (@ansible-collections) +# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) +# SPDX-License-Identifier: GPL-3.0-or-later + +from __future__ import absolute_import, division, print_function +__metaclass__ = type + + +DOCUMENTATION = r''' +--- +module: vcsa_settings +short_description: Configure vCenter Server Appliance settings +version_added: 1.2.0 +description: + - Configure vCenter Server Appliance settings. +author: + - Ansible Cloud Team (@ansible-collections) +options: + timezone: + description: + - Set time zone. + type: str + global_fips: + description: + - "Enable/Disable Global FIPS mode for the appliance. Caution: Changing the value of this setting will reboot the Appliance." + type: bool + resize_storage: + description: + - Resize all partitions to 100 percent of disk size. + type: bool + default: false + dns_mode: + choices: + - is_static + - dhcp + description: + - Set the DNS mode. + type: str + dns_append: + description: + - If I(true) items from C(dns_domains) and C(dns_servers) will be added to already configured DNS domains/servers. + - If I(false) domains/servers will be overriden. + type: bool + default: true + dns_hostname: + description: + - DNS hostname. + type: str + dns_servers: + description: + - List of DNS servers. + type: list + elements: str + dns_domains: + description: + - List of DNS domains. + type: list + elements: str + timesync_mode: + description: + - Set time synchronization mode. + type: str + choices: + - disabled + - host + - ntp + ntp_servers: + description: + - List of NTP servers. This method updates old NTP servers from configuration and sets the input NTP servers in the configuration. + - If NTP based time synchronization is used internally, the NTP daemon will be restarted to reload given NTP configuration. + - In case NTP based time synchronization is not used, this method only replaces servers in the NTP configuration. + type: list + elements: str + noproxy: + description: + - List of hosts that should be ignored by proxy configuration. + type: list + elements: str + proxy: + elements: dict + type: list + description: + - A list of proxy configurations. + suboptions: + enabled: + description: + - Define if this proxy configuration should be enabled. + type: bool + required: true + url: + description: + - Define the URL of the proxy server (including protocol ie. http://...). + type: str + required: true + port: + description: + - Define the port of the proxy server. + type: int + required: true + protocol: + description: + - Define the protocol of the proxy server(FTP, HTTP, HTTPS). + type: str + required: true + username: + description: + - Define username for the proxy server if proxy requires authentication. + type: str + password: + description: + - Define password for the proxy server if proxy requires authentication. + type: str + dcui_enabled: + description: + - Enable/Disable state of Direct Console User Interface (DCUI TTY2). + type: bool + shell_enabled: + description: + - Enable/Disable state of BASH, that is, access to BASH from within the controlled CLI. + type: bool + shell_timeout: + description: + - The timeout (in seconds) specifies how long you enable the Shell access. The maximum timeout is 86400 seconds(1 day). + - This parameter is mandatory in case C(shell_enabled) is I(true). + type: int + ssh_enabled: + description: + - Enable/Disable state of the SSH-based controlled CLI. + type: bool + consolecli_enabled: + description: + - Enable/Disable state of the console-based controlled CLI (TTY1). + type: bool + firewall_rules: + description: + - Set the ordered list of firewall rules to allow or deny traffic from one or more incoming IP addresses. + - Within the list of traffic rules, rules are processed in order of appearance, from top to bottom. + type: list + elements: dict + suboptions: + address: + description: + - IPv4 or IPv6 address. + type: str + prefix: + description: + - CIDR prefix used to mask address. For example, an IPv4 prefix of 24 ignores the low-order 8 bits of address. + type: int + policy: + description: + - Defines firewall rule policies. + type: str + choices: + - ACCEPT + - IGNORE + - REJECT + - RETURN + interface_name: + description: + - The interface to which this rule applies. An I(*) indicates that the rule applies to all interfaces. + type: str + firewall_rules_append: + description: + - If false the rules overwrites the existing firewall rules and creates a new rule list. If true we append the rules to existing rules. + type: bool + default: true +attributes: + check_mode: + description: The check_mode support. + support: full +extends_documentation_fragment: + - vmware.vmware.vmware_rest_client.documentation +''' + +EXAMPLES = r''' +- name: Enable shell and SSH + vmware.vmware.vcsa_settings: + hostname: "https://vcenter" + username: "username" + password: "password" + ssh_enabled: true + shell_enabled: true + shell_timeout: 120 + +- name: Set firewall rules + vmware.vmware.vcsa_settings: + hostname: "https://vcenter" + username: "username" + password: "password" + firewall_rules: + - address: '1.2.3.7' + interface_name: '*' + prefix: 24 + policy: 'ACCEPT' + +- name: Set NTP servers + vmware.vmware.vcsa_settings: + hostname: "https://vcenter" + username: "username" + password: "password" + timesync_mode: ntp + ntp_servers: + - time.google.com + +- name: Enable HTTP proxy + vmware.vmware.vcsa_settings: + hostname: "https://vcenter" + username: "username" + password: "password" + proxy: + - enabled: true + protocol: 'http' + url: 'http://myproxy' + port: 8080 +''' + +RETURN = r''' +vcsa_settings: + description: + - Information about appliance. + returned: On success + type: dict + sample: { + "consolecli_enabled": false, + "dcui_enabled": true, + "dns_domains": ["abc.com"], + "dns_mode": null, + "noproxy": ["abc.com"], + "ntp_servers": ["time.google.com"], + "proxy": [{"enabled": true, "password": null, "port": 80, "protocol": "http", "url": "http://127.0.0.1", "username": null}], + "resize_storage": false, + "shell_timeout": 350, + "ssh_enabled": true, + "timesync_mode": "ntp" + } +''' + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.vmware.vmware.plugins.module_utils.vmware_rest_client import VmwareRestClient + + +class VmwareVcsaSettings(VmwareRestClient): + def __init__(self, module): + super(VmwareVcsaSettings, self).__init__(module) + self.api_system = self.api_client.appliance.system + self.api_networking = self.api_client.appliance.networking + self.api_access = self.api_client.appliance.access + self.module = module + self.params = module.params + self.changed = False + self.info = {} + + def vcsa_settings(self): + # Security + self.__ssh_enabled() + self.__shell_enabled() + self.__consolecli_enabled() + self.__dcui_enabled() + self.__firewall_rules() + + # Proxy + self.__proxy() + self.__noproxy() + # Ntp + self.__ntp_servers() + self.__timesync_mode() + + # DNS + self.__dns_servers() + self.__dns_mode() + self.__dns_domains() + self.__dns_hostname() + + # General + self.__timezone() + self.__global_fips() + self.__resize_storage() + + # General + def __timezone(self): + self.set_param( + 'timezone', + lambda p: p != self.api_system.time.Timezone.get(), + self.api_system.time.Timezone.set + ) + + def __global_fips(self): + self.set_param( + 'global_fips', + lambda p: p != self.api_system.security.GlobalFips.get().enabled, + lambda p: self.api_system.security.GlobalFips.set( + self.api_system.security.GlobalFips.Info(enabled=p) + ) + ) + + def __resize_storage(self): + self.set_param( + 'resize_storage', + lambda p: p, + lambda _: self.api_system.Storage.resize() + ) + + # DNS + def __dns_mode(self): + dns_mode = self.params.get('dns_mode') + self.info['dns_mode'] = dns_mode + if dns_mode is None: + return + + dns_config = self.api_networking.dns.Servers.get() + if dns_mode == str(dns_config.mode): + return + + if not self.module.check_mode: + self.api_networking.dns.Servers.set( + self.api_networking.dns.Servers.DNSServerConfig( + mode=self.api_networking.dns.Servers.DNSServerMode(dns_mode), + servers=[] if dns_config.servers is None else dns_config.servers, + ) + ) + self.changed = True + + def __dns_servers(self): + dns_servers = self.params.get('dns_servers') + if dns_servers is None: + return + + dns_append = self.params.get('dns_append') + dns_config = self.api_networking.dns.Servers.get() + current = dns_config.servers + if dns_append: + servers_diff = set(dns_servers) - set(current) + if len(servers_diff) == 0: + return + self.changed = True + if not self.module.check_mode: + for s in servers_diff: + self.api_networking.dns.Servers.add(s) + self.info['dns_servers'] = current + servers_diff + else: + if set(dns_servers) == set(current): + return + self.changed = True + if not self.module.check_mode: + self.api_networking.dns.Servers.set( + self.api_networking.dns.Servers.DNSServerConfig( + mode=self.api_networking.dns.Servers.DNSServerMode(str(dns_config.mode)), + servers=self.params.get('dns_servers') + ) + ) + self.info['dns_servers'] = dns_servers + + def __dns_domains(self): + dns_domains = self.params.get('dns_domains') + if dns_domains is None: + return + + dns_append = self.params.get('dns_append') + current = self.api_networking.dns.Domains.list() + if dns_append: + domains_diff = list(set(dns_domains) - set(current)) + if len(domains_diff) == 0: + return + + self.changed = True + if not self.module.check_mode: + for d in domains_diff: + self.api_networking.dns.Domains.add(d) + self.info['dns_domains'] = current + domains_diff + else: + if set(dns_domains) == set(current): + return + + self.changed = True + if not self.module.check_mode: + self.api_networking.dns.Domains.set(dns_domains) + self.info['dns_domains'] = dns_domains + + def __dns_hostname(self): + self.set_param( + 'dns_hostname', + lambda p: p != self.api_networking.dns.Hostname.get(), + self.api_networking.dns.Hostname.set + ) + + # Ntp + def __timesync_mode(self): + self.set_param( + 'timesync_mode', + lambda p: p.lower() != str(self.api_client.appliance.Timesync.get()).lower(), + self.api_client.appliance.Ntp.set + ) + + def __ntp_servers(self): + self.set_param( + 'ntp_servers', + lambda p: p != self.api_client.appliance.Ntp.get(), + self.api_client.appliance.Ntp.set + ) + + # Proxy + def __proxy(self): + proxy = self.params.get('proxy') + if proxy is None: + return + + current = self.api_networking.Proxy.list() + for p in proxy: + if not p['enabled']: + continue + + c = current[p['protocol']] + if c.server != p['url'] or c.port != p['port'] or c.username != p['username'] or p['password'] is not None or c.enabled != p['enabled']: + self.changed = True + if not self.module.check_mode: + current = self.api_networking.Proxy.set( + protocol=p['protocol'], + config=self.api_networking.Proxy.Config( + server=p['url'], + port=p['port'], + username=p['username'], + password=p['password'], + enabled=p['enabled'] + ) + ) + self.info['proxy'] = proxy + + def __noproxy(self): + self.set_param( + 'noproxy', + lambda p: len((set(p) | set(['localhost', '127.0.0.1'])) ^ set(self.api_networking.NoProxy.get())) > 0, + self.api_networking.NoProxy.set + ) + + # Security + def __firewall_rules(self): + appendrules = self.params.get('firewall_rules_append') + firewall_rules = self.params.get('firewall_rules') + if firewall_rules is None: + return + + # Fetch the rules: + current_rules = [] + for r in self.api_networking.firewall.Inbound.get(): + inboud_dict = {} + self.obj_to_dict(r, inboud_dict) + current_rules.append(inboud_dict) + + if appendrules: + if all(elem in current_rules for elem in firewall_rules): + return + current_rules.extend(firewall_rules) + self.__firwall_rules_set(current_rules) + else: + if current_rules == firewall_rules: + return + self.__firwall_rules_set(firewall_rules) + + def __firwall_rules_set(self, firewall_rules): + # Update the dictionary with user provided values: + self.changed = True + self.info['firewall_rules'] = firewall_rules + + # Update the rules: + if self.module.check_mode: + return + + rules = [] + for r in firewall_rules: + rules.append( + self.api_networking.firewall.Inbound.Rule( + address=r['address'], + prefix=r['prefix'], + policy=self.api_networking.firewall.Inbound.Policy(r['policy']), + interface_name=r['interface_name'], + ) + ) + self.api_networking.firewall.Inbound.set(rules) + + def __consolecli_enabled(self): + self.set_param( + 'consolecli_enabled', + lambda p: self.api_access.Consolecli.get() != p, + self.api_access.Consolecli.set + ) + + def __dcui_enabled(self): + self.set_param( + 'dcui_enabled', + lambda p: self.api_access.Dcui.get() != p, + self.api_access.Dcui.set + ) + + def __ssh_enabled(self): + self.set_param( + 'ssh_enabled', + lambda p: self.api_access.Ssh.get() != p, + self.api_access.Ssh.set + ) + + def __shell_enabled(self): + shell_enabled = self.params.get('shell_enabled') + shell_timeout = self.params.get('shell_timeout') + if shell_enabled is None and shell_timeout is None: + return + + shell = self.api_access.Shell.get() + if shell_enabled is not None and shell.enabled != shell_enabled: + shell.enabled = shell_enabled + self.info['shell_enabled'] = shell_enabled + self.changed = True + + if shell_timeout is not None and shell.timeout != shell_timeout and shell_enabled: + shell.timeout = shell_timeout + self.info['shell_timeout'] = shell_timeout + self.changed = True + + if not self.module.check_mode: + self.api_access.Shell.set(shell) + + +def main(): + argument_spec = VmwareRestClient.vmware_client_argument_spec() + argument_spec.update( + dict( + ssh_enabled=dict(type='bool'), + shell_enabled=dict(type='bool'), + shell_timeout=dict(type='int'), + consolecli_enabled=dict(type='bool'), + dcui_enabled=dict(type='bool'), + firewall_rules_append=dict(type='bool', default=True), + firewall_rules=dict(type='list', elements='dict', options=dict( + address=dict(type='str'), + prefix=dict(type='int'), + interface_name=dict(type='str'), + policy=dict(type='str', choices=['ACCEPT', 'IGNORE', 'REJECT', 'RETURN']), + )), + noproxy=dict(type='list', elements='str'), + proxy=dict(type='list', elements='dict', options=dict( + enabled=dict(type='bool', required=True), + url=dict(type='str', required=True), + port=dict(type='int', required=True), + protocol=dict(type='str', required=True), + username=dict(type='str'), + password=dict(type='str', no_log=True), + )), + dns_servers=dict(type='list', elements='str'), + dns_domains=dict(type='list', elements='str'), + dns_hostname=dict(type='str'), + dns_mode=dict(type='str', choices=['is_static', 'dhcp']), + dns_append=dict(type='bool', default=True), + timesync_mode=dict(type='str', choices=['disabled', 'host', 'ntp']), + ntp_servers=dict(type='list', elements='str'), + timezone=dict(type='str'), + global_fips=dict(type='bool'), + resize_storage=dict(type='bool', default=False), + ) + ) + module = AnsibleModule( + argument_spec=argument_spec, + supports_check_mode=True, + required_if=[ + ('shell_enabled', True, ('shell_timeout',)), + ], + ) + + vmware_system = VmwareVcsaSettings(module) + vmware_system.vcsa_settings() + module.exit_json(changed=vmware_system.changed, vcsa_settings=vmware_system.info) + + +if __name__ == '__main__': + main() diff --git a/tests/integration/targets/vcsa_settings/mock.json b/tests/integration/targets/vcsa_settings/mock.json new file mode 100644 index 00000000..d2a37114 --- /dev/null +++ b/tests/integration/targets/vcsa_settings/mock.json @@ -0,0 +1,167 @@ +[ +{ + "httpRequest": { + "method": "POST", + "path": "/rest/com/vmware/cis/session" + }, + "httpResponse": { + "statusCode": 200, + "body": {"value": "72300ca9ff16c5743fa0a6328c8570ce"} + } +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api" + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\":0}, 'id': '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + } +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api", + "body": { + "type": "JSON", + "matchType": "PARTIAL", + "json": "{\"params\":{\"operationId\":\"set\"}}" + } + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\": null}, 'id': '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + }, + "priority": 1 +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api", + "body": { + "type": "JSON", + "matchType": "PARTIAL", + "json": "{\"params\":{\"serviceId\":\"com.vmware.appliance.networking.no_proxy\",\"operationId\":\"get\"}}" + } + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\":[\"abc.com\",\"localhost\",\"127.0.0.1\"]}, 'id': '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + }, + "priority": 1 +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api", + "body": { + "type": "JSON", + "matchType": "PARTIAL", + "json": "{\"params\":{\"serviceId\":\"com.vmware.appliance.networking.proxy\",\"operationId\":\"list\"}}" + } + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\":[{\"STRUCTURE\":{\"map-entry\":{\"value\":{\"STRUCTURE\":{\"com.vmware.appliance.networking.proxy.config\":{\"server\":\"http://127.0.0.1\",\"password\":{\"OPTIONAL\":null},\"port\":80,\"enabled\":true,\"username\":{\"OPTIONAL\":null}}}},\"key\":\"http\"}}},{\"STRUCTURE\":{\"map-entry\":{\"value\":{\"STRUCTURE\":{\"com.vmware.appliance.networking.proxy.config\":{\"server\":\"\",\"password\":{\"OPTIONAL\":null},\"port\":-1,\"enabled\":false,\"username\":{\"OPTIONAL\":null}}}},\"key\":\"https\"}}},{\"STRUCTURE\":{\"map-entry\":{\"value\":{\"STRUCTURE\":{\"com.vmware.appliance.networking.proxy.config\":{\"server\":\"\",\"password\":{\"OPTIONAL\":null},\"port\":-1,\"enabled\":false,\"username\":{\"OPTIONAL\":null}}}},\"key\":\"ftp\"}}}]}, 'id': '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + }, + "priority": 1 +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api", + "body": { + "type": "JSON", + "matchType": "PARTIAL", + "json": "{\"params\":{\"serviceId\":\"com.vmware.appliance.networking.firewall.inbound\",\"operationId\":\"get\"}}" + } + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\":[]}, 'id': '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + }, + "priority": 1 +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api", + "body": { + "type": "JSON", + "matchType": "PARTIAL", + "json": "{\"params\":{\"serviceId\":\"com.vmware.appliance.networking.dns.servers\",\"operationId\":\"get\"}}" + } + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\":{\"STRUCTURE\":{\"com.vmware.appliance.networking.dns.servers.DNS_server_config\":{\"mode\":\"is_static\",\"servers\":[\"10.185.246.1\",\"2607:f0d0:1f01:22:884::2\"]}}}}, 'id': '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + }, + "priority": 1 +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api", + "body": { + "type": "JSON", + "matchType": "PARTIAL", + "json": "{\"params\":{\"serviceId\":\"com.vmware.appliance.networking.dns.domains\",\"operationId\":\"list\"}}" + } + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\": [\"abc.com\"]}, 'id': '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + }, + "priority": 1 +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api", + "body": { + "type": "JSON", + "matchType": "PARTIAL", + "json": "{\"params\":{\"serviceId\":\"com.vmware.appliance.ntp\",\"operationId\":\"get\"}}" + } + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\": [\"time.google.com\"]}, 'id': '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + }, + "priority": 1 +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api", + "body": { + "type": "JSON", + "matchType": "PARTIAL", + "json": "{\"params\":{\"serviceId\":\"com.vmware.appliance.timesync\",\"operationId\":\"get\"}}" + } + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\": \"NTP\"}, 'id': '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + }, + "priority": 1 +}, +{ + "httpRequest": { + "method": "POST", + "path": "/api", + "body": { + "params" : { + "serviceId" : "com.vmware.cis.session", + "operationId" : "delete" + } + } + }, + "httpResponseTemplate": { + "template": "{\"statusCode\": 200, \"headers\": {\"Content-type\": \"application/json\"}, \"body\": {\"jsonrpc\": \"2.0\", \"result\": {\"output\": 0}, \"id\": '${json.parse($!request.body)['id']}'}}", + "templateType": "VELOCITY" + }, + "priority": 1 +} +] diff --git a/tests/integration/targets/vcsa_settings/run.yml b/tests/integration/targets/vcsa_settings/run.yml new file mode 100644 index 00000000..e93da304 --- /dev/null +++ b/tests/integration/targets/vcsa_settings/run.yml @@ -0,0 +1,12 @@ +- hosts: localhost + gather_facts: no + vars_files: + - vars.yml + tasks: + - name: Rest + ansible.builtin.import_role: + name: prepare_rest + + - name: Import vcsa_settings role + ansible.builtin.import_role: + name: vcsa_settings diff --git a/tests/integration/targets/vcsa_settings/runme.sh b/tests/integration/targets/vcsa_settings/runme.sh new file mode 100755 index 00000000..a4c36631 --- /dev/null +++ b/tests/integration/targets/vcsa_settings/runme.sh @@ -0,0 +1,3 @@ +#!/usr/bin/env bash +source ../init.sh +exec ansible-playbook run.yml diff --git a/tests/integration/targets/vcsa_settings/tasks/main.yml b/tests/integration/targets/vcsa_settings/tasks/main.yml new file mode 100644 index 00000000..953434c9 --- /dev/null +++ b/tests/integration/targets/vcsa_settings/tasks/main.yml @@ -0,0 +1,59 @@ +--- +- name: Set VCSA settings + vmware.vmware.vcsa_settings: + validate_certs: false + hostname: "{{ vcenter_hostname }}" + username: "{{ vcenter_username }}" + password: "{{ vcenter_password }}" + port: "{{ vcenter_port }}" + ssh_enabled: true + consolecli_enabled: false + dcui_enabled: true + firewall_rules_append: false + firewall_rules: + - address: '1.2.3.7' + interface_name: '*' + prefix: 24 + policy: 'ACCEPT' + dns_mode: 'dhcp' + dns_append: true + dns_domains: + - xyz.com + timesync_mode: ntp + ntp_servers: + - time.google.com + noproxy: + - abc.com + proxy: + - enabled: true + protocol: 'http' + url: 'http://127.0.0.1' + port: 80 + register: __res + +- name: Assert values + ansible.builtin.assert: + that: + - __res.changed == True + - __res.vcsa_settings.ssh_enabled == True + - __res.vcsa_settings.consolecli_enabled == False + - __res.vcsa_settings.dcui_enabled == True + - __res.vcsa_settings.noproxy | length == 1 + - __res.vcsa_settings.noproxy[0] == 'abc.com' + +- name: Set VCSA settings + vmware.vmware.vcsa_settings: + validate_certs: false + hostname: "{{ vcenter_hostname }}" + username: "{{ vcenter_username }}" + password: "{{ vcenter_password }}" + port: "{{ vcenter_port }}" + consolecli_enabled: false + noproxy: + - abc.com + register: __res + +- name: Assert values + ansible.builtin.assert: + that: + - __res.changed == False diff --git a/tests/integration/targets/vcsa_settings/vars.yml b/tests/integration/targets/vcsa_settings/vars.yml new file mode 100644 index 00000000..c416bd8a --- /dev/null +++ b/tests/integration/targets/vcsa_settings/vars.yml @@ -0,0 +1,6 @@ +vcenter_hostname: "127.0.0.1" +vcenter_username: "user" +vcenter_password: "pass" +vcenter_port: 1080 + +mock_file: "vcsa_settings"