2014-10-19 10:04:40 -05:00
|
|
|
#
|
|
|
|
# Copyright (C) 2014 FreeIPA Contributors see COPYING for license
|
|
|
|
#
|
|
|
|
|
2017-05-23 11:35:57 -05:00
|
|
|
import logging
|
|
|
|
|
2014-10-19 10:04:40 -05:00
|
|
|
import dns.name
|
2020-03-05 00:23:02 -06:00
|
|
|
import re
|
2016-11-15 05:57:13 -06:00
|
|
|
try:
|
|
|
|
from xml.etree import cElementTree as etree
|
|
|
|
except ImportError:
|
|
|
|
from xml.etree import ElementTree as etree
|
2014-10-19 10:04:40 -05:00
|
|
|
|
|
|
|
from ipapython import ipa_log_manager, ipautil
|
2020-03-12 10:23:03 -05:00
|
|
|
from ipaserver.dnssec.opendnssec import tasks
|
2014-10-19 10:04:40 -05:00
|
|
|
|
2017-05-23 11:35:57 -05:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2014-10-19 10:04:40 -05:00
|
|
|
# hack: zone object UUID is stored as path to imaginary zone file
|
|
|
|
ENTRYUUID_PREFIX = "/var/lib/ipa/dns/zone/entryUUID/"
|
|
|
|
ENTRYUUID_PREFIX_LEN = len(ENTRYUUID_PREFIX)
|
|
|
|
|
|
|
|
|
2018-09-26 04:59:50 -05:00
|
|
|
class ZoneListReader:
|
2014-10-19 10:04:40 -05:00
|
|
|
def __init__(self):
|
|
|
|
self.names = set() # dns.name
|
|
|
|
self.uuids = set() # UUID strings
|
|
|
|
self.mapping = dict() # {UUID: dns.name}
|
|
|
|
|
|
|
|
def _add_zone(self, name, zid):
|
|
|
|
"""Add zone & UUID to internal structures.
|
|
|
|
|
|
|
|
Zone with given name and UUID must not exist."""
|
|
|
|
# detect duplicate zone names
|
|
|
|
name = dns.name.from_text(name)
|
|
|
|
assert name not in self.names, \
|
|
|
|
'duplicate name (%s, %s) vs. %s' % (name, zid, self.mapping)
|
|
|
|
# duplicate non-None zid is not allowed
|
|
|
|
assert not zid or zid not in self.uuids, \
|
|
|
|
'duplicate UUID (%s, %s) vs. %s' % (name, zid, self.mapping)
|
|
|
|
|
|
|
|
self.names.add(name)
|
|
|
|
self.uuids.add(zid)
|
|
|
|
self.mapping[zid] = name
|
|
|
|
|
|
|
|
def _del_zone(self, name, zid):
|
|
|
|
"""Remove zone & UUID from internal structures.
|
|
|
|
|
|
|
|
Zone with given name and UUID must exist.
|
|
|
|
"""
|
|
|
|
name = dns.name.from_text(name)
|
|
|
|
assert zid is not None
|
|
|
|
assert name in self.names, \
|
|
|
|
'name (%s, %s) does not exist in %s' % (name, zid, self.mapping)
|
|
|
|
assert zid in self.uuids, \
|
|
|
|
'UUID (%s, %s) does not exist in %s' % (name, zid, self.mapping)
|
|
|
|
assert zid in self.mapping and name == self.mapping[zid], \
|
|
|
|
'pair {%s: %s} does not exist in %s' % (zid, name, self.mapping)
|
|
|
|
|
|
|
|
self.names.remove(name)
|
|
|
|
self.uuids.remove(zid)
|
|
|
|
del self.mapping[zid]
|
|
|
|
|
|
|
|
|
|
|
|
class ODSZoneListReader(ZoneListReader):
|
|
|
|
"""One-shot parser for ODS zonelist.xml."""
|
|
|
|
def __init__(self, zonelist_text):
|
|
|
|
super(ODSZoneListReader, self).__init__()
|
2016-11-15 05:57:13 -06:00
|
|
|
root = etree.fromstring(zonelist_text)
|
|
|
|
self._parse_zonelist(root)
|
2014-10-19 10:04:40 -05:00
|
|
|
|
2016-11-15 05:57:13 -06:00
|
|
|
def _parse_zonelist(self, root):
|
2014-10-19 10:04:40 -05:00
|
|
|
"""iterate over Zone elements with attribute 'name' and
|
|
|
|
add IPA zones to self.zones"""
|
2016-11-15 05:57:13 -06:00
|
|
|
if not root.tag == 'ZoneList':
|
|
|
|
raise ValueError(root.tag)
|
|
|
|
for zone_xml in root.findall('./Zone[@name]'):
|
2014-10-19 10:04:40 -05:00
|
|
|
name, zid = self._parse_ipa_zone(zone_xml)
|
|
|
|
self._add_zone(name, zid)
|
|
|
|
|
|
|
|
def _parse_ipa_zone(self, zone_xml):
|
|
|
|
"""Extract zone name, input adapter and detect IPA zones.
|
|
|
|
|
|
|
|
IPA zones have contains Adapters/Input/Adapter element with
|
|
|
|
attribute type = "File" and with value prefixed with ENTRYUUID_PREFIX.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
tuple (zone name, ID)
|
|
|
|
"""
|
|
|
|
name = zone_xml.get('name')
|
2016-11-15 05:57:13 -06:00
|
|
|
zids = []
|
|
|
|
for in_adapter in zone_xml.findall(
|
|
|
|
'./Adapters/Input/Adapter[@type="File"]'):
|
|
|
|
path = in_adapter.text
|
|
|
|
if path.startswith(ENTRYUUID_PREFIX):
|
|
|
|
# strip prefix from path
|
|
|
|
zids.append(path[ENTRYUUID_PREFIX_LEN:])
|
|
|
|
|
|
|
|
if len(zids) != 1:
|
|
|
|
raise ValueError('only IPA zones are supported: {}'.format(
|
|
|
|
etree.tostring(zone_xml)))
|
|
|
|
|
|
|
|
return name, zids[0]
|
2014-10-19 10:04:40 -05:00
|
|
|
|
|
|
|
|
|
|
|
class LDAPZoneListReader(ZoneListReader):
|
|
|
|
def __init__(self):
|
|
|
|
super(LDAPZoneListReader, self).__init__()
|
|
|
|
|
|
|
|
def process_ipa_zone(self, op, uuid, zone_ldap):
|
2018-07-11 15:30:12 -05:00
|
|
|
assert (op in ['add', 'del']), 'unsupported op %s' % op
|
2014-10-19 10:04:40 -05:00
|
|
|
assert uuid is not None
|
|
|
|
assert 'idnsname' in zone_ldap, \
|
|
|
|
'LDAP zone UUID %s without idnsName' % uuid
|
|
|
|
assert len(zone_ldap['idnsname']) == 1, \
|
|
|
|
'LDAP zone UUID %s with len(idnsname) != 1' % uuid
|
|
|
|
|
|
|
|
if op == 'add':
|
|
|
|
self._add_zone(zone_ldap['idnsname'][0], uuid)
|
|
|
|
elif op == 'del':
|
|
|
|
self._del_zone(zone_ldap['idnsname'][0], uuid)
|
|
|
|
|
|
|
|
|
2018-09-26 04:59:50 -05:00
|
|
|
class ODSMgr:
|
2014-10-19 10:04:40 -05:00
|
|
|
"""OpenDNSSEC zone manager. It does LDAP->ODS synchronization.
|
|
|
|
|
|
|
|
Zones with idnsSecInlineSigning attribute = TRUE in LDAP are added
|
|
|
|
or deleted from ODS as necessary. ODS->LDAP key synchronization
|
|
|
|
has to be solved seperatelly.
|
|
|
|
"""
|
|
|
|
def __init__(self):
|
|
|
|
self.zl_ldap = LDAPZoneListReader()
|
|
|
|
|
|
|
|
def ksmutil(self, params):
|
2019-04-17 07:14:26 -05:00
|
|
|
"""Call ods-ksmutil / ods-enforcer with parameters and return stdout.
|
2014-10-19 10:04:40 -05:00
|
|
|
|
|
|
|
Raises CalledProcessError if returncode != 0.
|
|
|
|
"""
|
2019-04-18 01:02:38 -05:00
|
|
|
result = tasks.run_ods_manager(params, capture_output=True)
|
2015-11-25 10:17:18 -06:00
|
|
|
return result.output
|
2014-10-19 10:04:40 -05:00
|
|
|
|
|
|
|
def get_ods_zonelist(self):
|
|
|
|
stdout = self.ksmutil(['zonelist', 'export'])
|
2020-03-05 00:23:02 -06:00
|
|
|
try:
|
|
|
|
reader = ODSZoneListReader(stdout)
|
|
|
|
except etree.ParseError:
|
|
|
|
# With OpenDNSSEC 2, the above command returns a message
|
|
|
|
# containing the zonelist filename instead of the XML text:
|
|
|
|
# "Exported zonelist to /etc/opendnssec/zonelist.xml successfully"
|
|
|
|
# extract the filename and read its content
|
|
|
|
pattern = re.compile(r'.* (/.*) .*')
|
|
|
|
matches = re.findall(pattern, stdout)
|
|
|
|
if matches:
|
|
|
|
with open(matches[0]) as f:
|
|
|
|
content = f.read()
|
|
|
|
reader = ODSZoneListReader(content)
|
|
|
|
|
2014-10-19 10:04:40 -05:00
|
|
|
return reader
|
|
|
|
|
|
|
|
def add_ods_zone(self, uuid, name):
|
|
|
|
zone_path = '%s%s' % (ENTRYUUID_PREFIX, uuid)
|
2020-03-05 08:54:40 -06:00
|
|
|
if name != dns.name.root:
|
|
|
|
name = name.relativize(dns.name.root)
|
2014-10-19 10:04:40 -05:00
|
|
|
cmd = ['zone', 'add', '--zone', str(name), '--input', zone_path]
|
2020-03-05 08:54:40 -06:00
|
|
|
output = None
|
|
|
|
try:
|
|
|
|
output = self.ksmutil(cmd)
|
|
|
|
except ipautil.CalledProcessError as e:
|
|
|
|
# Zone already exists in HSM
|
|
|
|
if e.returncode == 1 \
|
|
|
|
and str(e.output).endswith(str(name) + ' already exists!'):
|
|
|
|
# Just return
|
|
|
|
return
|
|
|
|
if output is not None:
|
|
|
|
logger.info('%s', output)
|
|
|
|
self.notify_enforcer()
|
2014-10-19 10:04:40 -05:00
|
|
|
|
|
|
|
def del_ods_zone(self, name):
|
|
|
|
# ods-ksmutil blows up if zone name has period at the end
|
2020-03-05 08:54:40 -06:00
|
|
|
if name != dns.name.root:
|
|
|
|
name = name.relativize(dns.name.root)
|
2015-01-21 05:19:17 -06:00
|
|
|
# detect if name is root zone
|
|
|
|
if name == dns.name.empty:
|
|
|
|
name = dns.name.root
|
2014-10-19 10:04:40 -05:00
|
|
|
cmd = ['zone', 'delete', '--zone', str(name)]
|
2020-03-05 08:54:40 -06:00
|
|
|
output = None
|
|
|
|
try:
|
|
|
|
output = self.ksmutil(cmd)
|
|
|
|
except ipautil.CalledProcessError as e:
|
|
|
|
# Zone already doesn't exist in HSM
|
|
|
|
if e.returncode == 1 \
|
|
|
|
and str(e.output).endswith(str(name) + ' not found!'):
|
|
|
|
# Just cleanup signer, no need to notify enforcer
|
|
|
|
self.cleanup_signer(name)
|
|
|
|
return
|
|
|
|
if output is not None:
|
|
|
|
logger.info('%s', output)
|
|
|
|
self.notify_enforcer()
|
|
|
|
self.cleanup_signer(name)
|
2014-10-19 10:04:40 -05:00
|
|
|
|
|
|
|
def notify_enforcer(self):
|
2020-03-05 08:54:40 -06:00
|
|
|
result = tasks.run_ods_notify(capture_output=True)
|
|
|
|
logger.info('%s', result.output)
|
2014-10-19 10:04:40 -05:00
|
|
|
|
2015-12-20 12:35:55 -06:00
|
|
|
def cleanup_signer(self, zone_name):
|
|
|
|
cmd = ['ods-signer', 'ldap-cleanup', str(zone_name)]
|
|
|
|
output = ipautil.run(cmd, capture_output=True)
|
2017-05-23 11:35:57 -05:00
|
|
|
logger.info('%s', output)
|
2015-12-20 12:35:55 -06:00
|
|
|
|
2014-10-19 10:04:40 -05:00
|
|
|
def ldap_event(self, op, uuid, attrs):
|
|
|
|
"""Record single LDAP event - zone addition or deletion.
|
|
|
|
|
|
|
|
Change is only recorded to memory.
|
|
|
|
self.sync() have to be called to synchronize change to ODS."""
|
2018-07-11 15:30:12 -05:00
|
|
|
assert op in ('add', 'del')
|
2014-10-19 10:04:40 -05:00
|
|
|
self.zl_ldap.process_ipa_zone(op, uuid, attrs)
|
2017-05-23 11:35:57 -05:00
|
|
|
logger.debug("LDAP zones: %s", self.zl_ldap.mapping)
|
2014-10-19 10:04:40 -05:00
|
|
|
|
|
|
|
def sync(self):
|
|
|
|
"""Synchronize list of zones in LDAP with ODS."""
|
|
|
|
zl_ods = self.get_ods_zonelist()
|
2017-05-23 11:35:57 -05:00
|
|
|
logger.debug("ODS zones: %s", zl_ods.mapping)
|
2014-10-19 10:04:40 -05:00
|
|
|
removed = self.diff_zl(zl_ods, self.zl_ldap)
|
2017-05-23 11:35:57 -05:00
|
|
|
logger.info("Zones removed from LDAP: %s", removed)
|
2014-10-19 10:04:40 -05:00
|
|
|
added = self.diff_zl(self.zl_ldap, zl_ods)
|
2017-05-23 11:35:57 -05:00
|
|
|
logger.info("Zones added to LDAP: %s", added)
|
2014-10-19 10:04:40 -05:00
|
|
|
for (uuid, name) in removed:
|
|
|
|
self.del_ods_zone(name)
|
|
|
|
for (uuid, name) in added:
|
|
|
|
self.add_ods_zone(uuid, name)
|
|
|
|
|
|
|
|
def diff_zl(self, s1, s2):
|
|
|
|
"""Compute zones present in s1 but not present in s2.
|
|
|
|
|
|
|
|
Returns: List of (uuid, name) tuples with zones present only in s1."""
|
|
|
|
s1_extra = s1.uuids - s2.uuids
|
|
|
|
removed = [(uuid, name) for (uuid, name) in s1.mapping.items()
|
|
|
|
if uuid in s1_extra]
|
|
|
|
return removed
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
ipa_log_manager.standard_logging_setup(debug=True)
|
|
|
|
ods = ODSMgr()
|
|
|
|
reader = ods.get_ods_zonelist()
|
2017-05-24 09:35:07 -05:00
|
|
|
logger.info('ODS zones: %s', reader.mapping)
|