test_rrsets_bulk.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401
  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_post_accepts_empty_list(self):
  123. self.assertResponse(
  124. self.client.bulk_post_rr_sets(domain_name=self.my_empty_domain.name, payload=[]),
  125. status.HTTP_201_CREATED,
  126. )
  127. def test_bulk_patch_fresh_rrsets_need_records(self):
  128. response = self.client.bulk_patch_rr_sets(self.my_empty_domain.name, payload=self.data_no_records)
  129. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  130. self.assertEqual(response.data, [{}, {'records': ['This field is required.']}])
  131. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(self.my_empty_domain.name)):
  132. response = self.client.bulk_patch_rr_sets(self.my_empty_domain.name, payload=self.data_empty_records)
  133. self.assertStatus(response, status.HTTP_200_OK)
  134. def test_bulk_patch_fresh_rrsets_need_subname(self):
  135. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data_no_subname)
  136. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  137. def test_bulk_patch_fresh_rrsets_need_ttl(self):
  138. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data_no_ttl)
  139. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  140. self.assertEqual(response.data, [{'ttl': ['This field is required.']}, {}])
  141. def test_bulk_patch_fresh_rrsets_need_type(self):
  142. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data_no_type)
  143. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  144. self.assertEqual(response.data, [{}, {'type': ['This field is required.']}])
  145. def test_bulk_patch_does_not_accept_single_objects(self):
  146. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data[0])
  147. self.assertContains(response, 'Expected a list of items but got dict.', status_code=status.HTTP_400_BAD_REQUEST)
  148. def test_bulk_patch_does_accept_empty_list(self):
  149. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=[])
  150. self.assertStatus(response, status.HTTP_200_OK)
  151. def test_bulk_patch_does_not_accept_empty_payload(self):
  152. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=None)
  153. self.assertContains(response, 'No data provided', status_code=status.HTTP_400_BAD_REQUEST)
  154. def test_bulk_patch_full_on_empty_domain(self):
  155. # Full patch always works
  156. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.my_empty_domain.name)):
  157. response = self.client.bulk_patch_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data)
  158. self.assertStatus(response, status.HTTP_200_OK)
  159. # Check that RRsets have been created
  160. response = self.client.get_rr_sets(self.my_empty_domain.name)
  161. self.assertStatus(response, status.HTTP_200_OK)
  162. self.assertRRSetsCount(response.data, self.data)
  163. def test_bulk_patch_change_records(self):
  164. data_no_ttl = copy.deepcopy(self.data_no_ttl)
  165. data_no_ttl[0]['records'] = ['4.3.2.1', '8.8.1.2']
  166. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.bulk_domain.name)):
  167. response = self.client.bulk_patch_rr_sets(domain_name=self.bulk_domain.name, payload=data_no_ttl)
  168. self.assertStatus(response, status.HTTP_200_OK)
  169. response = self.client.get_rr_sets(self.bulk_domain.name)
  170. self.assertStatus(response, status.HTTP_200_OK)
  171. self.assertRRSetsCount(response.data, data_no_ttl)
  172. def test_bulk_patch_change_ttl(self):
  173. data_no_records = copy.deepcopy(self.data_no_records)
  174. data_no_records[1]['ttl'] = 3911
  175. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.bulk_domain.name)):
  176. response = self.client.bulk_patch_rr_sets(domain_name=self.bulk_domain.name, payload=data_no_records)
  177. self.assertStatus(response, status.HTTP_200_OK)
  178. response = self.client.get_rr_sets(self.bulk_domain.name)
  179. self.assertStatus(response, status.HTTP_200_OK)
  180. self.assertRRSetsCount(response.data, data_no_records)
  181. def test_bulk_patch_does_not_need_ttl(self):
  182. self.assertResponse(
  183. self.client.bulk_patch_rr_sets(domain_name=self.bulk_domain.name, payload=self.data_no_ttl),
  184. status.HTTP_200_OK,
  185. )
  186. def test_bulk_patch_delete_non_existing_rr_sets(self):
  187. self.assertResponse(
  188. self.client.bulk_patch_rr_sets(
  189. domain_name=self.my_empty_domain.name,
  190. payload=[
  191. {'subname': 'a', 'type': 'A', 'records': [], 'ttl': 3622},
  192. {'subname': 'b', 'type': 'AAAA', 'records': []},
  193. ]),
  194. status.HTTP_200_OK,
  195. [],
  196. )
  197. def test_bulk_patch_missing_invalid_fields_1(self):
  198. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(self.my_empty_domain.name)):
  199. self.client.bulk_post_rr_sets(
  200. domain_name=self.my_empty_domain.name,
  201. payload=[
  202. {'subname': '', 'ttl': 3650, 'type': 'TXT', 'records': ['"foo"']},
  203. {'subname': 'c.1', 'records': ['dead::beef'], 'type': 'AAAA', 'ttl': 3603},
  204. {'subname': 'd.1', 'ttl': 3650, 'type': 'AAAA', 'records': ['::1', '::2']},
  205. ]
  206. )
  207. self.assertResponse(
  208. self.client.bulk_patch_rr_sets(
  209. domain_name=self.my_empty_domain.name,
  210. payload=[
  211. {'subname': 'a.1', 'records': ['dead::beef'], 'ttl': 3622},
  212. {'subname': 'b.1', 'ttl': -50, 'type': 'AAAA', 'records': ['dead::beef']},
  213. {'ttl': 3640, 'type': 'TXT', 'records': ['"bar"']},
  214. {'subname': 'c.1', 'records': ['dead::beef'], 'type': 'AAAA'},
  215. {'subname': 'd.1', 'ttl': 3650, 'type': 'AAAA'},
  216. ]),
  217. status.HTTP_400_BAD_REQUEST,
  218. [
  219. {'type': ['This field is required.']},
  220. {'ttl': [f'Ensure this value is greater than or equal to {settings.MINIMUM_TTL_DEFAULT}.']},
  221. {'subname': ['This field is required.']},
  222. {},
  223. {},
  224. ]
  225. )
  226. def test_bulk_patch_missing_invalid_fields_2(self):
  227. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(self.my_empty_domain.name)):
  228. self.client.bulk_post_rr_sets(
  229. domain_name=self.my_empty_domain.name,
  230. payload=[
  231. {'subname': '', 'ttl': 3650, 'type': 'TXT', 'records': ['"foo"']}
  232. ]
  233. )
  234. self.assertResponse(
  235. self.client.bulk_patch_rr_sets(
  236. domain_name=self.my_empty_domain.name,
  237. payload=[
  238. {'subname': 'a.1', 'records': ['dead::beef'], 'ttl': 3622},
  239. {'subname': 'b.1', 'ttl': -50, 'type': 'AAAA', 'records': ['dead::beef']},
  240. {'ttl': 3640, 'type': 'TXT', 'records': ['"bar"']},
  241. {'subname': 'c.1', 'records': ['dead::beef'], 'type': 'AAAA'},
  242. {'subname': 'd.1', 'ttl': 3650, 'type': 'AAAA'},
  243. ]),
  244. status.HTTP_400_BAD_REQUEST,
  245. [
  246. {'type': ['This field is required.']},
  247. {'ttl': [f'Ensure this value is greater than or equal to {settings.MINIMUM_TTL_DEFAULT}.']},
  248. {'subname': ['This field is required.']},
  249. {'ttl': ['This field is required.']},
  250. {'records': ['This field is required.']},
  251. ]
  252. )
  253. def test_bulk_put_partial(self):
  254. # Need all fields
  255. for domain in [self.my_empty_domain, self.bulk_domain]:
  256. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_records)
  257. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  258. self.assertEqual(response.data, [{}, {'records': ['This field is required.']}])
  259. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_ttl)
  260. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  261. self.assertEqual(response.data, [{'ttl': ['This field is required.']}, {}])
  262. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_records_no_ttl)
  263. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  264. self.assertEqual(response.data, [{},
  265. {'ttl': ['This field is required.'],
  266. 'records': ['This field is required.']}])
  267. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_subname)
  268. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  269. self.assertEqual(response.data, [{'subname': ['This field is required.']}, {}])
  270. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_subname_empty_records)
  271. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  272. self.assertEqual(response.data, [{'subname': ['This field is required.']}, {}])
  273. response = self.client.bulk_put_rr_sets(domain_name=domain.name, payload=self.data_no_type)
  274. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  275. self.assertEqual(response.data, [{}, {'type': ['This field is required.']}])
  276. def test_bulk_put_does_not_accept_single_objects(self):
  277. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data[0])
  278. self.assertContains(response, 'Expected a list of items but got dict.', status_code=status.HTTP_400_BAD_REQUEST)
  279. def test_bulk_put_does_accept_empty_list(self):
  280. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=[])
  281. self.assertStatus(response, status.HTTP_200_OK)
  282. def test_bulk_put_does_not_accept_empty_payload(self):
  283. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=None)
  284. self.assertContains(response, 'No data provided', status_code=status.HTTP_400_BAD_REQUEST)
  285. def test_bulk_put_does_not_accept_list_of_crap(self):
  286. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=['bla'])
  287. self.assertContains(response, 'Expected a dictionary, but got str.', status_code=status.HTTP_400_BAD_REQUEST)
  288. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=[42])
  289. self.assertContains(response, 'Expected a dictionary, but got int.', status_code=status.HTTP_400_BAD_REQUEST)
  290. def test_bulk_put_full(self):
  291. # Full PUT always works
  292. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.my_empty_domain.name)):
  293. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=self.data)
  294. self.assertStatus(response, status.HTTP_200_OK)
  295. # Check that RRsets have been created
  296. response = self.client.get_rr_sets(self.my_empty_domain.name)
  297. self.assertStatus(response, status.HTTP_200_OK)
  298. self.assertRRSetsCount(response.data, self.data)
  299. # Do not expect any updates, but successful code when PUT'ing only existing RRsets
  300. response = self.client.bulk_put_rr_sets(domain_name=self.bulk_domain.name, payload=self.data)
  301. self.assertStatus(response, status.HTTP_200_OK)
  302. def test_bulk_put_invalid_records(self):
  303. for records in [
  304. 'asfd',
  305. ['1.1.1.1', '2.2.2.2', 123],
  306. ['1.2.3.4', None],
  307. [True, '1.1.1.1'],
  308. dict(foobar='foobar', asdf='asdf'),
  309. ]:
  310. payload = [{'subname': 'a.2', 'ttl': 3600, 'type': 'MX', 'records': records}]
  311. response = self.client.bulk_put_rr_sets(domain_name=self.my_empty_domain.name, payload=payload)
  312. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  313. self.assertTrue('records' in response.data[0])
  314. def test_bulk_put_empty_records(self):
  315. with self.assertPdnsRequests(self.requests_desec_rr_sets_update(name=self.bulk_domain.name)):
  316. self.assertStatus(
  317. self.client.bulk_put_rr_sets(domain_name=self.bulk_domain.name, payload=self.data_empty_records),
  318. status.HTTP_200_OK
  319. )
  320. def test_bulk_duplicate_rrset(self):
  321. data = self.data + self.data
  322. for bulk_request_rr_sets in [
  323. self.client.bulk_patch_rr_sets,
  324. self.client.bulk_put_rr_sets,
  325. self.client.bulk_post_rr_sets,
  326. ]:
  327. response = bulk_request_rr_sets(domain_name=self.my_empty_domain.name, payload=data)
  328. self.assertStatus(response, status.HTTP_400_BAD_REQUEST)
  329. def test_bulk_patch_or_post_failure_with_single_rrset(self):
  330. for method in [self.client.bulk_patch_rr_sets, self.client.bulk_put_rr_sets]:
  331. response = method(domain_name=self.my_empty_domain.name, payload=self.data[0])
  332. self.assertContains(response, 'Expected a list of items but got dict.',
  333. status_code=status.HTTP_400_BAD_REQUEST)
  334. def test_bulk_delete_rrsets(self):
  335. self.assertStatus(
  336. self.client.delete(
  337. self.reverse('v1:rrsets', name=self.my_empty_domain.name),
  338. data=None,
  339. ),
  340. status.HTTP_405_METHOD_NOT_ALLOWED,
  341. )