diff options
author | Martin Basti <mbasti@redhat.com> | 2014-06-13 10:20:23 +0200 |
---|---|---|
committer | Martin Kosek <mkosek@redhat.com> | 2014-06-27 14:54:35 +0200 |
commit | aa2ef07b8c74d05e8a83928dab4b5cae8722a52f (patch) | |
tree | 580cceb3258b1b88de08499c7a53f1c3c78cc652 /ipaserver | |
parent | c1f3fd6831c47a2672f37e12db149be1b1122d75 (diff) | |
download | freeipa-aa2ef07b8c74d05e8a83928dab4b5cae8722a52f.tar.gz freeipa-aa2ef07b8c74d05e8a83928dab4b5cae8722a52f.tar.xz freeipa-aa2ef07b8c74d05e8a83928dab4b5cae8722a52f.zip |
Upgrade special master zones to forward zones
This upgrade is executed only if IPA version is older than 4.0
Requires detection if 'idnsforwardzone' objectclass is presented in
schema before schema is upgraded
Design: http://www.freeipa.org/page/V4/Forward_zones#Updates_and_Upgrades
Ticket: https://fedorahosted.org/freeipa/ticket/3210
Reviewed-By: Martin Kosek <mkosek@redhat.com>
Diffstat (limited to 'ipaserver')
-rw-r--r-- | ipaserver/install/plugins/dns.py | 179 |
1 files changed, 177 insertions, 2 deletions
diff --git a/ipaserver/install/plugins/dns.py b/ipaserver/install/plugins/dns.py index 76c57f2f0..c45457eae 100644 --- a/ipaserver/install/plugins/dns.py +++ b/ipaserver/install/plugins/dns.py @@ -17,13 +17,22 @@ # You should have received a copy of the GNU General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. -from ipaserver.install.plugins import MIDDLE -from ipaserver.install.plugins.baseupdate import PostUpdate +import ldap as _ldap +import traceback +import time + +from ldif import LDIFWriter + +from ipaserver.install.plugins import MIDDLE, LAST +from ipaserver.install.plugins.baseupdate import (PostUpdate, PreUpdate, + PreSchemaUpdate) +from ipaserver.install import sysupgrade from ipalib import api, errors, util from ipapython.dn import DN from ipalib.plugins.dns import dns_container_exists from ipapython.ipa_log_manager import * + class update_dnszones(PostUpdate): """ Update all zones to meet requirements in the new FreeIPA versions @@ -129,3 +138,169 @@ class update_dns_limits(PostUpdate): return (False, True, [dnsupdates]) api.register(update_dns_limits) + + +class update_check_forwardzones(PreSchemaUpdate): + """ + Check if the idnsforwardzone objectclass is in LDAP schema. + If not update is required (update_to_forward_zones), set sysupgrade state + 'update_to_forward_zones' to True + """ + + def execute(self, **options): + state = sysupgrade.get_upgrade_state('dns', 'update_to_forward_zones') + if state is False: + # no upgrade is needed + return (False, False, []) + ldap = self.obj.backend + result = ldap.schema.get_obj(_ldap.schema.models.ObjectClass, 'idnsforwardzone') + if result is None: + sysupgrade.set_upgrade_state('dns', 'update_to_forward_zones', True) + self.log.info('Prepared upgrade to forward zones') + else: + sysupgrade.set_upgrade_state('dns', 'update_to_forward_zones', False) + return (False, False, []) + +api.register(update_check_forwardzones) + + +class update_master_to_dnsforwardzones(PostUpdate): + """ + Update all zones to meet requirements in the new FreeIPA versions + + All masters zones with specified forwarders, and forward-policy different + than none, will be tranformed to forward zones. + Original masters zone will be backed up to ldif file. + + This should be applied only once, and only if original version was lower than 4.0 + """ + order = LAST + + backup_dir = u'/var/lib/ipa/backup/' + backup_filename = u'dns-forward-zones-backup-%Y-%m-%d-%H-%M-%S.ldif' + backup_path = u'%s%s' % (backup_dir, backup_filename) + + def execute(self, **options): + ldap = self.obj.backend + if not sysupgrade.get_upgrade_state('dns', 'update_to_forward_zones'): + # forward zones was tranformed before, nothing to do + return (False, False, []) + + try: + # raw values are required to store into ldif + zones = api.Command.dnszone_find(all=True, + raw=True, + sizelimit=0)['result'] + except errors.NotFound: + self.log.info('No DNS zone to update found') + return (False, False, []) + + zones_to_transform = [] + + for zone in zones: + if ( + zone.get('idnsforwardpolicy', [u'first'])[0] == u'none' or + zone.get('idnsforwarders', []) == [] + ): + continue # don't update zone + + zones_to_transform.append(zone) + + if zones_to_transform: + # add time to filename + self.backup_path = time.strftime(self.backup_path) + + self.log.info('Zones with specified forwarders with policy different' + ' than none will be transformed to forward zones.') + self.log.info('Original zones will be saved in LDIF format in ' + '%s file' % self.backup_path) + try: + + with open(self.backup_path, 'w') as f: + writer = LDIFWriter(f) + for zone in zones_to_transform: + # save backup to ldif + try: + + dn = str(zone['dn']) + del zone['dn'] # dn shouldn't be as attribute in ldif + writer.unparse(dn, zone) + + if 'managedBy' in zone: + entry = ldap.get_entry(DN(zone['managedBy'][0])) + writer.unparse(str(entry.dn), dict(entry)) + + # raw values are required to store into ldif + records = api.Command['dnsrecord_find']( + zone['idnsname'][0], + all=True, + raw=True, + sizelimit=0)['result'] + for record in records: + if record['idnsname'][0] == u'@': + # zone record was saved before + continue + dn = str(record['dn']) + del record['dn'] + writer.unparse(dn, record) + + except Exception, e: + self.log.error('Unable to backup zone %s' % + zone['idnsname'][0]) + self.log.error(traceback.format_exc()) + return (False, False, []) + f.close() + except Exception: + self.log.error('Unable to create backup file') + self.log.error(traceback.format_exc()) + return (False, False, []) + + # update + for zone in zones_to_transform: + # delete master zone + try: + api.Command['dnszone_del'](zone['idnsname']) + except Exception, e: + self.log.error('Transform to forwardzone terminated: ' + 'removing zone %s failed (%s)' % ( + zone['idnsname'][0], e) + ) + self.log.error(traceback.format_exc()) + continue + + # create forward zone + try: + kw = { + 'idnsforwarders': zone.get('idnsforwarders', []), + 'idnsforwardpolicy': zone.get('idnsforwardpolicy', [u'first'])[0] + } + api.Command['dnsforwardzone_add'](zone['idnsname'][0], **kw) + except Exception, e: + self.log.error('Transform to forwardzone terminated: creating ' + 'forwardzone %s failed' % + zone['idnsname'][0]) + self.log.error(traceback.format_exc()) + continue + + # create permission if original zone has one + if 'managedBy' in zone: + try: + api.Command['dnsforwardzone_add_permission'](zone['idnsname'][0]) + except Exception, e: + self.log.error('Transform to forwardzone terminated: ' + 'Adding managed by permission to forward zone' + ' %s failed' % zone['idnsname']) + self.log.error(traceback.format_exc()) + self.log.info('Zone %s was transformed to forward zone ' + ' without managed permissions', + zone['idnsname'][0]) + continue + + self.log.info('Zone %s was sucessfully transformed to forward zone', + zone['idnsname'][0]) + + sysupgrade.set_upgrade_state('dns', 'update_to_forward_zones', False) + + return (False, False, []) + +api.register(update_master_to_dnsforwardzones) |