Merge pull request #2126 from apollo13/email_bytes
Many thanks to @bitdancer and @aaugustin for answering my stupid questions about (mail)encodings and pointing me in the right direction.
This commit is contained in:
commit
df075c7489
|
@ -5,6 +5,7 @@ import sys
|
||||||
import threading
|
import threading
|
||||||
|
|
||||||
from django.core.mail.backends.base import BaseEmailBackend
|
from django.core.mail.backends.base import BaseEmailBackend
|
||||||
|
from django.utils import six
|
||||||
|
|
||||||
|
|
||||||
class EmailBackend(BaseEmailBackend):
|
class EmailBackend(BaseEmailBackend):
|
||||||
|
@ -13,6 +14,16 @@ class EmailBackend(BaseEmailBackend):
|
||||||
self._lock = threading.RLock()
|
self._lock = threading.RLock()
|
||||||
super(EmailBackend, self).__init__(*args, **kwargs)
|
super(EmailBackend, self).__init__(*args, **kwargs)
|
||||||
|
|
||||||
|
def write_message(self, message):
|
||||||
|
msg = message.message()
|
||||||
|
msg_data = msg.as_bytes()
|
||||||
|
if six.PY3:
|
||||||
|
charset = msg.get_charset().get_output_charset() if msg.get_charset() else 'utf-8'
|
||||||
|
msg_data = msg_data.decode(charset)
|
||||||
|
self.stream.write('%s\n' % msg_data)
|
||||||
|
self.stream.write('-' * 79)
|
||||||
|
self.stream.write('\n')
|
||||||
|
|
||||||
def send_messages(self, email_messages):
|
def send_messages(self, email_messages):
|
||||||
"""Write all messages to the stream in a thread-safe way."""
|
"""Write all messages to the stream in a thread-safe way."""
|
||||||
if not email_messages:
|
if not email_messages:
|
||||||
|
@ -22,9 +33,7 @@ class EmailBackend(BaseEmailBackend):
|
||||||
try:
|
try:
|
||||||
stream_created = self.open()
|
stream_created = self.open()
|
||||||
for message in email_messages:
|
for message in email_messages:
|
||||||
self.stream.write('%s\n' % message.message().as_string())
|
self.write_message(message)
|
||||||
self.stream.write('-' * 79)
|
|
||||||
self.stream.write('\n')
|
|
||||||
self.stream.flush() # flush after each message
|
self.stream.flush() # flush after each message
|
||||||
msg_count += 1
|
msg_count += 1
|
||||||
if stream_created:
|
if stream_created:
|
||||||
|
|
|
@ -38,6 +38,11 @@ class EmailBackend(ConsoleEmailBackend):
|
||||||
kwargs['stream'] = None
|
kwargs['stream'] = None
|
||||||
super(EmailBackend, self).__init__(*args, **kwargs)
|
super(EmailBackend, self).__init__(*args, **kwargs)
|
||||||
|
|
||||||
|
def write_message(self, message):
|
||||||
|
self.stream.write(message.message().as_bytes() + b'\n')
|
||||||
|
self.stream.write(b'-' * 79)
|
||||||
|
self.stream.write(b'\n')
|
||||||
|
|
||||||
def _get_filename(self):
|
def _get_filename(self):
|
||||||
"""Return a unique file name."""
|
"""Return a unique file name."""
|
||||||
if self._fname is None:
|
if self._fname is None:
|
||||||
|
@ -48,7 +53,7 @@ class EmailBackend(ConsoleEmailBackend):
|
||||||
|
|
||||||
def open(self):
|
def open(self):
|
||||||
if self.stream is None:
|
if self.stream is None:
|
||||||
self.stream = open(self._get_filename(), 'a')
|
self.stream = open(self._get_filename(), 'ab')
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,6 @@ from django.conf import settings
|
||||||
from django.core.mail.backends.base import BaseEmailBackend
|
from django.core.mail.backends.base import BaseEmailBackend
|
||||||
from django.core.mail.utils import DNS_NAME
|
from django.core.mail.utils import DNS_NAME
|
||||||
from django.core.mail.message import sanitize_address
|
from django.core.mail.message import sanitize_address
|
||||||
from django.utils.encoding import force_bytes
|
|
||||||
|
|
||||||
|
|
||||||
class EmailBackend(BaseEmailBackend):
|
class EmailBackend(BaseEmailBackend):
|
||||||
|
@ -111,10 +110,8 @@ class EmailBackend(BaseEmailBackend):
|
||||||
recipients = [sanitize_address(addr, email_message.encoding)
|
recipients = [sanitize_address(addr, email_message.encoding)
|
||||||
for addr in email_message.recipients()]
|
for addr in email_message.recipients()]
|
||||||
message = email_message.message()
|
message = email_message.message()
|
||||||
charset = message.get_charset().get_output_charset() if message.get_charset() else 'utf-8'
|
|
||||||
try:
|
try:
|
||||||
self.connection.sendmail(from_email, recipients,
|
self.connection.sendmail(from_email, recipients, message.as_bytes())
|
||||||
force_bytes(message.as_string(), charset))
|
|
||||||
except smtplib.SMTPException:
|
except smtplib.SMTPException:
|
||||||
if not self.fail_silently:
|
if not self.fail_silently:
|
||||||
raise
|
raise
|
||||||
|
|
|
@ -131,21 +131,25 @@ class MIMEMixin():
|
||||||
This overrides the default as_string() implementation to not mangle
|
This overrides the default as_string() implementation to not mangle
|
||||||
lines that begin with 'From '. See bug #13433 for details.
|
lines that begin with 'From '. See bug #13433 for details.
|
||||||
"""
|
"""
|
||||||
# Using a normal Generator on python 3 will yield a string, which will
|
fp = six.StringIO()
|
||||||
# get base64 encoded in some cases to ensure that it's always convertable
|
g = generator.Generator(fp, mangle_from_=False)
|
||||||
# to ascii. We don't want base64 encoded emails, so we use a BytesGenertor
|
g.flatten(self, unixfrom=unixfrom)
|
||||||
# which will do the right thing and then decode according to our known
|
return fp.getvalue()
|
||||||
# encoding. See #21093 and #3472 for details.
|
|
||||||
if six.PY3 and sys.version_info >= (3, 3, 3):
|
if six.PY2:
|
||||||
|
as_bytes = as_string
|
||||||
|
else:
|
||||||
|
def as_bytes(self, unixfrom=False):
|
||||||
|
"""Return the entire formatted message as bytes.
|
||||||
|
Optional `unixfrom' when True, means include the Unix From_ envelope
|
||||||
|
header.
|
||||||
|
|
||||||
|
This overrides the default as_bytes() implementation to not mangle
|
||||||
|
lines that begin with 'From '. See bug #13433 for details.
|
||||||
|
"""
|
||||||
fp = six.BytesIO()
|
fp = six.BytesIO()
|
||||||
g = generator.BytesGenerator(fp, mangle_from_=False)
|
g = generator.BytesGenerator(fp, mangle_from_=False)
|
||||||
g.flatten(self, unixfrom=unixfrom)
|
g.flatten(self, unixfrom=unixfrom)
|
||||||
encoding = self.get_charset().get_output_charset() if self.get_charset() else 'utf-8'
|
|
||||||
return fp.getvalue().decode(encoding)
|
|
||||||
else:
|
|
||||||
fp = six.StringIO()
|
|
||||||
g = generator.Generator(fp, mangle_from_=False)
|
|
||||||
g.flatten(self, unixfrom=unixfrom)
|
|
||||||
return fp.getvalue()
|
return fp.getvalue()
|
||||||
|
|
||||||
|
|
||||||
|
@ -167,9 +171,8 @@ class SafeMIMEText(MIMEMixin, MIMEText):
|
||||||
# We do it manually and trigger re-encoding of the payload.
|
# We do it manually and trigger re-encoding of the payload.
|
||||||
MIMEText.__init__(self, text, subtype, None)
|
MIMEText.__init__(self, text, subtype, None)
|
||||||
del self['Content-Transfer-Encoding']
|
del self['Content-Transfer-Encoding']
|
||||||
# Work around a bug in python 3.3.3 [sic], see
|
# Workaround for versions without http://bugs.python.org/issue19063
|
||||||
# http://bugs.python.org/issue19063 for details.
|
if (3, 2) < sys.version_info < (3, 3, 4):
|
||||||
if sys.version_info[:3] == (3, 3, 3):
|
|
||||||
payload = text.encode(utf8_charset.output_charset)
|
payload = text.encode(utf8_charset.output_charset)
|
||||||
self._payload = payload.decode('ascii', 'surrogateescape')
|
self._payload = payload.decode('ascii', 'surrogateescape')
|
||||||
self.set_charset(utf8_charset)
|
self.set_charset(utf8_charset)
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
import asyncore
|
import asyncore
|
||||||
from email import message_from_file, message_from_string
|
|
||||||
from email.mime.text import MIMEText
|
from email.mime.text import MIMEText
|
||||||
import os
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
|
@ -20,14 +19,17 @@ from django.core.mail.backends import console, dummy, locmem, filebased, smtp
|
||||||
from django.core.mail.message import BadHeaderError
|
from django.core.mail.message import BadHeaderError
|
||||||
from django.test import SimpleTestCase
|
from django.test import SimpleTestCase
|
||||||
from django.test import override_settings
|
from django.test import override_settings
|
||||||
from django.utils.encoding import force_str, force_text
|
from django.utils.encoding import force_text, force_bytes
|
||||||
from django.utils.six import PY3, StringIO, string_types
|
from django.utils.six import PY3, StringIO, binary_type
|
||||||
from django.utils.translation import ugettext_lazy
|
from django.utils.translation import ugettext_lazy
|
||||||
|
|
||||||
if PY3:
|
if PY3:
|
||||||
from email.utils import parseaddr
|
from email.utils import parseaddr
|
||||||
|
from email import message_from_bytes, message_from_binary_file
|
||||||
else:
|
else:
|
||||||
from email.Utils import parseaddr
|
from email.Utils import parseaddr
|
||||||
|
from email import (message_from_string as message_from_bytes,
|
||||||
|
message_from_file as message_from_binary_file)
|
||||||
|
|
||||||
|
|
||||||
class HeadersCheckMixin(object):
|
class HeadersCheckMixin(object):
|
||||||
|
@ -40,13 +42,9 @@ class HeadersCheckMixin(object):
|
||||||
string with the contens of an email message.
|
string with the contens of an email message.
|
||||||
:param headers: should be a set of (header-name, header-value) tuples.
|
:param headers: should be a set of (header-name, header-value) tuples.
|
||||||
"""
|
"""
|
||||||
if isinstance(message, string_types):
|
if isinstance(message, binary_type):
|
||||||
just_headers = message.split('\n\n', 1)[0]
|
message = message_from_bytes(message)
|
||||||
hlist = just_headers.split('\n')
|
msg_headers = set(message.items())
|
||||||
pairs = [hl.split(':', 1) for hl in hlist]
|
|
||||||
msg_headers = {(n, v.lstrip()) for (n, v) in pairs}
|
|
||||||
else:
|
|
||||||
msg_headers = set(message.items())
|
|
||||||
self.assertTrue(headers.issubset(msg_headers), msg='Message is missing '
|
self.assertTrue(headers.issubset(msg_headers), msg='Message is missing '
|
||||||
'the following headers: %s' % (headers - msg_headers),)
|
'the following headers: %s' % (headers - msg_headers),)
|
||||||
|
|
||||||
|
@ -231,13 +229,13 @@ class MailTests(HeadersCheckMixin, SimpleTestCase):
|
||||||
('MIME-Version', '1.0'),
|
('MIME-Version', '1.0'),
|
||||||
('Content-Type', 'text/plain; charset="iso-8859-1"'),
|
('Content-Type', 'text/plain; charset="iso-8859-1"'),
|
||||||
('Content-Transfer-Encoding', 'quoted-printable')})
|
('Content-Transfer-Encoding', 'quoted-printable')})
|
||||||
self.assertTrue(payload0.as_string().endswith('\n\nFirstname S=FCrname is a great guy.'))
|
self.assertTrue(payload0.as_bytes().endswith(b'\n\nFirstname S=FCrname is a great guy.'))
|
||||||
payload1 = msg.message().get_payload(1)
|
payload1 = msg.message().get_payload(1)
|
||||||
self.assertMessageHasHeaders(payload1, {
|
self.assertMessageHasHeaders(payload1, {
|
||||||
('MIME-Version', '1.0'),
|
('MIME-Version', '1.0'),
|
||||||
('Content-Type', 'text/html; charset="iso-8859-1"'),
|
('Content-Type', 'text/html; charset="iso-8859-1"'),
|
||||||
('Content-Transfer-Encoding', 'quoted-printable')})
|
('Content-Transfer-Encoding', 'quoted-printable')})
|
||||||
self.assertTrue(payload1.as_string().endswith('\n\n<p>Firstname S=FCrname is a <strong>great</strong> guy.</p>'))
|
self.assertTrue(payload1.as_bytes().endswith(b'\n\n<p>Firstname S=FCrname is a <strong>great</strong> guy.</p>'))
|
||||||
|
|
||||||
def test_attachments(self):
|
def test_attachments(self):
|
||||||
"""Regression test for #9367"""
|
"""Regression test for #9367"""
|
||||||
|
@ -248,8 +246,8 @@ class MailTests(HeadersCheckMixin, SimpleTestCase):
|
||||||
msg = EmailMultiAlternatives(subject, text_content, from_email, [to], headers=headers)
|
msg = EmailMultiAlternatives(subject, text_content, from_email, [to], headers=headers)
|
||||||
msg.attach_alternative(html_content, "text/html")
|
msg.attach_alternative(html_content, "text/html")
|
||||||
msg.attach("an attachment.pdf", b"%PDF-1.4.%...", mimetype="application/pdf")
|
msg.attach("an attachment.pdf", b"%PDF-1.4.%...", mimetype="application/pdf")
|
||||||
msg_str = msg.message().as_string()
|
msg_bytes = msg.message().as_bytes()
|
||||||
message = message_from_string(msg_str)
|
message = message_from_bytes(msg_bytes)
|
||||||
self.assertTrue(message.is_multipart())
|
self.assertTrue(message.is_multipart())
|
||||||
self.assertEqual(message.get_content_type(), 'multipart/mixed')
|
self.assertEqual(message.get_content_type(), 'multipart/mixed')
|
||||||
self.assertEqual(message.get_default_type(), 'text/plain')
|
self.assertEqual(message.get_default_type(), 'text/plain')
|
||||||
|
@ -265,8 +263,8 @@ class MailTests(HeadersCheckMixin, SimpleTestCase):
|
||||||
msg = EmailMessage(subject, content, from_email, [to], headers=headers)
|
msg = EmailMessage(subject, content, from_email, [to], headers=headers)
|
||||||
# Unicode in file name
|
# Unicode in file name
|
||||||
msg.attach("une pièce jointe.pdf", b"%PDF-1.4.%...", mimetype="application/pdf")
|
msg.attach("une pièce jointe.pdf", b"%PDF-1.4.%...", mimetype="application/pdf")
|
||||||
msg_str = msg.message().as_string()
|
msg_bytes = msg.message().as_bytes()
|
||||||
message = message_from_string(msg_str)
|
message = message_from_bytes(msg_bytes)
|
||||||
payload = message.get_payload()
|
payload = message.get_payload()
|
||||||
self.assertEqual(payload[1].get_filename(), 'une pièce jointe.pdf')
|
self.assertEqual(payload[1].get_filename(), 'une pièce jointe.pdf')
|
||||||
|
|
||||||
|
@ -348,31 +346,31 @@ class MailTests(HeadersCheckMixin, SimpleTestCase):
|
||||||
# Regression for #13433 - Make sure that EmailMessage doesn't mangle
|
# Regression for #13433 - Make sure that EmailMessage doesn't mangle
|
||||||
# 'From ' in message body.
|
# 'From ' in message body.
|
||||||
email = EmailMessage('Subject', 'From the future', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
email = EmailMessage('Subject', 'From the future', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
||||||
self.assertFalse('>From the future' in email.message().as_string())
|
self.assertFalse(b'>From the future' in email.message().as_bytes())
|
||||||
|
|
||||||
def test_dont_base64_encode(self):
|
def test_dont_base64_encode(self):
|
||||||
# Ticket #3472
|
# Ticket #3472
|
||||||
# Shouldn't use Base64 encoding at all
|
# Shouldn't use Base64 encoding at all
|
||||||
msg = EmailMessage('Subject', 'UTF-8 encoded body', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
msg = EmailMessage('Subject', 'UTF-8 encoded body', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
||||||
self.assertFalse('Content-Transfer-Encoding: base64' in msg.message().as_string())
|
self.assertFalse(b'Content-Transfer-Encoding: base64' in msg.message().as_bytes())
|
||||||
|
|
||||||
# Ticket #11212
|
# Ticket #11212
|
||||||
# Shouldn't use quoted printable, should detect it can represent content with 7 bit data
|
# Shouldn't use quoted printable, should detect it can represent content with 7 bit data
|
||||||
msg = EmailMessage('Subject', 'Body with only ASCII characters.', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
msg = EmailMessage('Subject', 'Body with only ASCII characters.', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
||||||
s = msg.message().as_string()
|
s = msg.message().as_bytes()
|
||||||
self.assertFalse('Content-Transfer-Encoding: quoted-printable' in s)
|
self.assertFalse(b'Content-Transfer-Encoding: quoted-printable' in s)
|
||||||
self.assertTrue('Content-Transfer-Encoding: 7bit' in s)
|
self.assertTrue(b'Content-Transfer-Encoding: 7bit' in s)
|
||||||
|
|
||||||
# Shouldn't use quoted printable, should detect it can represent content with 8 bit data
|
# Shouldn't use quoted printable, should detect it can represent content with 8 bit data
|
||||||
msg = EmailMessage('Subject', 'Body with latin characters: àáä.', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
msg = EmailMessage('Subject', 'Body with latin characters: àáä.', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
||||||
s = msg.message().as_string()
|
s = msg.message().as_bytes()
|
||||||
self.assertFalse(str('Content-Transfer-Encoding: quoted-printable') in s)
|
self.assertFalse(b'Content-Transfer-Encoding: quoted-printable' in s)
|
||||||
self.assertTrue(str('Content-Transfer-Encoding: 8bit') in s)
|
self.assertTrue(b'Content-Transfer-Encoding: 8bit' in s)
|
||||||
|
|
||||||
msg = EmailMessage('Subject', 'Body with non latin characters: А Б В Г Д Е Ж Ѕ З И І К Л М Н О П.', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
msg = EmailMessage('Subject', 'Body with non latin characters: А Б В Г Д Е Ж Ѕ З И І К Л М Н О П.', 'bounce@example.com', ['to@example.com'], headers={'From': 'from@example.com'})
|
||||||
s = msg.message().as_string()
|
s = msg.message().as_bytes()
|
||||||
self.assertFalse(str('Content-Transfer-Encoding: quoted-printable') in s)
|
self.assertFalse(b'Content-Transfer-Encoding: quoted-printable' in s)
|
||||||
self.assertTrue(str('Content-Transfer-Encoding: 8bit') in s)
|
self.assertTrue(b'Content-Transfer-Encoding: 8bit' in s)
|
||||||
|
|
||||||
def test_dont_base64_encode_message_rfc822(self):
|
def test_dont_base64_encode_message_rfc822(self):
|
||||||
# Ticket #18967
|
# Ticket #18967
|
||||||
|
@ -476,7 +474,7 @@ class BaseEmailBackendTests(HeadersCheckMixin, object):
|
||||||
self.assertEqual(num_sent, 1)
|
self.assertEqual(num_sent, 1)
|
||||||
message = self.get_the_message()
|
message = self.get_the_message()
|
||||||
self.assertEqual(message["subject"], '=?utf-8?q?Ch=C3=A8re_maman?=')
|
self.assertEqual(message["subject"], '=?utf-8?q?Ch=C3=A8re_maman?=')
|
||||||
self.assertEqual(force_text(message.get_payload()), 'Je t\'aime très fort')
|
self.assertEqual(force_text(message.get_payload(decode=True)), 'Je t\'aime très fort')
|
||||||
|
|
||||||
def test_send_many(self):
|
def test_send_many(self):
|
||||||
email1 = EmailMessage('Subject', 'Content1', 'from@example.com', ['to@example.com'])
|
email1 = EmailMessage('Subject', 'Content1', 'from@example.com', ['to@example.com'])
|
||||||
|
@ -692,9 +690,9 @@ class FileBackendTests(BaseEmailBackendTests, SimpleTestCase):
|
||||||
def get_mailbox_content(self):
|
def get_mailbox_content(self):
|
||||||
messages = []
|
messages = []
|
||||||
for filename in os.listdir(self.tmp_dir):
|
for filename in os.listdir(self.tmp_dir):
|
||||||
with open(os.path.join(self.tmp_dir, filename), 'r') as fp:
|
with open(os.path.join(self.tmp_dir, filename), 'rb') as fp:
|
||||||
session = force_text(fp.read()).split('\n' + ('-' * 79) + '\n')
|
session = fp.read().split(force_bytes('\n' + ('-' * 79) + '\n', encoding='ascii'))
|
||||||
messages.extend(message_from_string(force_str(m)) for m in session if m)
|
messages.extend(message_from_bytes(m) for m in session if m)
|
||||||
return messages
|
return messages
|
||||||
|
|
||||||
def test_file_sessions(self):
|
def test_file_sessions(self):
|
||||||
|
@ -704,8 +702,8 @@ class FileBackendTests(BaseEmailBackendTests, SimpleTestCase):
|
||||||
connection.send_messages([msg])
|
connection.send_messages([msg])
|
||||||
|
|
||||||
self.assertEqual(len(os.listdir(self.tmp_dir)), 1)
|
self.assertEqual(len(os.listdir(self.tmp_dir)), 1)
|
||||||
with open(os.path.join(self.tmp_dir, os.listdir(self.tmp_dir)[0])) as fp:
|
with open(os.path.join(self.tmp_dir, os.listdir(self.tmp_dir)[0]), 'rb') as fp:
|
||||||
message = message_from_file(fp)
|
message = message_from_binary_file(fp)
|
||||||
self.assertEqual(message.get_content_type(), 'text/plain')
|
self.assertEqual(message.get_content_type(), 'text/plain')
|
||||||
self.assertEqual(message.get('subject'), 'Subject')
|
self.assertEqual(message.get('subject'), 'Subject')
|
||||||
self.assertEqual(message.get('from'), 'from@example.com')
|
self.assertEqual(message.get('from'), 'from@example.com')
|
||||||
|
@ -746,8 +744,8 @@ class ConsoleBackendTests(BaseEmailBackendTests, SimpleTestCase):
|
||||||
self.stream = sys.stdout = StringIO()
|
self.stream = sys.stdout = StringIO()
|
||||||
|
|
||||||
def get_mailbox_content(self):
|
def get_mailbox_content(self):
|
||||||
messages = force_text(self.stream.getvalue()).split('\n' + ('-' * 79) + '\n')
|
messages = self.stream.getvalue().split(str('\n' + ('-' * 79) + '\n'))
|
||||||
return [message_from_string(force_str(m)) for m in messages if m]
|
return [message_from_bytes(force_bytes(m)) for m in messages if m]
|
||||||
|
|
||||||
def test_console_stream_kwarg(self):
|
def test_console_stream_kwarg(self):
|
||||||
"""
|
"""
|
||||||
|
@ -756,14 +754,15 @@ class ConsoleBackendTests(BaseEmailBackendTests, SimpleTestCase):
|
||||||
s = StringIO()
|
s = StringIO()
|
||||||
connection = mail.get_connection('django.core.mail.backends.console.EmailBackend', stream=s)
|
connection = mail.get_connection('django.core.mail.backends.console.EmailBackend', stream=s)
|
||||||
send_mail('Subject', 'Content', 'from@example.com', ['to@example.com'], connection=connection)
|
send_mail('Subject', 'Content', 'from@example.com', ['to@example.com'], connection=connection)
|
||||||
self.assertMessageHasHeaders(s.getvalue(), {
|
message = force_bytes(s.getvalue().split('\n' + ('-' * 79) + '\n')[0])
|
||||||
|
self.assertMessageHasHeaders(message, {
|
||||||
('MIME-Version', '1.0'),
|
('MIME-Version', '1.0'),
|
||||||
('Content-Type', 'text/plain; charset="utf-8"'),
|
('Content-Type', 'text/plain; charset="utf-8"'),
|
||||||
('Content-Transfer-Encoding', '7bit'),
|
('Content-Transfer-Encoding', '7bit'),
|
||||||
('Subject', 'Subject'),
|
('Subject', 'Subject'),
|
||||||
('From', 'from@example.com'),
|
('From', 'from@example.com'),
|
||||||
('To', 'to@example.com')})
|
('To', 'to@example.com')})
|
||||||
self.assertIn('\nDate: ', s.getvalue())
|
self.assertIn(b'\nDate: ', message)
|
||||||
|
|
||||||
|
|
||||||
class FakeSMTPChannel(smtpd.SMTPChannel):
|
class FakeSMTPChannel(smtpd.SMTPChannel):
|
||||||
|
@ -793,7 +792,9 @@ class FakeSMTPServer(smtpd.SMTPServer, threading.Thread):
|
||||||
self.sink_lock = threading.Lock()
|
self.sink_lock = threading.Lock()
|
||||||
|
|
||||||
def process_message(self, peer, mailfrom, rcpttos, data):
|
def process_message(self, peer, mailfrom, rcpttos, data):
|
||||||
m = message_from_string(data)
|
if PY3:
|
||||||
|
data = data.encode('utf-8')
|
||||||
|
m = message_from_bytes(data)
|
||||||
maddr = parseaddr(m.get('from'))[1]
|
maddr = parseaddr(m.get('from'))[1]
|
||||||
if mailfrom != maddr:
|
if mailfrom != maddr:
|
||||||
return "553 '%s' != '%s'" % (mailfrom, maddr)
|
return "553 '%s' != '%s'" % (mailfrom, maddr)
|
||||||
|
|
Loading…
Reference in New Issue