pdns_change_tracker.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  1. import random
  2. import socket
  3. from django.conf import settings
  4. from django.db.models.signals import post_save, post_delete
  5. from django.db.transaction import atomic
  6. from django.utils import timezone
  7. from desecapi.exceptions import PDNSValidationError
  8. from desecapi.models import RRset, RR, Domain
  9. from desecapi.pdns import _pdns_post, NSLORD, NSMASTER, _pdns_delete, _pdns_patch, _pdns_put, pdns_id, \
  10. construct_catalog_rrset
  11. class PDNSChangeTracker:
  12. """
  13. Hooks up to model signals to maintain two sets:
  14. - `domain_additions`: set of added domains
  15. - `domain_deletions`: set of deleted domains
  16. The two sets are guaranteed to be disjoint.
  17. Hooks up to model signals to maintain exactly three sets per domain:
  18. - `rr_set_additions`: set of added RR sets
  19. - `rr_set_modifications`: set of modified RR sets
  20. - `rr_set_deletions`: set of deleted RR sets
  21. `additions` and `deletions` are guaranteed to be disjoint:
  22. - If an item is in the set of additions while being deleted, it is removed from `rr_set_additions`.
  23. - If an item is in the set of deletions while being added, it is removed from `rr_set_deletions`.
  24. `modifications` and `deletions` are guaranteed to be disjoint.
  25. - If an item is in the set of deletions while being modified, an exception is raised.
  26. - If an item is in the set of modifications while being deleted, it is removed from `rr_set_modifications`.
  27. Note every change tracker object will track all changes to the model across threading.
  28. To avoid side-effects, it is recommended that in each Django process, only one change
  29. tracker is run at a time, i.e. do not use them in parallel (e.g., in a multi-threading
  30. scenario), do not use them nested.
  31. """
  32. _active_change_trackers = 0
  33. class PDNSChange:
  34. """
  35. A reversible, atomic operation against the powerdns API.
  36. """
  37. def __init__(self, domain_name):
  38. self._domain_name = domain_name
  39. @property
  40. def domain_name(self):
  41. return self._domain_name
  42. @property
  43. def domain_name_normalized(self):
  44. return self._domain_name + '.'
  45. @property
  46. def domain_pdns_id(self):
  47. return pdns_id(self._domain_name)
  48. @property
  49. def axfr_required(self):
  50. raise NotImplementedError()
  51. def pdns_do(self):
  52. raise NotImplementedError()
  53. def api_do(self):
  54. raise NotImplementedError()
  55. def update_catalog(self, delete=False):
  56. return _pdns_patch(NSMASTER, '/zones/' + pdns_id(settings.CATALOG_ZONE),
  57. {'rrsets': [construct_catalog_rrset(zone=self.domain_name, delete=delete)]})
  58. class CreateDomain(PDNSChange):
  59. @property
  60. def axfr_required(self):
  61. return True
  62. def pdns_do(self):
  63. salt = '%016x' % random.randrange(16 ** 16)
  64. _pdns_post(
  65. NSLORD, '/zones',
  66. {
  67. 'name': self.domain_name_normalized,
  68. 'kind': 'MASTER',
  69. 'dnssec': True,
  70. 'nsec3param': '1 0 127 %s' % salt,
  71. 'nameservers': settings.DEFAULT_NS
  72. }
  73. )
  74. _pdns_post(
  75. NSMASTER, '/zones',
  76. {
  77. 'name': self.domain_name_normalized,
  78. 'kind': 'SLAVE',
  79. 'masters': [socket.gethostbyname('nslord')]
  80. }
  81. )
  82. self.update_catalog()
  83. def api_do(self):
  84. rr_set = RRset(
  85. domain=Domain.objects.get(name=self.domain_name),
  86. type='NS', subname='',
  87. ttl=settings.DEFAULT_NS_TTL,
  88. )
  89. rr_set.save()
  90. rrs = [RR(rrset=rr_set, content=ns) for ns in settings.DEFAULT_NS]
  91. RR.objects.bulk_create(rrs) # One INSERT
  92. def __str__(self):
  93. return 'Create Domain %s' % self.domain_name
  94. class DeleteDomain(PDNSChange):
  95. @property
  96. def axfr_required(self):
  97. return False
  98. def pdns_do(self):
  99. _pdns_delete(NSLORD, '/zones/' + self.domain_pdns_id)
  100. _pdns_delete(NSMASTER, '/zones/' + self.domain_pdns_id)
  101. self.update_catalog(delete=True)
  102. def api_do(self):
  103. pass
  104. def __str__(self):
  105. return 'Delete Domain %s' % self.domain_name
  106. class CreateUpdateDeleteRRSets(PDNSChange):
  107. def __init__(self, domain_name, additions, modifications, deletions):
  108. super().__init__(domain_name)
  109. self._additions = additions
  110. self._modifications = modifications
  111. self._deletions = deletions
  112. @property
  113. def axfr_required(self):
  114. return True
  115. def pdns_do(self):
  116. data = {
  117. 'rrsets':
  118. [
  119. {
  120. 'name': RRset.construct_name(subname, self._domain_name),
  121. 'type': type_,
  122. 'ttl': 1, # some meaningless integer required by pdns's syntax
  123. 'changetype': 'REPLACE', # don't use "DELETE" due to desec-stack#220, PowerDNS/pdns#7501
  124. 'records': []
  125. }
  126. for type_, subname in self._deletions
  127. ] + [
  128. {
  129. 'name': RRset.construct_name(subname, self._domain_name),
  130. 'type': type_,
  131. 'ttl': RRset.objects.values_list('ttl', flat=True).get(domain__name=self._domain_name,
  132. type=type_, subname=subname),
  133. 'changetype': 'REPLACE',
  134. 'records': [
  135. {'content': rr.content, 'disabled': False}
  136. for rr in RR.objects.filter(
  137. rrset__domain__name=self._domain_name,
  138. rrset__type=type_,
  139. rrset__subname=subname)
  140. ]
  141. }
  142. for type_, subname in (self._additions | self._modifications) - self._deletions
  143. ]
  144. }
  145. if data['rrsets']:
  146. _pdns_patch(NSLORD, '/zones/' + self.domain_pdns_id, data)
  147. def api_do(self):
  148. pass
  149. def __str__(self):
  150. return 'Update RRsets of %s: additions=%s, modifications=%s, deletions=%s' % \
  151. (self.domain_name, list(self._additions), list(self._modifications), list(self._deletions))
  152. def __init__(self):
  153. self._domain_additions = set()
  154. self._domain_deletions = set()
  155. self._rr_set_additions = {}
  156. self._rr_set_modifications = {}
  157. self._rr_set_deletions = {}
  158. self.transaction = None
  159. @classmethod
  160. def track(cls, f):
  161. """
  162. Execute function f with the change tracker.
  163. :param f: Function to be tracked for PDNS-relevant changes.
  164. :return: Returns the return value of f.
  165. """
  166. with cls():
  167. return f()
  168. def _manage_signals(self, method):
  169. if method not in ['connect', 'disconnect']:
  170. raise ValueError()
  171. getattr(post_save, method)(self._on_rr_post_save, sender=RR, dispatch_uid=self.__module__)
  172. getattr(post_delete, method)(self._on_rr_post_delete, sender=RR, dispatch_uid=self.__module__)
  173. getattr(post_save, method)(self._on_rr_set_post_save, sender=RRset, dispatch_uid=self.__module__)
  174. getattr(post_delete, method)(self._on_rr_set_post_delete, sender=RRset, dispatch_uid=self.__module__)
  175. getattr(post_save, method)(self._on_domain_post_save, sender=Domain, dispatch_uid=self.__module__)
  176. getattr(post_delete, method)(self._on_domain_post_delete, sender=Domain, dispatch_uid=self.__module__)
  177. def __enter__(self):
  178. PDNSChangeTracker._active_change_trackers += 1
  179. assert PDNSChangeTracker._active_change_trackers == 1, 'Nesting %s is not supported.' % self.__class__.__name__
  180. self._domain_additions = set()
  181. self._domain_deletions = set()
  182. self._rr_set_additions = {}
  183. self._rr_set_modifications = {}
  184. self._rr_set_deletions = {}
  185. self._manage_signals('connect')
  186. self.transaction = atomic()
  187. self.transaction.__enter__()
  188. def __exit__(self, exc_type, exc_val, exc_tb):
  189. PDNSChangeTracker._active_change_trackers -= 1
  190. self._manage_signals('disconnect')
  191. if exc_type:
  192. # An exception occurred inside our context, exit db transaction and dismiss pdns changes
  193. self.transaction.__exit__(exc_type, exc_val, exc_tb)
  194. return
  195. # TODO introduce two phase commit protocol
  196. changes = self._compute_changes()
  197. axfr_required = set()
  198. for change in changes:
  199. try:
  200. change.pdns_do()
  201. change.api_do()
  202. if change.axfr_required:
  203. axfr_required.add(change.domain_name)
  204. except PDNSValidationError as e:
  205. self.transaction.__exit__(type(e), e, e.__traceback__)
  206. raise e
  207. except Exception as e:
  208. self.transaction.__exit__(type(e), e, e.__traceback__)
  209. exc = ValueError(f'For changes {list(map(str, changes))}, {type(e)} occured when applying {change}')
  210. raise exc from e
  211. self.transaction.__exit__(None, None, None)
  212. for name in axfr_required:
  213. _pdns_put(NSMASTER, '/zones/%s/axfr-retrieve' % pdns_id(name))
  214. Domain.objects.filter(name__in=axfr_required).update(published=timezone.now())
  215. def _compute_changes(self):
  216. changes = []
  217. for domain_name in self._domain_deletions:
  218. # discard any RR set modifications
  219. self._rr_set_additions.pop(domain_name, None)
  220. self._rr_set_modifications.pop(domain_name, None)
  221. self._rr_set_deletions.pop(domain_name, None)
  222. changes.append(PDNSChangeTracker.DeleteDomain(domain_name))
  223. for domain_name in self._rr_set_additions.keys() | self._domain_additions:
  224. if domain_name in self._domain_additions:
  225. changes.append(PDNSChangeTracker.CreateDomain(domain_name))
  226. additions = self._rr_set_additions.get(domain_name, set())
  227. modifications = self._rr_set_modifications.get(domain_name, set())
  228. deletions = self._rr_set_deletions.get(domain_name, set())
  229. assert not (additions & deletions)
  230. assert not (modifications & deletions)
  231. # Due to disjoint guarantees with `deletions`, we have four types of RR sets:
  232. # (1) purely added RR sets
  233. # (2) purely modified RR sets
  234. # (3) added and modified RR sets
  235. # (4) purely deleted RR sets
  236. # We send RR sets to PDNS if one of the following conditions holds:
  237. # (a) RR set was added and has at least one RR
  238. # (b) RR set was modified
  239. # (c) RR set was deleted
  240. # Conditions (b) and (c) are already covered in the modifications and deletions list,
  241. # we filter the additions list to remove newly-added, but empty RR sets
  242. additions -= {
  243. (type_, subname) for (type_, subname) in additions
  244. if not RR.objects.filter(
  245. rrset__domain__name=domain_name,
  246. rrset__type=type_,
  247. rrset__subname=subname).exists()
  248. }
  249. if additions | modifications | deletions:
  250. changes.append(PDNSChangeTracker.CreateUpdateDeleteRRSets(
  251. domain_name, additions, modifications, deletions))
  252. return changes
  253. def _rr_set_updated(self, rr_set: RRset, deleted=False, created=False):
  254. if self._rr_set_modifications.get(rr_set.domain.name, None) is None:
  255. self._rr_set_additions[rr_set.domain.name] = set()
  256. self._rr_set_modifications[rr_set.domain.name] = set()
  257. self._rr_set_deletions[rr_set.domain.name] = set()
  258. additions = self._rr_set_additions[rr_set.domain.name]
  259. modifications = self._rr_set_modifications[rr_set.domain.name]
  260. deletions = self._rr_set_deletions[rr_set.domain.name]
  261. item = (rr_set.type, rr_set.subname)
  262. if created:
  263. additions.add(item)
  264. assert item not in modifications
  265. deletions.discard(item)
  266. elif deleted:
  267. if item in additions:
  268. additions.remove(item)
  269. modifications.discard(item)
  270. # no change to deletions
  271. else:
  272. # item not in additions
  273. modifications.discard(item)
  274. deletions.add(item)
  275. elif not created and not deleted:
  276. # we don't care if item was created or not
  277. modifications.add(item)
  278. assert item not in deletions
  279. else:
  280. raise ValueError('An RR set cannot be created and deleted at the same time.')
  281. def _domain_updated(self, domain: Domain, created=False, deleted=False):
  282. if not created and not deleted:
  283. # NOTE that the name must not be changed by API contract with models, hence here no-op for pdns.
  284. return
  285. name = domain.name
  286. additions = self._domain_additions
  287. deletions = self._domain_deletions
  288. if created and deleted:
  289. raise ValueError('A domain set cannot be created and deleted at the same time.')
  290. if created:
  291. if name in deletions:
  292. deletions.remove(name)
  293. else:
  294. additions.add(name)
  295. elif deleted:
  296. if name in additions:
  297. additions.remove(name)
  298. else:
  299. deletions.add(name)
  300. # noinspection PyUnusedLocal
  301. def _on_rr_post_save(self, signal, sender, instance: RR, created, update_fields, raw, using, **kwargs):
  302. self._rr_set_updated(instance.rrset)
  303. # noinspection PyUnusedLocal
  304. def _on_rr_post_delete(self, signal, sender, instance: RR, using, **kwargs):
  305. self._rr_set_updated(instance.rrset)
  306. # noinspection PyUnusedLocal
  307. def _on_rr_set_post_save(self, signal, sender, instance: RRset, created, update_fields, raw, using, **kwargs):
  308. self._rr_set_updated(instance, created=created)
  309. # noinspection PyUnusedLocal
  310. def _on_rr_set_post_delete(self, signal, sender, instance: RRset, using, **kwargs):
  311. self._rr_set_updated(instance, deleted=True)
  312. # noinspection PyUnusedLocal
  313. def _on_domain_post_save(self, signal, sender, instance: Domain, created, update_fields, raw, using, **kwargs):
  314. self._domain_updated(instance, created=created)
  315. # noinspection PyUnusedLocal
  316. def _on_domain_post_delete(self, signal, sender, instance: Domain, using, **kwargs):
  317. self._domain_updated(instance, deleted=True)
  318. def __str__(self):
  319. all_rr_sets = self._rr_set_additions.keys() | self._rr_set_modifications.keys() | self._rr_set_deletions.keys()
  320. all_domains = self._domain_additions | self._domain_deletions
  321. return '<%s: %i added or deleted domains; %i added, modified or deleted RR sets>' % (
  322. self.__class__.__name__,
  323. len(all_domains),
  324. len(all_rr_sets)
  325. )