meraki_ssid - Improve change reporting (#56201)

* Improve change reporting for meraki_ssid
- Documentation is more clear about dependencies
- Not all change reports are accurate without new algorithm
- Improved integration tests

* Rename changelog fragment

* Enable all tests in integration tests
- Fix type merging

* Add more integration tests for code coverage

* Update URL creation
This commit is contained in:
Kevin Breit 2019-06-12 10:06:00 -05:00 committed by Nathaniel Case
parent 1fa7bfcd5d
commit 17c475b101
3 changed files with 235 additions and 60 deletions

View file

@ -0,0 +1,3 @@
bugfixes:
- "meraki_ssid - Provides more accurate change results for some operations."
- "meraki_ssid - Improved documentation about parameter dependencies."

View file

@ -110,6 +110,7 @@ options:
secret: secret:
description: description:
- RADIUS password. - RADIUS password.
- Setting password is not idempotent.
type: str type: str
radius_coa_enabled: radius_coa_enabled:
description: description:
@ -145,6 +146,7 @@ options:
secret: secret:
description: description:
- RADIUS password. - RADIUS password.
- Setting password is not idempotent.
type: str type: str
ip_assignment_mode: ip_assignment_mode:
description: description:
@ -158,18 +160,23 @@ options:
use_vlan_tagging: use_vlan_tagging:
description: description:
- Set whether to use VLAN tagging. - Set whether to use VLAN tagging.
- Requires C(default_vlan_id) to be set.
type: bool type: bool
default_vlan_id: default_vlan_id:
description: description:
- Default VLAN ID. - Default VLAN ID.
- Requires C(ip_assignment_mode) to be C(Bridge mode) or C(Layer 3 roaming).
type: str type: str
vlan_id: vlan_id:
description: description:
- ID number of VLAN on SSID. - ID number of VLAN on SSID.
- Requires C(ip_assignment_mode) to be C(ayer 3 roaming with a concentrator) or C(VPN).
type: int type: int
ap_tags_vlan_ids: ap_tags_vlan_ids:
description: description:
- List of VLAN tags. - List of VLAN tags.
- Requires C(ip_assignment_mode) to be C(Bridge mode) or C(Layer 3 roaming).
- Requires C(use_vlan_tagging) to be C(True).
type: list type: list
suboptions: suboptions:
tags: tags:
@ -490,7 +497,7 @@ def main():
meraki.params['follow_redirects'] = 'all' meraki.params['follow_redirects'] = 'all'
query_urls = {'ssid': '/networks/{net_id}/ssids'} query_urls = {'ssid': '/networks/{net_id}/ssids'}
query_url = {'ssid': '/networks/{net_id}/ssids/'} query_url = {'ssid': '/networks/{net_id}/ssids/{number}'}
update_url = {'ssid': '/networks/{net_id}/ssids/'} update_url = {'ssid': '/networks/{net_id}/ssids/'}
meraki.url_catalog['get_all'].update(query_urls) meraki.url_catalog['get_all'].update(query_urls)
@ -522,6 +529,9 @@ def main():
if meraki.params['auth_mode'] not in ('open-with-radius', '8021x-radius') or meraki.params['radius_accounting_enabled'] is False: if meraki.params['auth_mode'] not in ('open-with-radius', '8021x-radius') or meraki.params['radius_accounting_enabled'] is False:
meraki.fail_json(msg='radius_accounting_servers is only allowed when auth_mode is open_with_radius or 8021x-radius and \ meraki.fail_json(msg='radius_accounting_servers is only allowed when auth_mode is open_with_radius or 8021x-radius and \
radius_accounting_enabled is true') radius_accounting_enabled is true')
if meraki.params['use_vlan_tagging'] is True:
if meraki.params['default_vlan_id'] is None:
meraki.fail_json(msg="default_vlan_id is required when use_vlan_tagging is True")
# manipulate or modify the state as needed (this is going to be the # manipulate or modify the state as needed (this is going to be the
# part where your module will do what it needs to do) # part where your module will do what it needs to do)
@ -536,10 +546,10 @@ def main():
if meraki.params['state'] == 'query': if meraki.params['state'] == 'query':
if meraki.params['name']: if meraki.params['name']:
ssid_id = get_ssid_number(meraki.params['name'], get_ssids(meraki, net_id)) ssid_id = get_ssid_number(meraki.params['name'], get_ssids(meraki, net_id))
path = meraki.construct_path('get_one', net_id=net_id) + str(ssid_id) path = meraki.construct_path('get_one', net_id=net_id, custom={'number': ssid_id})
meraki.result['data'] = meraki.request(path, method='GET') meraki.result['data'] = meraki.request(path, method='GET')
elif meraki.params['number']: elif meraki.params['number'] is not None:
path = meraki.construct_path('get_one', net_id=net_id) + meraki.params['number'] path = meraki.construct_path('get_one', net_id=net_id, custom={'number': meraki.params['number']})
meraki.result['data'] = meraki.request(path, method='GET') meraki.result['data'] = meraki.request(path, method='GET')
else: else:
meraki.result['data'] = get_ssids(meraki, net_id) meraki.result['data'] = get_ssids(meraki, net_id)
@ -548,12 +558,21 @@ def main():
for k, v in param_map.items(): for k, v in param_map.items():
if meraki.params[v] is not None: if meraki.params[v] is not None:
payload[k] = meraki.params[v] payload[k] = meraki.params[v]
# Short term solution for camelCase/snake_case differences
# Will be addressed later with a method in module utils
if meraki.params['ap_tags_vlan_ids'] is not None:
for i in payload['apTagsAndVlanIds']:
try:
i['vlanId'] = i['vlan_id']
del i['vlan_id']
except KeyError:
pass
ssids = get_ssids(meraki, net_id) ssids = get_ssids(meraki, net_id)
number = meraki.params['number'] number = meraki.params['number']
if number is None: if number is None:
number = get_ssid_number(meraki.params['name'], ssids) number = get_ssid_number(meraki.params['name'], ssids)
original = ssids[number] original = ssids[number]
if meraki.is_update_required(original, payload): if meraki.is_update_required(original, payload, optional_ignore=['secret']):
ssid_id = meraki.params['number'] ssid_id = meraki.params['number']
if ssid_id is None: # Name should be used to lookup number if ssid_id is None: # Name should be used to lookup number
ssid_id = get_ssid_number(meraki.params['name'], ssids) ssid_id = get_ssid_number(meraki.params['name'], ssids)
@ -579,7 +598,6 @@ def main():
path = meraki.construct_path('update', net_id=net_id) + str(ssid_id) path = meraki.construct_path('update', net_id=net_id) + str(ssid_id)
payload = default_payload payload = default_payload
payload['name'] = payload['name'] + ' ' + str(ssid_id + 1) payload['name'] = payload['name'] + ' ' + str(ssid_id + 1)
# meraki.fail_json(msg='Payload', payload=payload)
result = meraki.request(path, 'PUT', payload=json.dumps(payload)) result = meraki.request(path, 'PUT', payload=json.dumps(payload))
meraki.result['data'] = result meraki.result['data'] = result
meraki.result['changed'] = True meraki.result['changed'] = True

View file

@ -9,16 +9,16 @@
msg: Please define an API key msg: Please define an API key
when: auth_key is not defined when: auth_key is not defined
# - name: Use an invalid domain - name: Use an invalid domain
# meraki_organization: meraki_organization:
# auth_key: '{{ auth_key }}' auth_key: '{{ auth_key }}'
# host: marrrraki.com host: marrrraki.com
# state: present state: present
# org_name: IntTestOrg org_name: IntTestOrg
# output_level: debug output_level: debug
# delegate_to: localhost delegate_to: localhost
# register: invalid_domain register: invalid_domain
# ignore_errors: yes ignore_errors: yes
- name: Disable HTTP - name: Disable HTTP
meraki_organization: meraki_organization:
@ -113,6 +113,23 @@
that: that:
- query_one.data.name == 'AnsibleSSID' - query_one.data.name == 'AnsibleSSID'
- name: Query one SSID with number
meraki_ssid:
auth_key: '{{auth_key}}'
state: query
org_name: '{{test_org_name}}'
net_name: TestNetSSID
number: 1
delegate_to: localhost
register: query_one_number
- debug:
msg: '{{query_one_number}}'
- assert:
that:
- query_one_number.data.name == 'AnsibleSSID'
- name: Disable SSID without specifying number - name: Disable SSID without specifying number
meraki_ssid: meraki_ssid:
auth_key: '{{auth_key}}' auth_key: '{{auth_key}}'
@ -149,6 +166,158 @@
that: that:
- enable_ssid_number.data.enabled == True - enable_ssid_number.data.enabled == True
- name: Set VLAN arg spec
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
number: 1
use_vlan_tagging: yes
ip_assignment_mode: Bridge mode
default_vlan_id: 1
ap_tags_vlan_ids:
- tags: wifi
vlan_id: 2
delegate_to: localhost
register: set_vlan_arg
- debug:
var: set_vlan_arg
- assert:
that: set_vlan_arg is changed
- name: Set VLAN arg spec
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
number: 1
use_vlan_tagging: yes
ip_assignment_mode: Bridge mode
default_vlan_id: 1
ap_tags_vlan_ids:
- tags: wifi
vlan_id: 2
delegate_to: localhost
register: set_vlan_arg_idempotent
- debug:
var: set_vlan_arg_idempotent
- assert:
that: set_vlan_arg_idempotent is not changed
- name: Set PSK
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
name: AnsibleSSID
auth_mode: psk
psk: abc1234567890
encryption_mode: wpa
delegate_to: localhost
register: psk
- debug:
msg: '{{ psk }}'
- assert:
that:
- psk.data.authMode == 'psk'
- psk.data.encryptionMode == 'wpa'
- psk.data.wpaEncryptionMode == 'WPA2 only'
- name: Set PSK with idempotency
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
name: AnsibleSSID
auth_mode: psk
psk: abc1234567890
encryption_mode: wpa
delegate_to: localhost
register: psk_idempotent
- debug:
msg: '{{ psk_idempotent }}'
- assert:
that:
- psk_idempotent is not changed
- name: Enable click-through splash page
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
name: AnsibleSSID
splash_page: Click-through splash page
delegate_to: localhost
register: splash_click
- debug:
msg: '{{ splash_click }}'
- assert:
that:
- splash_click.data.splashPage == 'Click-through splash page'
- name: Configure RADIUS servers
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
name: AnsibleSSID
auth_mode: open-with-radius
radius_servers:
- host: 192.0.1.200
port: 1234
secret: abc98765
delegate_to: localhost
register: set_radius_server
- debug:
msg: '{{ set_radius_server }}'
- assert:
that:
- set_radius_server.data.radiusServers.0.host == '192.0.1.200'
- name: Configure RADIUS servers with idempotency
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
name: AnsibleSSID
auth_mode: open-with-radius
radius_servers:
- host: 192.0.1.200
port: 1234
secret: abc98765
delegate_to: localhost
register: set_radius_server_idempotent
- debug:
var: set_radius_server_idempotent
- assert:
that:
- set_radius_server_idempotent is not changed
#################
# Error testing #
#################
- name: Set PSK with wrong mode - name: Set PSK with wrong mode
meraki_ssid: meraki_ssid:
auth_key: '{{auth_key}}' auth_key: '{{auth_key}}'
@ -190,7 +359,7 @@
that: that:
- psk_invalid_mode.msg == 'PSK requires encryption_mode be set to wpa' - psk_invalid_mode.msg == 'PSK requires encryption_mode be set to wpa'
- name: Set PSK - name: Error for PSK and RADIUS servers
meraki_ssid: meraki_ssid:
auth_key: '{{auth_key}}' auth_key: '{{auth_key}}'
state: present state: present
@ -198,59 +367,44 @@
net_name: TestNetSSID net_name: TestNetSSID
name: AnsibleSSID name: AnsibleSSID
auth_mode: psk auth_mode: psk
psk: abc1234567890
encryption_mode: wpa
delegate_to: localhost
register: psk
- debug:
msg: '{{ psk }}'
- assert:
that:
- psk.data.authMode == 'psk'
- psk.data.encryptionMode == 'wpa'
- psk.data.wpaEncryptionMode == 'WPA2 only'
- name: Enable click-through splash page
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
name: AnsibleSSID
splash_page: Click-through splash page
delegate_to: localhost
register: splash_click
- debug:
msg: '{{ splash_click }}'
- assert:
that:
- splash_click.data.splashPage == 'Click-through splash page'
- name: Configure RADIUS servers
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
name: AnsibleSSID
auth_mode: open-with-radius
radius_servers: radius_servers:
- host: 192.0.1.200 - host: 192.0.1.200
port: 1234 port: 1234
secret: abc98765 secret: abc98765
delegate_to: localhost delegate_to: localhost
register: set_radius_server register: err_radius_server_psk
ignore_errors: yes
- debug: - debug:
msg: '{{ set_radius_server }}' var: err_radius_server_psk
- assert: - assert:
that: that:
- set_radius_server.data.radiusServers.0.host == '192.0.1.200' - 'err_radius_server_psk.msg == "radius_servers requires auth_mode to be open-with-radius or 8021x-radius"'
- name: Set VLAN arg without default VLAN error
meraki_ssid:
auth_key: '{{auth_key}}'
state: present
org_name: '{{test_org_name}}'
net_name: TestNetSSID
number: 1
use_vlan_tagging: yes
ip_assignment_mode: Bridge mode
ap_tags_vlan_ids:
- tags: wifi
vlan_id: 2
delegate_to: localhost
register: set_vlan_arg_err
ignore_errors: yes
- debug:
var: set_vlan_arg_err
- assert:
that:
- 'set_vlan_arg_err.msg == "default_vlan_id is required when use_vlan_tagging is True"'
always: always:
- name: Delete SSID - name: Delete SSID
meraki_ssid: meraki_ssid: