# # # from __future__ import absolute_import, division, print_function, \ unicode_literals from ..source.base import BaseSource from ..zone import Zone class UnsafePlan(Exception): pass class Plan(object): MAX_SAFE_UPDATES = 4 MAX_SAFE_DELETES = 4 def __init__(self, existing, desired, changes): self.existing = existing self.desired = desired self.changes = changes change_counts = { 'Create': 0, 'Delete': 0, 'Update': 0 } for change in changes: change_counts[change.__class__.__name__] += 1 self.change_counts = change_counts def raise_if_unsafe(self): # TODO: what is safe really? if self.change_counts['Update'] > self.MAX_SAFE_UPDATES: raise UnsafePlan('Too many updates') if self.change_counts['Delete'] > self.MAX_SAFE_DELETES: raise UnsafePlan('Too many deletes') def __repr__(self): return 'Creates={}, Updates={}, Deletes={}, Existing Records={}' \ .format(self.change_counts['Create'], self.change_counts['Update'], self.change_counts['Delete'], len(self.existing.records)) class BaseProvider(BaseSource): def __init__(self, id, apply_disabled=False): super(BaseProvider, self).__init__(id) self.log.debug('__init__: id=%s, apply_disabled=%s', id, apply_disabled) self.apply_disabled = apply_disabled def _include_change(self, change): ''' An opportunity for providers to filter out false positives due to pecularities in their implementation. E.g. minimum TTLs. ''' return True def _extra_changes(self, existing, changes): ''' An opportunity for providers to add extra changes to the plan that are necessary to update ancilary record data or configure the zone. E.g. base NS records. ''' return [] def plan(self, desired): self.log.info('plan: desired=%s', desired.name) existing = Zone(desired.name, desired.sub_zones) self.populate(existing, target=True) # compute the changes at the zone/record level changes = existing.changes(desired, self) # allow the provider to filter out false positives before = len(changes) changes = filter(self._include_change, changes) after = len(changes) if before != after: self.log.info('plan: filtered out %s changes', before - after) # allow the provider to add extra changes it needs extra = self._extra_changes(existing, changes) if extra: self.log.info('plan: extra changes\n %s', '\n ' .join([str(c) for c in extra])) changes += extra if changes: plan = Plan(existing, desired, changes) self.log.info('plan: %s', plan) return plan self.log.info('plan: No changes') return None def apply(self, plan): ''' Submits actual planned changes to the provider. Returns the number of changes made ''' if self.apply_disabled: self.log.info('apply: disabled') return 0 self.log.info('apply: making changes') self._apply(plan) return len(plan.changes) def _apply(self, plan): raise NotImplementedError('Abstract base class, _apply method ' 'missing')