Merge branch 'rax-ptr' of https://github.com/sivel/ansible into sivel-rax-ptr

This commit is contained in:
James Cammarata 2014-07-14 12:27:46 -05:00
commit b1e99c2067
2 changed files with 151 additions and 24 deletions

View file

@ -45,9 +45,9 @@ options:
- Time to live of domain in seconds - Time to live of domain in seconds
default: 3600 default: 3600
notes: notes:
- "It is recommended that plays utilizing this module be run with C(serial: 1) - "It is recommended that plays utilizing this module be run with
to avoid exceeding the API request limit imposed by the Rackspace CloudDNS C(serial: 1) to avoid exceeding the API request limit imposed by
API" the Rackspace CloudDNS API"
author: Matt Martz author: Matt Martz
extends_documentation_fragment: rackspace extends_documentation_fragment: rackspace
''' '''

View file

@ -34,8 +34,12 @@ options:
required: True required: True
domain: domain:
description: description:
- Domain name to create the record in - Domain name to create the record in. This is an invalid option when
required: True type=PTR
loadbalancer:
description:
- Load Balancer ID to create a PTR record for. Only used with type=PTR
version_added: 1.7
name: name:
description: description:
- FQDN record name to create - FQDN record name to create
@ -44,6 +48,10 @@ options:
description: description:
- Required for MX and SRV records, but forbidden for other record types. - Required for MX and SRV records, but forbidden for other record types.
If specified, must be an integer from 0 to 65535. If specified, must be an integer from 0 to 65535.
server:
description:
- Server ID to create a PTR record for. Only used with type=PTR
version_added: 1.7
state: state:
description: description:
- Indicate desired state of the resource - Indicate desired state of the resource
@ -53,7 +61,7 @@ options:
default: present default: present
ttl: ttl:
description: description:
- Time to live of domain in seconds - Time to live of record in seconds
default: 3600 default: 3600
type: type:
description: description:
@ -66,32 +74,45 @@ options:
- NS - NS
- SRV - SRV
- TXT - TXT
default: A - PTR
required: true
notes: notes:
- "It is recommended that plays utilizing this module be run with C(serial: 1) - "It is recommended that plays utilizing this module be run with
to avoid exceeding the API request limit imposed by the Rackspace CloudDNS C(serial: 1) to avoid exceeding the API request limit imposed by
API" the Rackspace CloudDNS API"
- To manipulate a C(PTR) record either C(loadbalancer) or C(server) must be
supplied
- C(PTR) record support was added in version 1.7
author: Matt Martz author: Matt Martz
extends_documentation_fragment: rackspace extends_documentation_fragment: rackspace
''' '''
EXAMPLES = ''' EXAMPLES = '''
- name: Create record - name: Create DNS Records
hosts: all hosts: all
gather_facts: False gather_facts: False
tasks: tasks:
- name: Record create request - name: Create A record
local_action: local_action:
module: rax_dns_record module: rax_dns_record
credentials: ~/.raxpub credentials: ~/.raxpub
domain: example.org domain: example.org
name: www.example.org name: www.example.org
data: 127.0.0.1 data: "{{ rax_accessipv4 }}"
type: A type: A
register: rax_dns_record register: a_record
- name: Create PTR record
local_action:
module: rax_dns_record
credentials: ~/.raxpub
server: "{{ rax_id }}"
name: "{{ inventory_hostname }}"
region: DFW
register: ptr_record
''' '''
from types import NoneType from uuid import UUID
try: try:
import pyrax import pyrax
@ -99,7 +120,7 @@ try:
except ImportError: except ImportError:
HAS_PYRAX = False HAS_PYRAX = False
NON_CALLABLES = (basestring, bool, dict, int, list, NoneType) NON_CALLABLES = (basestring, bool, dict, int, list, type(None))
def to_dict(obj): def to_dict(obj):
@ -111,8 +132,95 @@ def to_dict(obj):
return instance return instance
def rax_dns_record(module, comment, data, domain, name, priority, record_type, def rax_dns_record_ptr(module, data=None, comment=None, loadbalancer=None,
state, ttl): name=None, server=None, state='present', ttl=7200):
"""Function for manipulating DNS PTR records"""
changed = False
results = []
cs = pyrax.cloudservers
clb = pyrax.cloud_loadbalancers
dns = pyrax.cloud_dns
if not cs or not clb or not dns:
module.fail_json(msg='Failed to instantiate client. This '
'typically indicates an invalid region or an '
'incorrectly capitalized region name.')
if loadbalancer:
try:
UUID(loadbalancer)
found = [clb.get(loadbalancer)]
except:
for lb in clb.list():
if loadbalancer == lb.name:
found.append(lb)
if len(found) != 1:
module.fail_json(msg='Could not match a loadbalancer with %s' %
loadbalancer)
elif server:
try:
UUID(server)
found = [cs.servers.get(server)]
except:
found = cs.servers.list(search_opts=dict(name='^%s$' % server))
if len(found) != 1:
module.fail_json(msg='Could not match a server with %s' %
server)
item = found[0]
if state == 'present':
current = dns.list_ptr_records(item)
for record in current:
if record.data == data:
if record.ttl != ttl or record.name != name:
try:
dns.update_ptr_record(item, record, name, data, ttl)
changed = True
except Exception, e:
module.fail_json(msg='%s' % e.message)
record.ttl = ttl
record.name = name
results.append(to_dict(record))
break
else:
results.append(to_dict(record))
break
if not results:
record = dict(name=name, type='PTR', data=data, ttl=ttl,
comment=comment)
try:
results = dns.add_ptr_records(item, [record])
changed = True
except Exception, e:
module.fail_json(msg='%s' % e.message)
module.exit_json(changed=changed, records=results)
elif state == 'absent':
current = dns.list_ptr_records(item)
for record in current:
if record.data == data:
results.append(to_dict(record))
break
if results:
try:
dns.delete_ptr_records(item, data)
changed = True
except Exception, e:
module.fail_json(msg='%s' % e.message)
module.exit_json(changed=changed, records=results)
def rax_dns_record(module, comment=None, data=None, domain=None, name=None,
priority=None, record_type='A', state='present', ttl=7200):
"""Function for manipulating record types other than PTR"""
changed = False changed = False
dns = pyrax.cloud_dns dns = pyrax.cloud_dns
@ -145,7 +253,7 @@ def rax_dns_record(module, comment, data, domain, name, priority, record_type,
} }
if comment: if comment:
record_data.update(dict(comment=comment)) record_data.update(dict(comment=comment))
if priority: if priority and record_type.upper() in ['MX', 'SRV']:
record_data.update(dict(priority=priority)) record_data.update(dict(priority=priority))
record = domain.add_records([record_data])[0] record = domain.add_records([record_data])[0]
@ -201,19 +309,27 @@ def main():
dict( dict(
comment=dict(), comment=dict(),
data=dict(required=True), data=dict(required=True),
domain=dict(required=True), domain=dict(),
loadbalancer=dict(),
name=dict(required=True), name=dict(required=True),
priority=dict(type='int'), priority=dict(type='int'),
server=dict(),
state=dict(default='present', choices=['present', 'absent']), state=dict(default='present', choices=['present', 'absent']),
ttl=dict(type='int', default=3600), ttl=dict(type='int', default=3600),
type=dict(default='A', choices=['A', 'AAAA', 'CNAME', 'MX', 'NS', type=dict(required=True, choices=['A', 'AAAA', 'CNAME', 'MX', 'NS',
'SRV', 'TXT']) 'SRV', 'TXT', 'PTR'])
) )
) )
module = AnsibleModule( module = AnsibleModule(
argument_spec=argument_spec, argument_spec=argument_spec,
required_together=rax_required_together(), required_together=rax_required_together(),
mutually_exclusive=[
['server', 'loadbalancer', 'domain'],
],
required_one_of=[
['server', 'loadbalancer', 'domain'],
],
) )
if not HAS_PYRAX: if not HAS_PYRAX:
@ -222,16 +338,27 @@ def main():
comment = module.params.get('comment') comment = module.params.get('comment')
data = module.params.get('data') data = module.params.get('data')
domain = module.params.get('domain') domain = module.params.get('domain')
loadbalancer = module.params.get('loadbalancer')
name = module.params.get('name') name = module.params.get('name')
priority = module.params.get('priority') priority = module.params.get('priority')
server = module.params.get('server')
state = module.params.get('state') state = module.params.get('state')
ttl = module.params.get('ttl') ttl = module.params.get('ttl')
record_type = module.params.get('type') record_type = module.params.get('type')
setup_rax_module(module, pyrax) setup_rax_module(module, pyrax)
rax_dns_record(module, comment, data, domain, name, priority, record_type, if record_type.upper() == 'PTR':
state, ttl) if not server and not loadbalancer:
module.fail_json(msg='one of the following is required: '
'server,loadbalancer')
rax_dns_record_ptr(module, data=data, comment=comment,
loadbalancer=loadbalancer, name=name, server=server,
state=state, ttl=ttl)
else:
rax_dns_record(module, comment=comment, data=data, domain=domain,
name=name, priority=priority, record_type=record_type,
state=state, ttl=ttl)
# import module snippets # import module snippets