pdns_change_tracker.py 17 KB

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