|
@@ -4,6 +4,7 @@ from unittest.mock import patch
|
|
|
|
|
|
|
|
import django_rq
|
|
import django_rq
|
|
|
from django.http import HttpResponse
|
|
from django.http import HttpResponse
|
|
|
|
|
+from django.test import RequestFactory
|
|
|
from django.urls import reverse
|
|
from django.urls import reverse
|
|
|
from requests import Session
|
|
from requests import Session
|
|
|
from rest_framework import status
|
|
from rest_framework import status
|
|
@@ -12,6 +13,7 @@ from core.models import ObjectType
|
|
|
from dcim.choices import SiteStatusChoices
|
|
from dcim.choices import SiteStatusChoices
|
|
|
from dcim.models import Site
|
|
from dcim.models import Site
|
|
|
from extras.choices import EventRuleActionChoices, ObjectChangeActionChoices
|
|
from extras.choices import EventRuleActionChoices, ObjectChangeActionChoices
|
|
|
|
|
+from extras.context_managers import event_tracking
|
|
|
from extras.events import enqueue_object, flush_events, serialize_for_event
|
|
from extras.events import enqueue_object, flush_events, serialize_for_event
|
|
|
from extras.models import EventRule, Tag, Webhook
|
|
from extras.models import EventRule, Tag, Webhook
|
|
|
from extras.webhooks import generate_signature, send_webhook
|
|
from extras.webhooks import generate_signature, send_webhook
|
|
@@ -360,7 +362,7 @@ class EventRuleTest(APITestCase):
|
|
|
return HttpResponse()
|
|
return HttpResponse()
|
|
|
|
|
|
|
|
# Enqueue a webhook for processing
|
|
# Enqueue a webhook for processing
|
|
|
- webhooks_queue = []
|
|
|
|
|
|
|
+ webhooks_queue = {}
|
|
|
site = Site.objects.create(name='Site 1', slug='site-1')
|
|
site = Site.objects.create(name='Site 1', slug='site-1')
|
|
|
enqueue_object(
|
|
enqueue_object(
|
|
|
webhooks_queue,
|
|
webhooks_queue,
|
|
@@ -369,7 +371,7 @@ class EventRuleTest(APITestCase):
|
|
|
request_id=request_id,
|
|
request_id=request_id,
|
|
|
action=ObjectChangeActionChoices.ACTION_CREATE
|
|
action=ObjectChangeActionChoices.ACTION_CREATE
|
|
|
)
|
|
)
|
|
|
- flush_events(webhooks_queue)
|
|
|
|
|
|
|
+ flush_events(list(webhooks_queue.values()))
|
|
|
|
|
|
|
|
# Retrieve the job from queue
|
|
# Retrieve the job from queue
|
|
|
job = self.queue.jobs[0]
|
|
job = self.queue.jobs[0]
|
|
@@ -377,3 +379,24 @@ class EventRuleTest(APITestCase):
|
|
|
# Patch the Session object with our dummy_send() method, then process the webhook for sending
|
|
# Patch the Session object with our dummy_send() method, then process the webhook for sending
|
|
|
with patch.object(Session, 'send', dummy_send) as mock_send:
|
|
with patch.object(Session, 'send', dummy_send) as mock_send:
|
|
|
send_webhook(**job.kwargs)
|
|
send_webhook(**job.kwargs)
|
|
|
|
|
+
|
|
|
|
|
+ def test_duplicate_triggers(self):
|
|
|
|
|
+ """
|
|
|
|
|
+ Test for erroneous duplicate event triggers resulting from saving an object multiple times
|
|
|
|
|
+ within the span of a single request.
|
|
|
|
|
+ """
|
|
|
|
|
+ url = reverse('dcim:site_add')
|
|
|
|
|
+ request = RequestFactory().get(url)
|
|
|
|
|
+ request.id = uuid.uuid4()
|
|
|
|
|
+ request.user = self.user
|
|
|
|
|
+
|
|
|
|
|
+ self.assertEqual(self.queue.count, 0, msg="Unexpected jobs found in queue")
|
|
|
|
|
+
|
|
|
|
|
+ with event_tracking(request):
|
|
|
|
|
+ site = Site(name='Site 1', slug='site-1')
|
|
|
|
|
+ site.save()
|
|
|
|
|
+
|
|
|
|
|
+ # Save the site a second time
|
|
|
|
|
+ site.save()
|
|
|
|
|
+
|
|
|
|
|
+ self.assertEqual(self.queue.count, 1, msg="Duplicate jobs found in queue")
|