mirror of
https://github.com/pacnpal/django-anymail.git
synced 2025-12-20 11:51:05 -05:00
Includes type-checking on alternative message part, and switches to ValueError (rather than ImproperlyConfigured) for unacceptable alternatives. [Also fixes bug where fail_silently=True wasn't respected.] Cherry-picked from: medmunds/Djrill@faf53a1a0 (and parts of medmunds/Djrill@62d48c5f)
181 lines
7.9 KiB
Python
181 lines
7.9 KiB
Python
from mock import patch
|
|
|
|
from django.conf import settings
|
|
from django.core import mail
|
|
from django.core.exceptions import ImproperlyConfigured
|
|
from django.test import TestCase
|
|
from django.utils import simplejson as json
|
|
|
|
from djrill.mail import DjrillMessage
|
|
|
|
class DjrillBackendMockAPITestCase(TestCase):
|
|
"""TestCase that uses Djrill EmailBackend with a mocked Mandrill API"""
|
|
|
|
class MockResponse:
|
|
"""requests.post return value mock sufficient for DjrillBackend"""
|
|
def __init__(self, status_code=200):
|
|
self.status_code = status_code
|
|
|
|
def setUp(self):
|
|
self.patch = patch('requests.post')
|
|
self.mock_post = self.patch.start()
|
|
self.mock_post.return_value = self.MockResponse()
|
|
|
|
settings.MANDRILL_API_KEY = "FAKE_API_KEY_FOR_TESTING"
|
|
settings.MANDRILL_API_URL = "http://mandrillapp.com/api/1.0"
|
|
|
|
# Django TestCase sets up locmem EmailBackend; override it here
|
|
self.original_email_backend = settings.EMAIL_BACKEND
|
|
settings.EMAIL_BACKEND = "djrill.mail.backends.djrill.DjrillBackend"
|
|
|
|
def tearDown(self):
|
|
self.patch.stop()
|
|
settings.EMAIL_BACKEND = self.original_email_backend
|
|
|
|
def get_api_call_data(self):
|
|
"""Returns the data posted to the Mandrill API.
|
|
|
|
Fails test if API wasn't called.
|
|
"""
|
|
if self.mock_post.call_args is None:
|
|
raise AssertionError("Mandrill API was not called")
|
|
(args, kwargs) = self.mock_post.call_args
|
|
if 'data' not in kwargs:
|
|
raise AssertionError("requests.post was called without data kwarg "
|
|
"-- Maybe tests need to be updated for backend changes?")
|
|
return json.loads(kwargs['data'])
|
|
|
|
|
|
class DjrillBackendTests(DjrillBackendMockAPITestCase):
|
|
"""Test Djrill's support for Django mail wrappers"""
|
|
|
|
def test_send_mail(self):
|
|
mail.send_mail('Subject here', 'Here is the message.',
|
|
'from@example.com', ['to@example.com'], fail_silently=False)
|
|
data = self.get_api_call_data()
|
|
self.assertEqual(data['message']['subject'], "Subject here")
|
|
self.assertEqual(data['message']['text'], "Here is the message.")
|
|
self.assertEqual(data['message']['from_email'], "from@example.com")
|
|
self.assertEqual(len(data['message']['to']), 1)
|
|
self.assertEqual(data['message']['to'][0]['email'], "to@example.com")
|
|
|
|
def test_missing_api_key(self):
|
|
del settings.MANDRILL_API_KEY
|
|
with self.assertRaises(ImproperlyConfigured):
|
|
mail.send_mail('Subject', 'Message', 'from@example.com',
|
|
['to@example.com'])
|
|
|
|
def test_email_message(self):
|
|
email = mail.EmailMessage('Subject', 'Body goes here',
|
|
'from@example.com',
|
|
['to1@example.com', 'Also To <to2@example.com>'],
|
|
bcc=['bcc1@example.com', 'Also BCC <bcc2@example.com>'],
|
|
cc=['cc1@example.com', 'Also CC <cc2@example.com>'],
|
|
headers={'Reply-To': 'another@example.com',
|
|
'X-MyHeader': 'my value',
|
|
'Errors-To': 'silently stripped'})
|
|
email.send()
|
|
data = self.get_api_call_data()
|
|
self.assertEqual(data['message']['subject'], "Subject")
|
|
self.assertEqual(data['message']['text'], "Body goes here")
|
|
self.assertEqual(data['message']['from_email'], "from@example.com")
|
|
self.assertEqual(data['message']['headers'],
|
|
{ 'Reply-To': 'another@example.com', 'X-MyHeader': 'my value' })
|
|
# Mandrill doesn't have a notion of cc, and only allows a single bcc.
|
|
# Djrill just treats cc and bcc as though they were "to" addresses,
|
|
# which may or may not be what you want.
|
|
self.assertEqual(len(data['message']['to']), 6)
|
|
self.assertEqual(data['message']['to'][0]['email'], "to1@example.com")
|
|
self.assertEqual(data['message']['to'][1]['email'], "to2@example.com")
|
|
self.assertEqual(data['message']['to'][2]['email'], "cc1@example.com")
|
|
self.assertEqual(data['message']['to'][3]['email'], "cc2@example.com")
|
|
self.assertEqual(data['message']['to'][4]['email'], "bcc1@example.com")
|
|
self.assertEqual(data['message']['to'][5]['email'], "bcc2@example.com")
|
|
|
|
def test_html_message(self):
|
|
text_content = 'This is an important message.'
|
|
html_content = '<p>This is an <strong>important</strong> message.</p>'
|
|
email = mail.EmailMultiAlternatives('Subject', text_content,
|
|
'from@example.com', ['to@example.com'])
|
|
email.attach_alternative(html_content, "text/html")
|
|
email.send()
|
|
data = self.get_api_call_data()
|
|
self.assertEqual(data['message']['text'], text_content)
|
|
self.assertEqual(data['message']['html'], html_content)
|
|
|
|
def test_alternative_errors(self):
|
|
# Multiple alternatives not allowed
|
|
email = mail.EmailMultiAlternatives('Subject', 'Body',
|
|
'from@example.com', ['to@example.com'])
|
|
email.attach_alternative("<p>First html is OK</p>", "text/html")
|
|
email.attach_alternative("<p>But not second html</p>", "text/html")
|
|
with self.assertRaises(ValueError):
|
|
email.send()
|
|
|
|
# Only html alternatives allowed
|
|
email = mail.EmailMultiAlternatives('Subject', 'Body',
|
|
'from@example.com', ['to@example.com'])
|
|
email.attach_alternative("{'not': 'allowed'}", "application/json")
|
|
with self.assertRaises(ValueError):
|
|
email.send()
|
|
|
|
# Make sure fail_silently is respected
|
|
email = mail.EmailMultiAlternatives('Subject', 'Body',
|
|
'from@example.com', ['to@example.com'])
|
|
email.attach_alternative("{'not': 'allowed'}", "application/json")
|
|
sent = email.send(fail_silently=True)
|
|
self.assertFalse(self.mock_post.called,
|
|
msg="Mandrill API should not be called when send fails silently")
|
|
self.assertEqual(sent, 0)
|
|
|
|
|
|
class DjrillMessageTests(TestCase):
|
|
def setUp(self):
|
|
self.subject = "Djrill baby djrill."
|
|
self.from_name = "Tarzan"
|
|
self.from_email = "test@example"
|
|
self.to = ["King Kong <kingkong@example.com>",
|
|
"Cheetah <cheetah@example.com", "bubbles@example.com"]
|
|
self.text_content = "Wonderful fallback text content."
|
|
self.html_content = "<h1>That's a nice HTML email right there.</h1>"
|
|
self.headers = {"Reply-To": "tarzan@example.com"}
|
|
self.tags = ["track", "this"]
|
|
|
|
def test_djrill_message_success(self):
|
|
msg = DjrillMessage(self.subject, self.text_content, self.from_email,
|
|
self.to, tags=self.tags, headers=self.headers,
|
|
from_name=self.from_name)
|
|
|
|
self.assertIsInstance(msg, DjrillMessage)
|
|
self.assertEqual(msg.body, self.text_content)
|
|
self.assertEqual(msg.recipients(), self.to)
|
|
self.assertEqual(msg.tags, self.tags)
|
|
self.assertEqual(msg.extra_headers, self.headers)
|
|
self.assertEqual(msg.from_name, self.from_name)
|
|
|
|
def test_djrill_message_html_success(self):
|
|
msg = DjrillMessage(self.subject, self.text_content, self.from_email,
|
|
self.to, tags=self.tags)
|
|
msg.attach_alternative(self.html_content, "text/html")
|
|
|
|
self.assertEqual(msg.alternatives[0][0], self.html_content)
|
|
|
|
def test_djrill_message_tag_failure(self):
|
|
with self.assertRaises(ImproperlyConfigured):
|
|
DjrillMessage(self.subject, self.text_content, self.from_email,
|
|
self.to, tags=["_fail"])
|
|
|
|
def test_djrill_message_tag_skip(self):
|
|
"""
|
|
Test that tags over 50 chars are not included in the tags list.
|
|
"""
|
|
tags = ["works", "awesomesauce",
|
|
"iwilltestmycodeiwilltestmycodeiwilltestmycodeiwilltestmycode"]
|
|
msg = DjrillMessage(self.subject, self.text_content, self.from_email,
|
|
self.to, tags=tags)
|
|
|
|
self.assertIsInstance(msg, DjrillMessage)
|
|
self.assertIn(tags[0], msg.tags)
|
|
self.assertIn(tags[1], msg.tags)
|
|
self.assertNotIn(tags[2], msg.tags)
|