test_rrsets_bulk.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. import copy
  2. from django.conf import settings
  3. from rest_framework import status
  4. from desecapi.tests.base import AuthenticatedRRSetBaseTestCase
  5. class AuthenticatedRRSetBulkTestCase(AuthenticatedRRSetBaseTestCase):
  6. @classmethod
  7. def setUpTestDataWithPdns(cls):
  8. super().setUpTestDataWithPdns()
  9. cls.data = [
  10. {'subname': 'my-bulk', 'records': ['1.2.3.4'], 'ttl': 3600, 'type': 'A'},
  11. {'subname': 'my-bulk', 'records': ['desec.io.', 'foobar.example.'], 'ttl': 3600, 'type': 'PTR'},
  12. ]
  13. cls.data_no_records = copy.deepcopy(cls.data)
  14. cls.data_no_records[1].pop('records')
  15. cls.data_empty_records = copy.deepcopy(cls.data)
  16. cls.data_empty_records[1]['records'] = []
  17. cls.data_no_subname = copy.deepcopy(cls.data)
  18. cls.data_no_subname[0].pop('subname')
  19. cls.data_no_ttl = copy.deepcopy(cls.data)
  20. cls.data_no_ttl[0].pop('ttl')
  21. cls.data_no_type = copy.deepcopy(cls.data)
  22. cls.data_no_type[1].pop('type')
  23. cls.data_no_records_no_ttl = copy.deepcopy(cls.data_no_records)
  24. cls.data_no_records_no_ttl[1].pop('ttl')
  25. cls.data_no_subname_empty_records = copy.deepcopy(cls.data_no_subname)
  26. cls.data_no_subname_empty_records[0]['records'] = []
  27. cls.bulk_domain = cls.create_domain(owner=cls.owner)
  28. for data in cls.data:
  29. cls.create_rr_set(cls.bulk_domain, **data)
  30. def test_bulk_post_my_rr_sets(self):
  31. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.my_empty_domain.name)):
  32. response = self.client.bulk_post_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data)
  33. self.assertStatus(response, status.HTTP_201_CREATED)
  34. response = self.client.get_rr_sets(self.my_empty_domain.name)
  35. self.assertStatus(response, status.HTTP_200_OK)
  36. self.assertRRSetsCount(response.data, self.data)
  37. # Check subname requirement on bulk endpoint (and uniqueness at the same time)
  38. self.assertResponse(
  39. self.client.bulk_post_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data_no_subname),
  40. status.HTTP_400_BAD_REQUEST,
  41. [
  42. {'subname': ['This field is required.']},
  43. {'non_field_errors': ['Another RRset with the same subdomain and type exists for this domain.']}
  44. ]
  45. )
  46. def test_bulk_post_rr_sets_empty_records(self):
  47. expected_response_data = [copy.deepcopy(self.data_empty_records[0]), None]
  48. expected_response_data[0]['domain'] = self.my_empty_domain.name
  49. expected_response_data[0]['name'] = '%s.%s.' % (self.data_empty_records[0]['subname'],
  50. self.my_empty_domain.name)
  51. self.assertResponse(
  52. self.client.bulk_post_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data_empty_records),
  53. status.HTTP_400_BAD_REQUEST,
  54. [
  55. {},
  56. {'records': ['This field must not be empty when using POST.']}
  57. ]
  58. )
  59. def test_bulk_post_existing_rrsets(self):
  60. self.assertResponse(
  61. self.client.bulk_post_rr_sets(
  62. domain_name=self.bulk_domain,
  63. payload=self.data,
  64. ),
  65. status.HTTP_400_BAD_REQUEST,
  66. 2 * [{
  67. 'non_field_errors': ['Another RRset with the same subdomain and type exists for this domain.']
  68. }]
  69. )
  70. def test_bulk_post_duplicates(self):
  71. data = 2 * [self.data[0]] + [self.data[1]]
  72. self.assertResponse(
  73. self.client.bulk_post_rr_sets(domain_name=self.my_empty_domain.name, payload=data),
  74. status.HTTP_400_BAD_REQUEST,
  75. [
  76. {'non_field_errors': ['Same subname and type as in position(s) 1, but must be unique.']},
  77. {'non_field_errors': ['Same subname and type as in position(s) 0, but must be unique.']},
  78. {},
  79. ]
  80. )
  81. data = 2 * [self.data[0]] + [self.data[1]] + [self.data[0]]
  82. self.assertResponse(
  83. self.client.bulk_post_rr_sets(domain_name=self.my_empty_domain.name, payload=data),
  84. status.HTTP_400_BAD_REQUEST,
  85. [
  86. {'non_field_errors': ['Same subname and type as in position(s) 1, 3, but must be unique.']},
  87. {'non_field_errors': ['Same subname and type as in position(s) 0, 3, but must be unique.']},
  88. {},
  89. {'non_field_errors': ['Same subname and type as in position(s) 0, 1, but must be unique.']},
  90. ]
  91. )
  92. def test_bulk_post_missing_fields(self):
  93. self.assertResponse(
  94. self.client.bulk_post_rr_sets(
  95. domain_name=self.my_empty_domain.name,
  96. payload=[
  97. {'subname': 'a.1', 'records': ['dead::beef'], 'ttl': 3622},
  98. {'subname': 'b.1', 'ttl': -50, 'type': 'AAAA', 'records': ['dead::beef']},
  99. {'ttl': 3640, 'type': 'TXT', 'records': ['"bar"']},
  100. {'subname': '', 'ttl': 3640, 'type': 'TXT', 'records': ['"bar"']},
  101. {'subname': 'c.1', 'records': ['dead::beef'], 'type': 'AAAA'},
  102. {'subname': 'd.1', 'ttl': 3650, 'type': 'AAAA'},
  103. {'subname': 'd.1', 'ttl': 3650, 'type': 'SOA',
  104. 'records': ['set.an.example. get.desec.io. 2018034419 10800 3600 604800 60']},
  105. {'subname': 'd.1', 'ttl': 3650, 'type': 'OPT', 'records': ['9999']},
  106. {'subname': 'd.1', 'ttl': 3650, 'type': 'TYPE099', 'records': ['v=spf1 mx -all']},
  107. ]
  108. ),
  109. status.HTTP_400_BAD_REQUEST,
  110. [
  111. {'type': ['This field is required.']},
  112. {'ttl': [f'Ensure this value is greater than or equal to {settings.MINIMUM_TTL_DEFAULT}.']},
  113. {'subname': ['This field is required.']},
  114. {},
  115. {'ttl': ['This field is required.']},
  116. {'records': ['This field is required.']},
  117. {'type': ['You cannot tinker with the SOA RRset.']},
  118. {'type': ['You cannot tinker with the OPT RRset.']},
  119. {'type': ['Generic type format is not supported.']},
  120. ]
  121. )
  122. def test_bulk_patch_fresh_rrsets_need_records(self):
  123. response = self.client.bulk_patch_rr_sets(self.my_empty_domain.name, payload=self.data_no_records)
  124. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  125. self.assertEqual(response.data, [{}, {'records': ['This field is required.']}])
  126. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(self.my_empty_domain.name)):
  127. response = self.client.bulk_patch_rr_sets(self.my_empty_domain.name, payload=self.data_empty_records)
  128. self.assertStatus(response, status.HTTP_200_OK)
  129. def test_bulk_patch_fresh_rrsets_need_subname(self):
  130. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data_no_subname)
  131. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  132. def test_bulk_patch_fresh_rrsets_need_ttl(self):
  133. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data_no_ttl)
  134. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  135. self.assertEqual(response.data, [{'ttl': ['This field is required.']}, {}])
  136. def test_bulk_patch_fresh_rrsets_need_type(self):
  137. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data_no_type)
  138. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  139. self.assertEqual(response.data, [{}, {'type': ['This field is required.']}])
  140. def test_bulk_patch_does_not_accept_single_objects(self):
  141. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data[0])
  142. self.assertContains(response, 'Expected a list of items but got dict.', status_code=status.HTTP_400_BAD_REQUEST)
  143. def test_bulk_patch_full_on_empty_domain(self):
  144. # Full patch always works
  145. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.my_empty_domain.name)):
  146. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data)
  147. self.assertStatus(response, status.HTTP_200_OK)
  148. # Check that RRsets have been created
  149. response = self.client.get_rr_sets(self.my_empty_domain.name)
  150. self.assertStatus(response, status.HTTP_200_OK)
  151. self.assertRRSetsCount(response.data, self.data)
  152. def test_bulk_patch_change_records(self):
  153. data_no_ttl = copy.deepcopy(self.data_no_ttl)
  154. data_no_ttl[0]['records'] = ['4.3.2.1', '8.8.1.2']
  155. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.bulk_domain.name)):
  156. response = self.client.bulk_patch_rr_sets(domain_name=self.bulk_domain.name, payload=data_no_ttl)
  157. self.assertStatus(response, status.HTTP_200_OK)
  158. response = self.client.get_rr_sets(self.bulk_domain.name)
  159. self.assertStatus(response, status.HTTP_200_OK)
  160. self.assertRRSetsCount(response.data, data_no_ttl)
  161. def test_bulk_patch_change_ttl(self):
  162. data_no_records = copy.deepcopy(self.data_no_records)
  163. data_no_records[1]['ttl'] = 3911
  164. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.bulk_domain.name)):
  165. response = self.client.bulk_patch_rr_sets(domain_name=self.bulk_domain.name, payload=data_no_records)
  166. self.assertStatus(response, status.HTTP_200_OK)
  167. response = self.client.get_rr_sets(self.bulk_domain.name)
  168. self.assertStatus(response, status.HTTP_200_OK)
  169. self.assertRRSetsCount(response.data, data_no_records)
  170. def test_bulk_patch_does_not_need_ttl(self):
  171. self.assertResponse(
  172. self.client.bulk_patch_rr_sets(domain_name=self.bulk_domain.name, payload=self.data_no_ttl),
  173. status.HTTP_200_OK,
  174. )
  175. def test_bulk_patch_delete_non_existing_rr_sets(self):
  176. self.assertResponse(
  177. self.client.bulk_patch_rr_sets(
  178. domain_name=self.my_empty_domain.name,
  179. payload=[
  180. {'subname': 'a', 'type': 'A', 'records': [], 'ttl': 3622},
  181. {'subname': 'b', 'type': 'AAAA', 'records': []},
  182. ]),
  183. status.HTTP_200_OK,
  184. [],
  185. )
  186. def test_bulk_patch_missing_invalid_fields_1(self):
  187. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(self.my_empty_domain.name)):
  188. self.client.bulk_post_rr_sets(
  189. domain_name=self.my_empty_domain.name,
  190. payload=[
  191. {'subname': '', 'ttl': 3650, 'type': 'TXT', 'records': ['"foo"']},
  192. {'subname': 'c.1', 'records': ['dead::beef'], 'type': 'AAAA', 'ttl': 3603},
  193. {'subname': 'd.1', 'ttl': 3650, 'type': 'AAAA', 'records': ['::1', '::2']},
  194. ]
  195. )
  196. self.assertResponse(
  197. self.client.bulk_patch_rr_sets(
  198. domain_name=self.my_empty_domain.name,
  199. payload=[
  200. {'subname': 'a.1', 'records': ['dead::beef'], 'ttl': 3622},
  201. {'subname': 'b.1', 'ttl': -50, 'type': 'AAAA', 'records': ['dead::beef']},
  202. {'ttl': 3640, 'type': 'TXT', 'records': ['"bar"']},
  203. {'subname': 'c.1', 'records': ['dead::beef'], 'type': 'AAAA'},
  204. {'subname': 'd.1', 'ttl': 3650, 'type': 'AAAA'},
  205. ]),
  206. status.HTTP_400_BAD_REQUEST,
  207. [
  208. {'type': ['This field is required.']},
  209. {'ttl': [f'Ensure this value is greater than or equal to {settings.MINIMUM_TTL_DEFAULT}.']},
  210. {'subname': ['This field is required.']},
  211. {},
  212. {},
  213. ]
  214. )
  215. def test_bulk_patch_missing_invalid_fields_2(self):
  216. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(self.my_empty_domain.name)):
  217. self.client.bulk_post_rr_sets(
  218. domain_name=self.my_empty_domain.name,
  219. payload=[
  220. {'subname': '', 'ttl': 3650, 'type': 'TXT', 'records': ['"foo"']}
  221. ]
  222. )
  223. self.assertResponse(
  224. self.client.bulk_patch_rr_sets(
  225. domain_name=self.my_empty_domain.name,
  226. payload=[
  227. {'subname': 'a.1', 'records': ['dead::beef'], 'ttl': 3622},
  228. {'subname': 'b.1', 'ttl': -50, 'type': 'AAAA', 'records': ['dead::beef']},
  229. {'ttl': 3640, 'type': 'TXT', 'records': ['"bar"']},
  230. {'subname': 'c.1', 'records': ['dead::beef'], 'type': 'AAAA'},
  231. {'subname': 'd.1', 'ttl': 3650, 'type': 'AAAA'},
  232. ]),
  233. status.HTTP_400_BAD_REQUEST,
  234. [
  235. {'type': ['This field is required.']},
  236. {'ttl': [f'Ensure this value is greater than or equal to {settings.MINIMUM_TTL_DEFAULT}.']},
  237. {'subname': ['This field is required.']},
  238. {'ttl': ['This field is required.']},
  239. {'records': ['This field is required.']},
  240. ]
  241. )
  242. def test_bulk_put_partial(self):
  243. # Need all fields
  244. for domain in [self.my_empty_domain, self.bulk_domain]:
  245. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_records)
  246. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  247. self.assertEqual(response.data, [{}, {'records': ['This field is required.']}])
  248. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_ttl)
  249. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  250. self.assertEqual(response.data, [{'ttl': ['This field is required.']}, {}])
  251. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_records_no_ttl)
  252. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  253. self.assertEqual(response.data, [{},
  254. {'ttl': ['This field is required.'],
  255. 'records': ['This field is required.']}])
  256. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_subname)
  257. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  258. self.assertEqual(response.data, [{'subname': ['This field is required.']}, {}])
  259. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_subname_empty_records)
  260. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  261. self.assertEqual(response.data, [{'subname': ['This field is required.']}, {}])
  262. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_type)
  263. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  264. self.assertEqual(response.data, [{}, {'type': ['This field is required.']}])
  265. def test_bulk_put_does_not_accept_single_objects(self):
  266. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data[0])
  267. self.assertContains(response, 'Expected a list of items but got dict.', status_code=status.HTTP_400_BAD_REQUEST)
  268. def test_bulk_put_does_not_accept_list_of_crap(self):
  269. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=['bla'])
  270. self.assertContains(response, 'Expected a dictionary, but got str.', status_code=status.HTTP_400_BAD_REQUEST)
  271. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=[42])
  272. self.assertContains(response, 'Expected a dictionary, but got int.', status_code=status.HTTP_400_BAD_REQUEST)
  273. def test_bulk_put_full(self):
  274. # Full PUT always works
  275. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.my_empty_domain.name)):
  276. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data)
  277. self.assertStatus(response, status.HTTP_200_OK)
  278. # Check that RRsets have been created
  279. response = self.client.get_rr_sets(self.my_empty_domain.name)
  280. self.assertStatus(response, status.HTTP_200_OK)
  281. self.assertRRSetsCount(response.data, self.data)
  282. # Do not expect any updates, but successful code when PUT'ing only existing RRsets
  283. response = self.client.bulk_put_rr_sets(domain_name=self.bulk_domain.name, payload=self.data)
  284. self.assertStatus(response, status.HTTP_200_OK)
  285. def test_bulk_put_invalid_records(self):
  286. for records in [
  287. 'asfd',
  288. ['1.1.1.1', '2.2.2.2', 123],
  289. ['1.2.3.4', None],
  290. [True, '1.1.1.1'],
  291. dict(foobar='foobar', asdf='asdf'),
  292. ]:
  293. s = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=[
  294. {'subname': 'a.2', 'ttl': 50, 'type': 'MX', 'records': records}
  295. ])
  296. self.assertStatus(
  297. s,
  298. status.HTTP_400_BAD_REQUEST
  299. )
  300. def test_bulk_put_empty_records(self):
  301. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.bulk_domain.name)):
  302. self.assertStatus(
  303. self.client.bulk_put_rr_sets(domain_name=self.bulk_domain.name, payload=self.data_empty_records),
  304. status.HTTP_200_OK
  305. )
  306. def test_bulk_duplicate_rrset(self):
  307. data = self.data + self.data
  308. for bulk_request_rr_sets in [
  309. self.client.bulk_patch_rr_sets,
  310. self.client.bulk_put_rr_sets,
  311. self.client.bulk_post_rr_sets,
  312. ]:
  313. response = bulk_request_rr_sets(domain_name=self.my_empty_domain.name, payload=data)
  314. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  315. def test_bulk_patch_or_post_failure_with_single_rrset(self):
  316. for method in [self.client.bulk_patch_rr_sets, self.client.bulk_put_rr_sets]:
  317. response = method(domain_name=self.my_empty_domain.name, payload=self.data[0])
  318. self.assertContains(response, 'Expected a list of items but got dict.',
  319. status_code=status.HTTP_400_BAD_REQUEST)