test_webhooks.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. import json
  2. import uuid
  3. from unittest.mock import patch
  4. import django_rq
  5. from django.contrib.contenttypes.models import ContentType
  6. from django.http import HttpResponse
  7. from django.urls import reverse
  8. from requests import Session
  9. from rest_framework import status
  10. from dcim.models import Site
  11. from extras.choices import ObjectChangeActionChoices
  12. from extras.models import Tag, Webhook
  13. from extras.webhooks import enqueue_object, flush_webhooks, generate_signature
  14. from extras.webhooks_worker import process_webhook
  15. from utilities.testing import APITestCase
  16. class WebhookTest(APITestCase):
  17. def setUp(self):
  18. super().setUp()
  19. self.queue = django_rq.get_queue('default')
  20. self.queue.empty()
  21. @classmethod
  22. def setUpTestData(cls):
  23. site_ct = ContentType.objects.get_for_model(Site)
  24. DUMMY_URL = "http://localhost/"
  25. DUMMY_SECRET = "LOOKATMEIMASECRETSTRING"
  26. webhooks = Webhook.objects.bulk_create((
  27. Webhook(name='Webhook 1', type_create=True, payload_url=DUMMY_URL, secret=DUMMY_SECRET, additional_headers='X-Foo: Bar'),
  28. Webhook(name='Webhook 2', type_update=True, payload_url=DUMMY_URL, secret=DUMMY_SECRET),
  29. Webhook(name='Webhook 3', type_delete=True, payload_url=DUMMY_URL, secret=DUMMY_SECRET),
  30. ))
  31. for webhook in webhooks:
  32. webhook.content_types.set([site_ct])
  33. Tag.objects.bulk_create((
  34. Tag(name='Foo', slug='foo'),
  35. Tag(name='Bar', slug='bar'),
  36. Tag(name='Baz', slug='baz'),
  37. ))
  38. def test_enqueue_webhook_create(self):
  39. # Create an object via the REST API
  40. data = {
  41. 'name': 'Site 1',
  42. 'slug': 'site-1',
  43. 'tags': [
  44. {'name': 'Foo'},
  45. {'name': 'Bar'},
  46. ]
  47. }
  48. url = reverse('dcim-api:site-list')
  49. self.add_permissions('dcim.add_site')
  50. response = self.client.post(url, data, format='json', **self.header)
  51. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  52. self.assertEqual(Site.objects.count(), 1)
  53. self.assertEqual(Site.objects.first().tags.count(), 2)
  54. # Verify that a job was queued for the object creation webhook
  55. self.assertEqual(self.queue.count, 1)
  56. job = self.queue.jobs[0]
  57. self.assertEqual(job.kwargs['webhook'], Webhook.objects.get(type_create=True))
  58. self.assertEqual(job.kwargs['event'], ObjectChangeActionChoices.ACTION_CREATE)
  59. self.assertEqual(job.kwargs['model_name'], 'site')
  60. self.assertEqual(job.kwargs['data']['id'], response.data['id'])
  61. self.assertEqual(len(job.kwargs['data']['tags']), len(response.data['tags']))
  62. self.assertEqual(job.kwargs['snapshots']['postchange']['name'], 'Site 1')
  63. self.assertEqual(job.kwargs['snapshots']['postchange']['tags'], ['Bar', 'Foo'])
  64. def test_enqueue_webhook_bulk_create(self):
  65. # Create multiple objects via the REST API
  66. data = [
  67. {
  68. 'name': 'Site 1',
  69. 'slug': 'site-1',
  70. 'tags': [
  71. {'name': 'Foo'},
  72. {'name': 'Bar'},
  73. ]
  74. },
  75. {
  76. 'name': 'Site 2',
  77. 'slug': 'site-2',
  78. 'tags': [
  79. {'name': 'Foo'},
  80. {'name': 'Bar'},
  81. ]
  82. },
  83. {
  84. 'name': 'Site 3',
  85. 'slug': 'site-3',
  86. 'tags': [
  87. {'name': 'Foo'},
  88. {'name': 'Bar'},
  89. ]
  90. },
  91. ]
  92. url = reverse('dcim-api:site-list')
  93. self.add_permissions('dcim.add_site')
  94. response = self.client.post(url, data, format='json', **self.header)
  95. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  96. self.assertEqual(Site.objects.count(), 3)
  97. self.assertEqual(Site.objects.first().tags.count(), 2)
  98. # Verify that a webhook was queued for each object
  99. self.assertEqual(self.queue.count, 3)
  100. for i, job in enumerate(self.queue.jobs):
  101. self.assertEqual(job.kwargs['webhook'], Webhook.objects.get(type_create=True))
  102. self.assertEqual(job.kwargs['event'], ObjectChangeActionChoices.ACTION_CREATE)
  103. self.assertEqual(job.kwargs['model_name'], 'site')
  104. self.assertEqual(job.kwargs['data']['id'], response.data[i]['id'])
  105. self.assertEqual(len(job.kwargs['data']['tags']), len(response.data[i]['tags']))
  106. self.assertEqual(job.kwargs['snapshots']['postchange']['name'], response.data[i]['name'])
  107. self.assertEqual(job.kwargs['snapshots']['postchange']['tags'], ['Bar', 'Foo'])
  108. def test_enqueue_webhook_update(self):
  109. site = Site.objects.create(name='Site 1', slug='site-1')
  110. site.tags.set(*Tag.objects.filter(name__in=['Foo', 'Bar']))
  111. # Update an object via the REST API
  112. data = {
  113. 'name': 'Site X',
  114. 'comments': 'Updated the site',
  115. 'tags': [
  116. {'name': 'Baz'}
  117. ]
  118. }
  119. url = reverse('dcim-api:site-detail', kwargs={'pk': site.pk})
  120. self.add_permissions('dcim.change_site')
  121. response = self.client.patch(url, data, format='json', **self.header)
  122. self.assertHttpStatus(response, status.HTTP_200_OK)
  123. # Verify that a job was queued for the object update webhook
  124. self.assertEqual(self.queue.count, 1)
  125. job = self.queue.jobs[0]
  126. self.assertEqual(job.kwargs['webhook'], Webhook.objects.get(type_update=True))
  127. self.assertEqual(job.kwargs['event'], ObjectChangeActionChoices.ACTION_UPDATE)
  128. self.assertEqual(job.kwargs['model_name'], 'site')
  129. self.assertEqual(job.kwargs['data']['id'], site.pk)
  130. self.assertEqual(len(job.kwargs['data']['tags']), len(response.data['tags']))
  131. self.assertEqual(job.kwargs['snapshots']['prechange']['name'], 'Site 1')
  132. self.assertEqual(job.kwargs['snapshots']['prechange']['tags'], ['Bar', 'Foo'])
  133. self.assertEqual(job.kwargs['snapshots']['postchange']['name'], 'Site X')
  134. self.assertEqual(job.kwargs['snapshots']['postchange']['tags'], ['Baz'])
  135. def test_enqueue_webhook_bulk_update(self):
  136. sites = (
  137. Site(name='Site 1', slug='site-1'),
  138. Site(name='Site 2', slug='site-2'),
  139. Site(name='Site 3', slug='site-3'),
  140. )
  141. Site.objects.bulk_create(sites)
  142. for site in sites:
  143. site.tags.set(*Tag.objects.filter(name__in=['Foo', 'Bar']))
  144. # Update three objects via the REST API
  145. data = [
  146. {
  147. 'id': sites[0].pk,
  148. 'name': 'Site X',
  149. 'tags': [
  150. {'name': 'Baz'}
  151. ]
  152. },
  153. {
  154. 'id': sites[1].pk,
  155. 'name': 'Site Y',
  156. 'tags': [
  157. {'name': 'Baz'}
  158. ]
  159. },
  160. {
  161. 'id': sites[2].pk,
  162. 'name': 'Site Z',
  163. 'tags': [
  164. {'name': 'Baz'}
  165. ]
  166. },
  167. ]
  168. url = reverse('dcim-api:site-list')
  169. self.add_permissions('dcim.change_site')
  170. response = self.client.patch(url, data, format='json', **self.header)
  171. self.assertHttpStatus(response, status.HTTP_200_OK)
  172. # Verify that a job was queued for the object update webhook
  173. self.assertEqual(self.queue.count, 3)
  174. for i, job in enumerate(self.queue.jobs):
  175. self.assertEqual(job.kwargs['webhook'], Webhook.objects.get(type_update=True))
  176. self.assertEqual(job.kwargs['event'], ObjectChangeActionChoices.ACTION_UPDATE)
  177. self.assertEqual(job.kwargs['model_name'], 'site')
  178. self.assertEqual(job.kwargs['data']['id'], data[i]['id'])
  179. self.assertEqual(len(job.kwargs['data']['tags']), len(response.data[i]['tags']))
  180. self.assertEqual(job.kwargs['snapshots']['prechange']['name'], sites[i].name)
  181. self.assertEqual(job.kwargs['snapshots']['prechange']['tags'], ['Bar', 'Foo'])
  182. self.assertEqual(job.kwargs['snapshots']['postchange']['name'], response.data[i]['name'])
  183. self.assertEqual(job.kwargs['snapshots']['postchange']['tags'], ['Baz'])
  184. def test_enqueue_webhook_delete(self):
  185. site = Site.objects.create(name='Site 1', slug='site-1')
  186. site.tags.set(*Tag.objects.filter(name__in=['Foo', 'Bar']))
  187. # Delete an object via the REST API
  188. url = reverse('dcim-api:site-detail', kwargs={'pk': site.pk})
  189. self.add_permissions('dcim.delete_site')
  190. response = self.client.delete(url, **self.header)
  191. self.assertHttpStatus(response, status.HTTP_204_NO_CONTENT)
  192. # Verify that a job was queued for the object update webhook
  193. self.assertEqual(self.queue.count, 1)
  194. job = self.queue.jobs[0]
  195. self.assertEqual(job.kwargs['webhook'], Webhook.objects.get(type_delete=True))
  196. self.assertEqual(job.kwargs['event'], ObjectChangeActionChoices.ACTION_DELETE)
  197. self.assertEqual(job.kwargs['model_name'], 'site')
  198. self.assertEqual(job.kwargs['data']['id'], site.pk)
  199. self.assertEqual(job.kwargs['snapshots']['prechange']['name'], 'Site 1')
  200. self.assertEqual(job.kwargs['snapshots']['prechange']['tags'], ['Bar', 'Foo'])
  201. def test_enqueue_webhook_bulk_delete(self):
  202. sites = (
  203. Site(name='Site 1', slug='site-1'),
  204. Site(name='Site 2', slug='site-2'),
  205. Site(name='Site 3', slug='site-3'),
  206. )
  207. Site.objects.bulk_create(sites)
  208. for site in sites:
  209. site.tags.set(*Tag.objects.filter(name__in=['Foo', 'Bar']))
  210. # Delete three objects via the REST API
  211. data = [
  212. {'id': site.pk} for site in sites
  213. ]
  214. url = reverse('dcim-api:site-list')
  215. self.add_permissions('dcim.delete_site')
  216. response = self.client.delete(url, data, format='json', **self.header)
  217. self.assertHttpStatus(response, status.HTTP_204_NO_CONTENT)
  218. # Verify that a job was queued for the object update webhook
  219. self.assertEqual(self.queue.count, 3)
  220. for i, job in enumerate(self.queue.jobs):
  221. self.assertEqual(job.kwargs['webhook'], Webhook.objects.get(type_delete=True))
  222. self.assertEqual(job.kwargs['event'], ObjectChangeActionChoices.ACTION_DELETE)
  223. self.assertEqual(job.kwargs['model_name'], 'site')
  224. self.assertEqual(job.kwargs['data']['id'], sites[i].pk)
  225. self.assertEqual(job.kwargs['snapshots']['prechange']['name'], sites[i].name)
  226. self.assertEqual(job.kwargs['snapshots']['prechange']['tags'], ['Bar', 'Foo'])
  227. def test_webhooks_worker(self):
  228. request_id = uuid.uuid4()
  229. def dummy_send(_, request, **kwargs):
  230. """
  231. A dummy implementation of Session.send() to be used for testing.
  232. Always returns a 200 HTTP response.
  233. """
  234. webhook = Webhook.objects.get(type_create=True)
  235. signature = generate_signature(request.body, webhook.secret)
  236. # Validate the outgoing request headers
  237. self.assertEqual(request.headers['Content-Type'], webhook.http_content_type)
  238. self.assertEqual(request.headers['X-Hook-Signature'], signature)
  239. self.assertEqual(request.headers['X-Foo'], 'Bar')
  240. # Validate the outgoing request body
  241. body = json.loads(request.body)
  242. self.assertEqual(body['event'], 'created')
  243. self.assertEqual(body['timestamp'], job.kwargs['timestamp'])
  244. self.assertEqual(body['model'], 'site')
  245. self.assertEqual(body['username'], 'testuser')
  246. self.assertEqual(body['request_id'], str(request_id))
  247. self.assertEqual(body['data']['name'], 'Site 1')
  248. return HttpResponse()
  249. # Enqueue a webhook for processing
  250. webhooks_queue = []
  251. site = Site.objects.create(name='Site 1', slug='site-1')
  252. enqueue_object(
  253. webhooks_queue,
  254. instance=site,
  255. user=self.user,
  256. request_id=request_id,
  257. action=ObjectChangeActionChoices.ACTION_CREATE
  258. )
  259. flush_webhooks(webhooks_queue)
  260. # Retrieve the job from queue
  261. job = self.queue.jobs[0]
  262. # Patch the Session object with our dummy_send() method, then process the webhook for sending
  263. with patch.object(Session, 'send', dummy_send) as mock_send:
  264. process_webhook(**job.kwargs)