mirror of
https://github.com/pacnpal/django-anymail.git
synced 2025-12-20 03:41:05 -05:00
Reformat code with automated tools
Apply standardized code style
This commit is contained in:
@@ -6,19 +6,23 @@ from django.test import tag
|
||||
|
||||
from anymail.signals import AnymailTrackingEvent
|
||||
from anymail.webhooks.sendinblue import SendinBlueTrackingWebhookView
|
||||
|
||||
from .webhook_cases import WebhookBasicAuthTestCase, WebhookTestCase
|
||||
|
||||
|
||||
@tag('sendinblue')
|
||||
@tag("sendinblue")
|
||||
class SendinBlueWebhookSecurityTestCase(WebhookBasicAuthTestCase):
|
||||
def call_webhook(self):
|
||||
return self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps({}))
|
||||
return self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps({}),
|
||||
)
|
||||
|
||||
# Actual tests are in WebhookBasicAuthTestCase
|
||||
|
||||
|
||||
@tag('sendinblue')
|
||||
@tag("sendinblue")
|
||||
class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
# SendinBlue's webhook payload data is partially documented at
|
||||
# https://help.sendinblue.com/hc/en-us/articles/360007666479,
|
||||
@@ -29,16 +33,15 @@ class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
raw_event = {
|
||||
"event": "request",
|
||||
"email": "recipient@example.com",
|
||||
"id": 9999999, # this appears to be a SendinBlue account id (not an event id)
|
||||
"id": 9999999, # this seems to be SendinBlue account id (not an event id)
|
||||
"message-id": "<201803062010.27287306012@smtp-relay.mailin.fr>",
|
||||
"subject": "Test subject",
|
||||
|
||||
# From a message sent at 2018-03-06 11:10:23-08:00 (2018-03-06 19:10:23+00:00)...
|
||||
"date": "2018-03-06 11:10:23", # uses time zone from SendinBlue account's preferences
|
||||
# From a message sent at 2018-03-06 11:10:23-08:00
|
||||
# (2018-03-06 19:10:23+00:00)...
|
||||
"date": "2018-03-06 11:10:23", # tz from SendinBlue account's preferences
|
||||
"ts": 1520331023, # 2018-03-06 10:10:23 -- what time zone is this?
|
||||
"ts_event": 1520331023, # unclear if this ever differs from "ts"
|
||||
"ts_epoch": 1520363423000, # 2018-03-06 19:10:23.000+00:00 -- UTC (milliseconds)
|
||||
|
||||
"ts_epoch": 1520363423000, # 2018-03-06 19:10:23.000+00:00 -- UTC (msec)
|
||||
"X-Mailin-custom": '{"meta": "data"}',
|
||||
# "tag" is JSON-serialized tags array if `tags` param set on send,
|
||||
# else single tag string if `X-Mailin-Tag` header set on send,
|
||||
@@ -50,18 +53,31 @@ class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
"template_id": 12,
|
||||
"sending_ip": "333.33.33.33",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertIsInstance(event, AnymailTrackingEvent)
|
||||
self.assertEqual(event.event_type, "queued")
|
||||
self.assertEqual(event.esp_event, raw_event)
|
||||
self.assertEqual(event.timestamp, datetime(2018, 3, 6, 19, 10, 23, microsecond=0, tzinfo=timezone.utc))
|
||||
self.assertEqual(event.message_id, "<201803062010.27287306012@smtp-relay.mailin.fr>")
|
||||
self.assertIsNone(event.event_id) # SendinBlue does not provide a unique event id
|
||||
self.assertEqual(
|
||||
event.timestamp,
|
||||
datetime(2018, 3, 6, 19, 10, 23, microsecond=0, tzinfo=timezone.utc),
|
||||
)
|
||||
self.assertEqual(
|
||||
event.message_id, "<201803062010.27287306012@smtp-relay.mailin.fr>"
|
||||
)
|
||||
# SendinBlue does not provide a unique event id:
|
||||
self.assertIsNone(event.event_id)
|
||||
self.assertEqual(event.recipient, "recipient@example.com")
|
||||
self.assertEqual(event.metadata, {"meta": "data"})
|
||||
self.assertEqual(event.tags, ["tag1", "tag2"])
|
||||
@@ -75,18 +91,28 @@ class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
"ts_epoch": 1520363423000,
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertIsInstance(event, AnymailTrackingEvent)
|
||||
self.assertEqual(event.event_type, "delivered")
|
||||
self.assertEqual(event.esp_event, raw_event)
|
||||
self.assertEqual(event.message_id, "<201803011158.9876543210@smtp-relay.mailin.fr>")
|
||||
self.assertEqual(
|
||||
event.message_id, "<201803011158.9876543210@smtp-relay.mailin.fr>"
|
||||
)
|
||||
self.assertEqual(event.recipient, "recipient@example.com")
|
||||
self.assertEqual(event.metadata, {}) # empty dict when no X-Mailin-custom header given
|
||||
# empty dict when no X-Mailin-custom header given:
|
||||
self.assertEqual(event.metadata, {})
|
||||
self.assertEqual(event.tags, []) # empty list when no tags given
|
||||
|
||||
def test_hard_bounce(self):
|
||||
@@ -96,19 +122,30 @@ class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
"ts_epoch": 1520363423000,
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
# the leading space in the reason is as received in actual testing:
|
||||
"reason": " RecipientError: 550 5.5.0 Requested action not taken: mailbox unavailable.",
|
||||
"reason": " RecipientError: 550 5.5.0"
|
||||
" Requested action not taken: mailbox unavailable.",
|
||||
"tag": "header-tag",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "bounced")
|
||||
self.assertEqual(event.reject_reason, "bounced")
|
||||
self.assertEqual(event.mta_response,
|
||||
" RecipientError: 550 5.5.0 Requested action not taken: mailbox unavailable.")
|
||||
self.assertEqual(
|
||||
event.mta_response,
|
||||
" RecipientError: 550 5.5.0"
|
||||
" Requested action not taken: mailbox unavailable.",
|
||||
)
|
||||
self.assertEqual(event.tags, ["header-tag"])
|
||||
|
||||
def test_soft_bounce_event(self):
|
||||
@@ -119,16 +156,27 @@ class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
"reason": "undefined Unable to find MX of domain no-mx.example.com",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "bounced")
|
||||
self.assertEqual(event.reject_reason, "bounced")
|
||||
self.assertIsNone(event.description) # no human-readable description consistently available
|
||||
self.assertEqual(event.mta_response, "undefined Unable to find MX of domain no-mx.example.com")
|
||||
# no human-readable description consistently available:
|
||||
self.assertIsNone(event.description)
|
||||
self.assertEqual(
|
||||
event.mta_response,
|
||||
"undefined Unable to find MX of domain no-mx.example.com",
|
||||
)
|
||||
|
||||
def test_blocked(self):
|
||||
raw_event = {
|
||||
@@ -138,12 +186,19 @@ class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
"reason": "blocked : due to blacklist user",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "rejected")
|
||||
self.assertEqual(event.reject_reason, "blocked")
|
||||
self.assertEqual(event.mta_response, "blocked : due to blacklist user")
|
||||
@@ -157,18 +212,26 @@ class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
"ts_epoch": 1520363423000,
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "complained")
|
||||
|
||||
def test_invalid_email(self):
|
||||
# "If a ISP again indicated us that the email is not valid or if we discovered that the email is not valid."
|
||||
# (unclear whether this error originates with the receiving MTA or with SendinBlue pre-send)
|
||||
# (haven't observed "invalid_email" event in actual testing; payload below is a guess)
|
||||
# "If a ISP again indicated us that the email is not valid or if we discovered
|
||||
# that the email is not valid." (unclear whether this error originates with the
|
||||
# receiving MTA or with SendinBlue pre-send) (haven't observed "invalid_email"
|
||||
# event in actual testing; payload below is a guess)
|
||||
raw_event = {
|
||||
"event": "invalid_email",
|
||||
"email": "recipient@example.com",
|
||||
@@ -176,73 +239,106 @@ class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
"reason": "(guessing invalid_email includes a reason)",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "bounced")
|
||||
self.assertEqual(event.reject_reason, "invalid")
|
||||
self.assertEqual(event.mta_response, "(guessing invalid_email includes a reason)")
|
||||
self.assertEqual(
|
||||
event.mta_response, "(guessing invalid_email includes a reason)"
|
||||
)
|
||||
|
||||
def test_deferred_event(self):
|
||||
# Note: the example below is an actual event capture (with 'example.com' substituted
|
||||
# for the real receiving domain). It's pretty clearly a bounce, not a deferral.
|
||||
# It looks like SendinBlue mis-categorizes this SMTP response code.
|
||||
# Note: the example below is an actual event capture (with 'example.com'
|
||||
# substituted for the real receiving domain). It's pretty clearly a bounce, not
|
||||
# a deferral. It looks like SendinBlue mis-categorizes this SMTP response code.
|
||||
raw_event = {
|
||||
"event": "deferred",
|
||||
"email": "notauser@example.com",
|
||||
"ts_epoch": 1520363423000,
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
"reason": "550 RecipientError: 550 5.1.1 <notauser@example.com>: Recipient address rejected: "
|
||||
"User unknown in virtual alias table",
|
||||
"reason": "550 RecipientError: 550 5.1.1 <notauser@example.com>: Recipient"
|
||||
" address rejected: User unknown in virtual alias table",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "deferred")
|
||||
self.assertIsNone(event.description) # no human-readable description consistently available
|
||||
self.assertEqual(event.mta_response,
|
||||
"550 RecipientError: 550 5.1.1 <notauser@example.com>: Recipient address rejected: "
|
||||
"User unknown in virtual alias table")
|
||||
# no human-readable description consistently available:
|
||||
self.assertIsNone(event.description)
|
||||
self.assertEqual(
|
||||
event.mta_response,
|
||||
"550 RecipientError: 550 5.1.1 <notauser@example.com>:"
|
||||
" Recipient address rejected: User unknown in virtual alias table",
|
||||
)
|
||||
|
||||
def test_opened_event(self):
|
||||
# SendinBlue delivers unique_opened *and* opened on the first open.
|
||||
# To avoid double-counting, you should only enable one of the two events in SendinBlue.
|
||||
# SendinBlue delivers unique_opened *and* opened on the first open. To avoid
|
||||
# double-counting, you should only enable one of the two events in SendinBlue.
|
||||
raw_event = {
|
||||
"event": "opened",
|
||||
"email": "recipient@example.com",
|
||||
"ts_epoch": 1520363423000,
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "opened")
|
||||
self.assertIsNone(event.user_agent) # SendinBlue doesn't report user agent
|
||||
|
||||
def test_unique_opened_event(self):
|
||||
# SendinBlue delivers unique_opened *and* opened on the first open.
|
||||
# To avoid double-counting, you should only enable one of the two events in SendinBlue.
|
||||
# SendinBlue delivers unique_opened *and* opened on the first open. To avoid
|
||||
# double-counting, you should only enable one of the two events in SendinBlue.
|
||||
raw_event = {
|
||||
"event": "unique_opened",
|
||||
"email": "recipient@example.com",
|
||||
"ts_epoch": 1520363423000,
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "opened")
|
||||
|
||||
def test_clicked_event(self):
|
||||
@@ -253,29 +349,44 @@ class SendinBlueDeliveryTestCase(WebhookTestCase):
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
"link": "https://example.com/click/me",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "clicked")
|
||||
self.assertEqual(event.click_url, "https://example.com/click/me")
|
||||
self.assertIsNone(event.user_agent) # SendinBlue doesn't report user agent
|
||||
|
||||
def test_unsubscribe(self):
|
||||
# "When a person unsubscribes from the email received."
|
||||
# (haven't observed "unsubscribe" event in actual testing; payload below is a guess)
|
||||
# (haven't observed "unsubscribe" event in actual testing;
|
||||
# payload below is a guess)
|
||||
raw_event = {
|
||||
"event": "unsubscribe",
|
||||
"email": "recipient@example.com",
|
||||
"ts_epoch": 1520363423000,
|
||||
"message-id": "<201803011158.9876543210@smtp-relay.mailin.fr>",
|
||||
}
|
||||
response = self.client.post('/anymail/sendinblue/tracking/',
|
||||
content_type='application/json', data=json.dumps(raw_event))
|
||||
response = self.client.post(
|
||||
"/anymail/sendinblue/tracking/",
|
||||
content_type="application/json",
|
||||
data=json.dumps(raw_event),
|
||||
)
|
||||
self.assertEqual(response.status_code, 200)
|
||||
kwargs = self.assert_handler_called_once_with(self.tracking_handler, sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY, esp_name='SendinBlue')
|
||||
event = kwargs['event']
|
||||
kwargs = self.assert_handler_called_once_with(
|
||||
self.tracking_handler,
|
||||
sender=SendinBlueTrackingWebhookView,
|
||||
event=ANY,
|
||||
esp_name="SendinBlue",
|
||||
)
|
||||
event = kwargs["event"]
|
||||
self.assertEqual(event.event_type, "unsubscribed")
|
||||
|
||||
Reference in New Issue
Block a user