2015-07-23 15:47:32 +05:30
|
|
|
#!/usr/bin/python
|
|
|
|
|
|
|
|
# Copyright (c) 2015 VMware, Inc. All Rights Reserved.
|
|
|
|
#
|
2015-07-28 15:51:57 +05:30
|
|
|
# This file is part of Ansible
|
2015-07-23 15:47:32 +05:30
|
|
|
#
|
2015-07-28 15:51:57 +05:30
|
|
|
# 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.
|
2015-07-23 15:47:32 +05:30
|
|
|
#
|
2015-07-28 15:51:57 +05:30
|
|
|
# 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/>.
|
|
|
|
|
2015-07-23 15:47:32 +05:30
|
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: vca_nat
|
|
|
|
short_description: add remove nat rules in a gateway in a vca
|
|
|
|
description:
|
|
|
|
- Adds or removes nat rules from a gateway in a vca environment
|
|
|
|
version_added: "2.0"
|
2015-09-17 14:52:49 -04:00
|
|
|
author: Peter Sprygada (@privateip)
|
2015-07-23 15:47:32 +05:30
|
|
|
options:
|
|
|
|
username:
|
|
|
|
description:
|
|
|
|
- The vca username or email address, if not set the environment variable VCA_USER is checked for the username.
|
|
|
|
required: false
|
|
|
|
default: None
|
|
|
|
password:
|
|
|
|
description:
|
|
|
|
- The vca password, if not set the environment variable VCA_PASS is checked for the password
|
|
|
|
required: false
|
|
|
|
default: None
|
|
|
|
org:
|
|
|
|
description:
|
|
|
|
- The org to login to for creating vapp, mostly set when the service_type is vdc.
|
|
|
|
required: false
|
|
|
|
default: None
|
2015-10-12 08:35:21 -05:00
|
|
|
instance_id:
|
2015-07-23 15:47:32 +05:30
|
|
|
description:
|
2015-10-12 08:35:21 -05:00
|
|
|
- The instance id in a vchs environment to be used for creating the vapp
|
2015-07-23 15:47:32 +05:30
|
|
|
required: false
|
|
|
|
default: None
|
|
|
|
host:
|
|
|
|
description:
|
|
|
|
- The authentication host to be used when service type is vcd.
|
|
|
|
required: false
|
|
|
|
default: None
|
|
|
|
api_version:
|
|
|
|
description:
|
|
|
|
- The api version to be used with the vca
|
|
|
|
required: false
|
|
|
|
default: "5.7"
|
|
|
|
service_type:
|
|
|
|
description:
|
|
|
|
- The type of service we are authenticating against
|
|
|
|
required: false
|
|
|
|
default: vca
|
2015-09-08 10:50:13 -04:00
|
|
|
choices: [ "vca", "vchs", "vcd" ]
|
2015-07-23 15:47:32 +05:30
|
|
|
state:
|
|
|
|
description:
|
|
|
|
- if the object should be added or removed
|
|
|
|
required: false
|
|
|
|
default: present
|
|
|
|
choices: [ "present", "absent" ]
|
|
|
|
verify_certs:
|
|
|
|
description:
|
|
|
|
- If the certificates of the authentication is to be verified
|
|
|
|
required: false
|
|
|
|
default: True
|
|
|
|
vdc_name:
|
|
|
|
description:
|
|
|
|
- The name of the vdc where the gateway is located.
|
|
|
|
required: false
|
|
|
|
default: None
|
|
|
|
gateway_name:
|
|
|
|
description:
|
|
|
|
- The name of the gateway of the vdc where the rule should be added
|
|
|
|
required: false
|
|
|
|
default: gateway
|
|
|
|
purge_rules:
|
|
|
|
description:
|
|
|
|
- If set to true, it will delete all rules in the gateway that are not given as paramter to this module.
|
|
|
|
required: false
|
|
|
|
default: false
|
|
|
|
nat_rules:
|
|
|
|
description:
|
|
|
|
- A list of rules to be added to the gateway, Please see examples on valid entries
|
|
|
|
required: True
|
|
|
|
default: false
|
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = '''
|
|
|
|
|
|
|
|
#An example for a source nat
|
|
|
|
|
|
|
|
- hosts: localhost
|
|
|
|
connection: local
|
|
|
|
tasks:
|
|
|
|
- vca_nat:
|
|
|
|
instance_id: 'b15ff1e5-1024-4f55-889f-ea0209726282'
|
|
|
|
vdc_name: 'benz_ansible'
|
|
|
|
state: 'present'
|
2015-09-08 10:50:13 -04:00
|
|
|
nat_rules:
|
2015-07-23 15:47:32 +05:30
|
|
|
- rule_type: SNAT
|
|
|
|
original_ip: 192.168.2.10
|
|
|
|
translated_ip: 107.189.95.208
|
|
|
|
|
|
|
|
#example for a DNAT
|
|
|
|
- hosts: localhost
|
|
|
|
connection: local
|
|
|
|
tasks:
|
|
|
|
- vca_nat:
|
|
|
|
instance_id: 'b15ff1e5-1024-4f55-889f-ea0209726282'
|
|
|
|
vdc_name: 'benz_ansible'
|
|
|
|
state: 'present'
|
2015-09-08 10:50:13 -04:00
|
|
|
nat_rules:
|
2015-07-23 15:47:32 +05:30
|
|
|
- rule_type: DNAT
|
|
|
|
original_ip: 107.189.95.208
|
|
|
|
original_port: 22
|
|
|
|
translated_ip: 192.168.2.10
|
|
|
|
translated_port: 22
|
|
|
|
|
|
|
|
'''
|
|
|
|
|
2015-09-17 14:52:49 -04:00
|
|
|
import time
|
|
|
|
import json
|
|
|
|
import xmltodict
|
|
|
|
|
|
|
|
VALID_RULE_KEYS = ['rule_type', 'original_ip', 'original_port',
|
|
|
|
'translated_ip', 'translated_port', 'protocol']
|
|
|
|
|
|
|
|
|
|
|
|
def validate_nat_rules(nat_rules):
|
2015-07-23 15:47:32 +05:30
|
|
|
for rule in nat_rules:
|
|
|
|
if not isinstance(rule, dict):
|
2015-09-17 14:52:49 -04:00
|
|
|
raise VcaError("nat rules must be a list of dictionaries, "
|
|
|
|
"Please check", valid_keys=VALID_RULE_KEYS)
|
|
|
|
|
2015-07-23 15:47:32 +05:30
|
|
|
for k in rule.keys():
|
|
|
|
if k not in VALID_RULE_KEYS:
|
2015-09-17 14:52:49 -04:00
|
|
|
raise VcaError("%s is not a valid key in nat rules, please "
|
|
|
|
"check above.." % k, valid_keys=VALID_RULE_KEYS)
|
|
|
|
|
|
|
|
rule['original_port'] = str(rule.get('original_port', 'any')).lower()
|
|
|
|
rule['original_ip'] = rule.get('original_ip', 'any').lower()
|
|
|
|
rule['translated_ip'] = rule.get('translated_ip', 'any').lower()
|
|
|
|
rule['translated_port'] = str(rule.get('translated_port', 'any')).lower()
|
|
|
|
rule['protocol'] = rule.get('protocol', 'any').lower()
|
|
|
|
rule['rule_type'] = rule.get('rule_type', 'DNAT').lower()
|
|
|
|
|
2015-07-23 15:47:32 +05:30
|
|
|
return nat_rules
|
|
|
|
|
|
|
|
|
2015-09-17 14:52:49 -04:00
|
|
|
def nat_rules_to_dict(nat_rules):
|
2015-07-23 15:47:32 +05:30
|
|
|
result = []
|
2015-09-17 14:52:49 -04:00
|
|
|
for rule in nat_rules:
|
|
|
|
gw_rule = rule.get_GatewayNatRule()
|
|
|
|
result.append(
|
|
|
|
dict(
|
|
|
|
rule_type=rule.get_RuleType().lower(),
|
|
|
|
original_ip=gw_rule.get_OriginalIp().lower(),
|
|
|
|
original_port=(gw_rule.get_OriginalPort().lower() or 'any'),
|
|
|
|
translated_ip=gw_rule.get_TranslatedIp().lower(),
|
|
|
|
translated_port=(gw_rule.get_TranslatedPort().lower() or 'any'),
|
|
|
|
protocol=(gw_rule.get_Protocol().lower() or 'any')
|
|
|
|
)
|
|
|
|
)
|
2015-07-23 15:47:32 +05:30
|
|
|
return result
|
|
|
|
|
2015-09-17 14:52:49 -04:00
|
|
|
def rule_to_string(rule):
|
|
|
|
strings = list()
|
|
|
|
for key, value in rule.items():
|
|
|
|
strings.append('%s=%s' % (key, value))
|
|
|
|
return ', '.join(string)
|
2015-07-23 15:47:32 +05:30
|
|
|
|
|
|
|
def main():
|
2015-09-17 14:52:49 -04:00
|
|
|
argument_spec = vca_argument_spec()
|
|
|
|
argument_spec.update(
|
|
|
|
dict(
|
|
|
|
nat_rules = dict(type='list', default=[]),
|
|
|
|
gateway_name = dict(default='gateway'),
|
|
|
|
purge_rules = dict(default=False, type='bool'),
|
|
|
|
state = dict(default='present', choices=['present', 'absent'])
|
2015-07-23 15:47:32 +05:30
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2015-09-17 14:52:49 -04:00
|
|
|
module = AnsibleModule(argument_spec, supports_check_mode=True)
|
2015-07-23 15:47:32 +05:30
|
|
|
|
2015-09-17 14:52:49 -04:00
|
|
|
vdc_name = module.params.get('vdc_name')
|
|
|
|
state = module.params['state']
|
|
|
|
nat_rules = module.params['nat_rules']
|
|
|
|
gateway_name = module.params['gateway_name']
|
|
|
|
purge_rules = module.params['purge_rules']
|
|
|
|
|
|
|
|
if not purge_rules and not nat_rules:
|
2015-10-12 08:35:21 -05:00
|
|
|
module.fail_json(msg='Must define purge_rules or nat_rules')
|
2015-09-08 10:50:13 -04:00
|
|
|
|
2015-07-23 15:47:32 +05:30
|
|
|
vca = vca_login(module)
|
|
|
|
|
|
|
|
gateway = vca.get_gateway(vdc_name, gateway_name)
|
|
|
|
if not gateway:
|
2015-09-17 14:52:49 -04:00
|
|
|
module.fail_json(msg="Not able to find the gateway %s, please check "
|
|
|
|
"the gateway_name param" % gateway_name)
|
|
|
|
|
|
|
|
try:
|
|
|
|
desired_rules = validate_nat_rules(nat_rules)
|
|
|
|
except VcaError, e:
|
|
|
|
module.fail_json(msg=e.message)
|
|
|
|
|
2015-07-23 15:47:32 +05:30
|
|
|
rules = gateway.get_nat_rules()
|
2015-09-17 14:52:49 -04:00
|
|
|
|
|
|
|
result = dict(changed=False, rules_purged=0)
|
|
|
|
|
|
|
|
deletions = 0
|
|
|
|
additions = 0
|
|
|
|
|
|
|
|
if purge_rules is True and len(rules) > 0:
|
|
|
|
result['rules_purged'] = len(rules)
|
|
|
|
deletions = result['rules_purged']
|
|
|
|
rules = list()
|
|
|
|
if not module.check_mode:
|
|
|
|
gateway.del_all_nat_rules()
|
2015-07-23 15:47:32 +05:30
|
|
|
task = gateway.save_services_configuration()
|
2015-09-17 14:52:49 -04:00
|
|
|
vca.block_until_completed(task)
|
|
|
|
rules = gateway.get_nat_rules()
|
|
|
|
result['changed'] = True
|
|
|
|
|
|
|
|
current_rules = nat_rules_to_dict(rules)
|
|
|
|
|
|
|
|
result['current_rules'] = current_rules
|
|
|
|
result['desired_rules'] = desired_rules
|
|
|
|
|
|
|
|
for rule in desired_rules:
|
|
|
|
if rule not in current_rules:
|
|
|
|
additions += 1
|
|
|
|
if not module.check_mode:
|
|
|
|
gateway.add_nat_rule(**rule)
|
|
|
|
result['changed'] = True
|
|
|
|
result['rules_added'] = additions
|
|
|
|
|
|
|
|
result['delete_rule'] = list()
|
|
|
|
result['delete_rule_rc'] = list()
|
|
|
|
for rule in current_rules:
|
|
|
|
if rule not in desired_rules:
|
|
|
|
deletions += 1
|
|
|
|
if not module.check_mode:
|
|
|
|
result['delete_rule'].append(rule)
|
|
|
|
rc = gateway.del_nat_rule(**rule)
|
|
|
|
result['delete_rule_rc'].append(rc)
|
|
|
|
result['changed'] = True
|
|
|
|
result['rules_deleted'] = deletions
|
|
|
|
|
|
|
|
if not module.check_mode and (additions > 0 or deletions > 0):
|
|
|
|
task = gateway.save_services_configuration()
|
|
|
|
vca.block_until_completed(task)
|
2015-07-23 15:47:32 +05:30
|
|
|
|
2015-09-17 14:52:49 -04:00
|
|
|
module.exit_json(**result)
|
2015-07-23 15:47:32 +05:30
|
|
|
|
2015-09-08 10:50:13 -04:00
|
|
|
|
2015-07-23 15:47:32 +05:30
|
|
|
# import module snippets
|
|
|
|
from ansible.module_utils.basic import *
|
2015-09-17 14:52:49 -04:00
|
|
|
from ansible.module_utils.vca import *
|
|
|
|
|
2015-07-23 15:47:32 +05:30
|
|
|
if __name__ == '__main__':
|
2015-09-17 14:52:49 -04:00
|
|
|
main()
|