pdns_change_tracker.py 16 KB

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