1
0
Fork 0
mirror of https://github.com/ansible-collections/community.general.git synced 2024-09-14 20:13:21 +02:00

dig: Support multiple domains in a single lookup (#6334)

* dig: Support multiple domains in a single lookup (#6334)

The docs for this plugin indicated that multiple domains could be
specified at once, but the code did not support multiple domains.

* Address review feedback.
This commit is contained in:
Kevin P. Fleming 2023-04-16 08:26:44 -04:00 committed by GitHub
parent a35542d0d1
commit 64e797d077
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 75 additions and 48 deletions

View file

@ -0,0 +1,3 @@
minor_changes:
- dig lookup plugin - Support multiple domains to be queried as indicated in docs
(https://github.com/ansible-collections/community.general/pull/6334).

View file

@ -61,6 +61,7 @@ DOCUMENTATION = '''
description: description:
- Return empty result without empty strings, and return empty list instead of C(NXDOMAIN). - Return empty result without empty strings, and return empty list instead of C(NXDOMAIN).
- The default for this option will likely change to C(true) in the future. - The default for this option will likely change to C(true) in the future.
- This option will be forced to C(true) if multiple domains to be queried are specified.
default: false default: false
type: bool type: bool
version_added: 6.0.0 version_added: 6.0.0
@ -95,6 +96,21 @@ EXAMPLES = """
msg: "MX record for gmail.com {{ item }}" msg: "MX record for gmail.com {{ item }}"
with_items: "{{ lookup('community.general.dig', 'gmail.com./MX', wantlist=true) }}" with_items: "{{ lookup('community.general.dig', 'gmail.com./MX', wantlist=true) }}"
- name: Lookup multiple names at once
ansible.builtin.debug:
msg: "A record found {{ item }}"
loop: "{{ query('community.general.dig', 'example.org.', 'example.com.', 'gmail.com.') }}"
- name: Lookup multiple names at once (from list variable)
ansible.builtin.debug:
msg: "A record found {{ item }}"
loop: "{{ query('community.general.dig', *hosts) }}"
vars:
hosts:
- example.org.
- example.com.
- gmail.com.
- ansible.builtin.debug: - ansible.builtin.debug:
msg: "Reverse DNS for 192.0.2.5 is {{ lookup('community.general.dig', '192.0.2.5/PTR') }}" msg: "Reverse DNS for 192.0.2.5 is {{ lookup('community.general.dig', '192.0.2.5/PTR') }}"
- ansible.builtin.debug: - ansible.builtin.debug:
@ -308,7 +324,7 @@ class LookupModule(LookupBase):
edns_size = 4096 edns_size = 4096
myres.use_edns(0, ednsflags=dns.flags.DO, payload=edns_size) myres.use_edns(0, ednsflags=dns.flags.DO, payload=edns_size)
domain = None domains = []
qtype = self.get_option('qtype') qtype = self.get_option('qtype')
flat = self.get_option('flat') flat = self.get_option('flat')
fail_on_error = self.get_option('fail_on_error') fail_on_error = self.get_option('fail_on_error')
@ -365,63 +381,71 @@ class LookupModule(LookupBase):
if '/' in t: if '/' in t:
try: try:
domain, qtype = t.split('/') domain, qtype = t.split('/')
domains.append(domain)
except Exception: except Exception:
domain = t domains.append(t)
else: else:
domain = t domains.append(t)
# print "--- domain = {0} qtype={1} rdclass={2}".format(domain, qtype, rdclass) # print "--- domain = {0} qtype={1} rdclass={2}".format(domain, qtype, rdclass)
if qtype.upper() == 'PTR':
reversed_domains = []
for domain in domains:
try:
n = dns.reversename.from_address(domain)
reversed_domains.append(n.to_text())
except dns.exception.SyntaxError:
pass
except Exception as e:
raise AnsibleError("dns.reversename unhandled exception %s" % to_native(e))
domains = reversed_domains
if len(domains) > 1:
real_empty = True
ret = [] ret = []
if qtype.upper() == 'PTR': for domain in domains:
try: try:
n = dns.reversename.from_address(domain) answers = myres.query(domain, qtype, rdclass=rdclass)
domain = n.to_text() for rdata in answers:
except dns.exception.SyntaxError: s = rdata.to_text()
pass if qtype.upper() == 'TXT':
except Exception as e: s = s[1:-1] # Strip outside quotes on TXT rdata
raise AnsibleError("dns.reversename unhandled exception %s" % to_native(e))
try: if flat:
answers = myres.query(domain, qtype, rdclass=rdclass) ret.append(s)
for rdata in answers: else:
s = rdata.to_text() try:
if qtype.upper() == 'TXT': rd = make_rdata_dict(rdata)
s = s[1:-1] # Strip outside quotes on TXT rdata rd['owner'] = answers.canonical_name.to_text()
rd['type'] = dns.rdatatype.to_text(rdata.rdtype)
rd['ttl'] = answers.rrset.ttl
rd['class'] = dns.rdataclass.to_text(rdata.rdclass)
if flat: ret.append(rd)
ret.append(s) except Exception as err:
else: if fail_on_error:
try: raise AnsibleError("Lookup failed: %s" % str(err))
rd = make_rdata_dict(rdata) ret.append(str(err))
rd['owner'] = answers.canonical_name.to_text()
rd['type'] = dns.rdatatype.to_text(rdata.rdtype)
rd['ttl'] = answers.rrset.ttl
rd['class'] = dns.rdataclass.to_text(rdata.rdclass)
ret.append(rd) except dns.resolver.NXDOMAIN as err:
except Exception as err: if fail_on_error:
if fail_on_error: raise AnsibleError("Lookup failed: %s" % str(err))
raise AnsibleError("Lookup failed: %s" % str(err)) if not real_empty:
ret.append(str(err)) ret.append('NXDOMAIN')
except dns.resolver.NoAnswer as err:
except dns.resolver.NXDOMAIN as err: if fail_on_error:
if fail_on_error: raise AnsibleError("Lookup failed: %s" % str(err))
raise AnsibleError("Lookup failed: %s" % str(err)) if not real_empty:
if not real_empty: ret.append("")
ret.append('NXDOMAIN') except dns.resolver.Timeout as err:
except dns.resolver.NoAnswer as err: if fail_on_error:
if fail_on_error: raise AnsibleError("Lookup failed: %s" % str(err))
raise AnsibleError("Lookup failed: %s" % str(err)) if not real_empty:
if not real_empty: ret.append("")
ret.append("") except dns.exception.DNSException as err:
except dns.resolver.Timeout as err: raise AnsibleError("dns.resolver unhandled exception %s" % to_native(err))
if fail_on_error:
raise AnsibleError("Lookup failed: %s" % str(err))
if not real_empty:
ret.append("")
except dns.exception.DNSException as err:
raise AnsibleError("dns.resolver unhandled exception %s" % to_native(err))
return ret return ret