2011-04-18 14:59:37 -03:00
|
|
|
import io
|
|
|
|
import textwrap
|
|
|
|
import unittest
|
|
|
|
from email import message_from_string, message_from_bytes
|
2015-05-17 12:29:21 -03:00
|
|
|
from email.message import EmailMessage
|
2011-04-18 14:59:37 -03:00
|
|
|
from email.generator import Generator, BytesGenerator
|
|
|
|
from email import policy
|
2012-05-31 19:00:45 -03:00
|
|
|
from test.test_email import TestEmailBase, parameterize
|
2011-04-18 14:59:37 -03:00
|
|
|
|
|
|
|
|
2012-05-31 19:00:45 -03:00
|
|
|
@parameterize
|
|
|
|
class TestGeneratorBase:
|
2011-04-18 14:59:37 -03:00
|
|
|
|
2012-05-25 19:42:14 -03:00
|
|
|
policy = policy.default
|
2011-04-18 14:59:37 -03:00
|
|
|
|
2012-05-25 19:42:14 -03:00
|
|
|
def msgmaker(self, msg, policy=None):
|
|
|
|
policy = self.policy if policy is None else policy
|
|
|
|
return self.msgfunc(msg, policy=policy)
|
2012-05-25 16:01:48 -03:00
|
|
|
|
2012-05-25 19:42:14 -03:00
|
|
|
refold_long_expected = {
|
2011-04-18 14:59:37 -03:00
|
|
|
0: textwrap.dedent("""\
|
|
|
|
To: whom_it_may_concern@example.com
|
|
|
|
From: nobody_you_want_to_know@example.com
|
|
|
|
Subject: We the willing led by the unknowing are doing the
|
|
|
|
impossible for the ungrateful. We have done so much for so long with so little
|
|
|
|
we are now qualified to do anything with nothing.
|
|
|
|
|
|
|
|
None
|
|
|
|
"""),
|
|
|
|
40: textwrap.dedent("""\
|
|
|
|
To: whom_it_may_concern@example.com
|
2012-05-25 19:42:14 -03:00
|
|
|
From:
|
2011-04-18 14:59:37 -03:00
|
|
|
nobody_you_want_to_know@example.com
|
|
|
|
Subject: We the willing led by the
|
2012-05-25 19:42:14 -03:00
|
|
|
unknowing are doing the impossible for
|
|
|
|
the ungrateful. We have done so much
|
|
|
|
for so long with so little we are now
|
|
|
|
qualified to do anything with nothing.
|
2011-04-18 14:59:37 -03:00
|
|
|
|
|
|
|
None
|
|
|
|
"""),
|
|
|
|
20: textwrap.dedent("""\
|
2017-12-03 19:51:41 -04:00
|
|
|
To:
|
|
|
|
whom_it_may_concern@example.com
|
|
|
|
From:
|
|
|
|
nobody_you_want_to_know@example.com
|
2011-04-18 14:59:37 -03:00
|
|
|
Subject: We the
|
|
|
|
willing led by the
|
|
|
|
unknowing are doing
|
2012-05-25 19:42:14 -03:00
|
|
|
the impossible for
|
|
|
|
the ungrateful. We
|
|
|
|
have done so much
|
|
|
|
for so long with so
|
|
|
|
little we are now
|
2011-04-18 14:59:37 -03:00
|
|
|
qualified to do
|
|
|
|
anything with
|
|
|
|
nothing.
|
|
|
|
|
|
|
|
None
|
|
|
|
"""),
|
|
|
|
}
|
2012-05-25 19:42:14 -03:00
|
|
|
refold_long_expected[100] = refold_long_expected[0]
|
|
|
|
|
|
|
|
refold_all_expected = refold_long_expected.copy()
|
|
|
|
refold_all_expected[0] = (
|
|
|
|
"To: whom_it_may_concern@example.com\n"
|
|
|
|
"From: nobody_you_want_to_know@example.com\n"
|
|
|
|
"Subject: We the willing led by the unknowing are doing the "
|
|
|
|
"impossible for the ungrateful. We have done so much for "
|
|
|
|
"so long with so little we are now qualified to do anything "
|
|
|
|
"with nothing.\n"
|
|
|
|
"\n"
|
|
|
|
"None\n")
|
|
|
|
refold_all_expected[100] = (
|
|
|
|
"To: whom_it_may_concern@example.com\n"
|
|
|
|
"From: nobody_you_want_to_know@example.com\n"
|
|
|
|
"Subject: We the willing led by the unknowing are doing the "
|
|
|
|
"impossible for the ungrateful. We have\n"
|
|
|
|
" done so much for so long with so little we are now qualified "
|
|
|
|
"to do anything with nothing.\n"
|
|
|
|
"\n"
|
|
|
|
"None\n")
|
|
|
|
|
2012-05-30 22:53:40 -03:00
|
|
|
length_params = [n for n in refold_long_expected]
|
|
|
|
|
|
|
|
def length_as_maxheaderlen_parameter(self, n):
|
2012-05-25 19:42:14 -03:00
|
|
|
msg = self.msgmaker(self.typ(self.refold_long_expected[0]))
|
2011-04-18 14:59:37 -03:00
|
|
|
s = self.ioclass()
|
2012-05-25 19:42:14 -03:00
|
|
|
g = self.genclass(s, maxheaderlen=n, policy=self.policy)
|
2011-04-18 14:59:37 -03:00
|
|
|
g.flatten(msg)
|
2012-05-25 19:42:14 -03:00
|
|
|
self.assertEqual(s.getvalue(), self.typ(self.refold_long_expected[n]))
|
2011-04-18 14:59:37 -03:00
|
|
|
|
2012-05-30 22:53:40 -03:00
|
|
|
def length_as_max_line_length_policy(self, n):
|
2012-05-25 19:42:14 -03:00
|
|
|
msg = self.msgmaker(self.typ(self.refold_long_expected[0]))
|
2011-04-18 14:59:37 -03:00
|
|
|
s = self.ioclass()
|
2012-05-25 19:42:14 -03:00
|
|
|
g = self.genclass(s, policy=self.policy.clone(max_line_length=n))
|
2011-04-18 14:59:37 -03:00
|
|
|
g.flatten(msg)
|
2012-05-25 19:42:14 -03:00
|
|
|
self.assertEqual(s.getvalue(), self.typ(self.refold_long_expected[n]))
|
2011-04-18 14:59:37 -03:00
|
|
|
|
2012-05-30 22:53:40 -03:00
|
|
|
def length_as_maxheaderlen_parm_overrides_policy(self, n):
|
2012-05-25 19:42:14 -03:00
|
|
|
msg = self.msgmaker(self.typ(self.refold_long_expected[0]))
|
2011-04-18 14:59:37 -03:00
|
|
|
s = self.ioclass()
|
|
|
|
g = self.genclass(s, maxheaderlen=n,
|
2012-05-25 19:42:14 -03:00
|
|
|
policy=self.policy.clone(max_line_length=10))
|
2011-04-18 14:59:37 -03:00
|
|
|
g.flatten(msg)
|
2012-05-25 19:42:14 -03:00
|
|
|
self.assertEqual(s.getvalue(), self.typ(self.refold_long_expected[n]))
|
2011-04-18 14:59:37 -03:00
|
|
|
|
2012-05-30 22:53:40 -03:00
|
|
|
def length_as_max_line_length_with_refold_none_does_not_fold(self, n):
|
2012-05-25 19:42:14 -03:00
|
|
|
msg = self.msgmaker(self.typ(self.refold_long_expected[0]))
|
|
|
|
s = self.ioclass()
|
|
|
|
g = self.genclass(s, policy=self.policy.clone(refold_source='none',
|
|
|
|
max_line_length=n))
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), self.typ(self.refold_long_expected[0]))
|
|
|
|
|
2012-05-30 22:53:40 -03:00
|
|
|
def length_as_max_line_length_with_refold_all_folds(self, n):
|
2012-05-25 19:42:14 -03:00
|
|
|
msg = self.msgmaker(self.typ(self.refold_long_expected[0]))
|
|
|
|
s = self.ioclass()
|
|
|
|
g = self.genclass(s, policy=self.policy.clone(refold_source='all',
|
|
|
|
max_line_length=n))
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), self.typ(self.refold_all_expected[n]))
|
2011-04-18 14:59:37 -03:00
|
|
|
|
2012-05-25 16:01:48 -03:00
|
|
|
def test_crlf_control_via_policy(self):
|
|
|
|
source = "Subject: test\r\n\r\ntest body\r\n"
|
|
|
|
expected = source
|
|
|
|
msg = self.msgmaker(self.typ(source))
|
|
|
|
s = self.ioclass()
|
|
|
|
g = self.genclass(s, policy=policy.SMTP)
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), self.typ(expected))
|
|
|
|
|
|
|
|
def test_flatten_linesep_overrides_policy(self):
|
|
|
|
source = "Subject: test\n\ntest body\n"
|
|
|
|
expected = source
|
|
|
|
msg = self.msgmaker(self.typ(source))
|
|
|
|
s = self.ioclass()
|
|
|
|
g = self.genclass(s, policy=policy.SMTP)
|
|
|
|
g.flatten(msg, linesep='\n')
|
|
|
|
self.assertEqual(s.getvalue(), self.typ(expected))
|
|
|
|
|
2015-05-17 15:24:33 -03:00
|
|
|
def test_set_mangle_from_via_policy(self):
|
|
|
|
source = textwrap.dedent("""\
|
|
|
|
Subject: test that
|
2016-08-30 14:47:49 -03:00
|
|
|
from is mangled in the body!
|
2015-05-17 15:24:33 -03:00
|
|
|
|
|
|
|
From time to time I write a rhyme.
|
|
|
|
""")
|
|
|
|
variants = (
|
|
|
|
(None, True),
|
|
|
|
(policy.compat32, True),
|
|
|
|
(policy.default, False),
|
|
|
|
(policy.default.clone(mangle_from_=True), True),
|
|
|
|
)
|
|
|
|
for p, mangle in variants:
|
|
|
|
expected = source.replace('From ', '>From ') if mangle else source
|
|
|
|
with self.subTest(policy=p, mangle_from_=mangle):
|
|
|
|
msg = self.msgmaker(self.typ(source))
|
|
|
|
s = self.ioclass()
|
|
|
|
g = self.genclass(s, policy=p)
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), self.typ(expected))
|
|
|
|
|
2017-06-12 03:43:41 -03:00
|
|
|
def test_compat32_max_line_length_does_not_fold_when_none(self):
|
|
|
|
msg = self.msgmaker(self.typ(self.refold_long_expected[0]))
|
|
|
|
s = self.ioclass()
|
|
|
|
g = self.genclass(s, policy=policy.compat32.clone(max_line_length=None))
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), self.typ(self.refold_long_expected[0]))
|
|
|
|
|
2017-12-03 19:51:41 -04:00
|
|
|
def test_rfc2231_wrapping(self):
|
|
|
|
# This is pretty much just to make sure we don't have an infinite
|
|
|
|
# loop; I don't expect anyone to hit this in the field.
|
|
|
|
msg = self.msgmaker(self.typ(textwrap.dedent("""\
|
|
|
|
To: nobody
|
|
|
|
Content-Disposition: attachment;
|
|
|
|
filename="afilenamelongenoghtowraphere"
|
|
|
|
|
|
|
|
None
|
|
|
|
""")))
|
|
|
|
expected = textwrap.dedent("""\
|
|
|
|
To: nobody
|
|
|
|
Content-Disposition: attachment;
|
|
|
|
filename*0*=us-ascii''afilename;
|
|
|
|
filename*1*=longenoghtowraphere
|
|
|
|
|
|
|
|
None
|
|
|
|
""")
|
|
|
|
s = self.ioclass()
|
|
|
|
g = self.genclass(s, policy=self.policy.clone(max_line_length=33))
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), self.typ(expected))
|
|
|
|
|
|
|
|
def test_rfc2231_wrapping_switches_to_default_len_if_too_narrow(self):
|
|
|
|
# This is just to make sure we don't have an infinite loop; I don't
|
|
|
|
# expect anyone to hit this in the field, so I'm not bothering to make
|
|
|
|
# the result optimal (the encoding isn't needed).
|
|
|
|
msg = self.msgmaker(self.typ(textwrap.dedent("""\
|
|
|
|
To: nobody
|
|
|
|
Content-Disposition: attachment;
|
|
|
|
filename="afilenamelongenoghtowraphere"
|
|
|
|
|
|
|
|
None
|
|
|
|
""")))
|
|
|
|
expected = textwrap.dedent("""\
|
|
|
|
To: nobody
|
|
|
|
Content-Disposition:
|
|
|
|
attachment;
|
|
|
|
filename*0*=us-ascii''afilenamelongenoghtowraphere
|
|
|
|
|
|
|
|
None
|
|
|
|
""")
|
|
|
|
s = self.ioclass()
|
|
|
|
g = self.genclass(s, policy=self.policy.clone(max_line_length=20))
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), self.typ(expected))
|
|
|
|
|
2011-04-18 14:59:37 -03:00
|
|
|
|
|
|
|
class TestGenerator(TestGeneratorBase, TestEmailBase):
|
|
|
|
|
2012-05-25 19:42:14 -03:00
|
|
|
msgfunc = staticmethod(message_from_string)
|
2011-04-18 14:59:37 -03:00
|
|
|
genclass = Generator
|
|
|
|
ioclass = io.StringIO
|
2012-05-25 16:01:48 -03:00
|
|
|
typ = str
|
|
|
|
|
2011-04-18 14:59:37 -03:00
|
|
|
|
|
|
|
class TestBytesGenerator(TestGeneratorBase, TestEmailBase):
|
|
|
|
|
2012-05-25 19:42:14 -03:00
|
|
|
msgfunc = staticmethod(message_from_bytes)
|
2011-04-18 14:59:37 -03:00
|
|
|
genclass = BytesGenerator
|
|
|
|
ioclass = io.BytesIO
|
2012-05-25 16:01:48 -03:00
|
|
|
typ = lambda self, x: x.encode('ascii')
|
|
|
|
|
|
|
|
def test_cte_type_7bit_handles_unknown_8bit(self):
|
|
|
|
source = ("Subject: Maintenant je vous présente mon "
|
|
|
|
"collègue\n\n").encode('utf-8')
|
2012-05-25 19:42:14 -03:00
|
|
|
expected = ('Subject: Maintenant je vous =?unknown-8bit?q?'
|
|
|
|
'pr=C3=A9sente_mon_coll=C3=A8gue?=\n\n').encode('ascii')
|
2012-05-25 16:01:48 -03:00
|
|
|
msg = message_from_bytes(source)
|
|
|
|
s = io.BytesIO()
|
|
|
|
g = BytesGenerator(s, policy=self.policy.clone(cte_type='7bit'))
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), expected)
|
|
|
|
|
|
|
|
def test_cte_type_7bit_transforms_8bit_cte(self):
|
|
|
|
source = textwrap.dedent("""\
|
|
|
|
From: foo@bar.com
|
|
|
|
To: Dinsdale
|
|
|
|
Subject: Nudge nudge, wink, wink
|
|
|
|
Mime-Version: 1.0
|
|
|
|
Content-Type: text/plain; charset="latin-1"
|
|
|
|
Content-Transfer-Encoding: 8bit
|
|
|
|
|
|
|
|
oh là là, know what I mean, know what I mean?
|
|
|
|
""").encode('latin1')
|
|
|
|
msg = message_from_bytes(source)
|
|
|
|
expected = textwrap.dedent("""\
|
|
|
|
From: foo@bar.com
|
|
|
|
To: Dinsdale
|
|
|
|
Subject: Nudge nudge, wink, wink
|
|
|
|
Mime-Version: 1.0
|
|
|
|
Content-Type: text/plain; charset="iso-8859-1"
|
|
|
|
Content-Transfer-Encoding: quoted-printable
|
|
|
|
|
|
|
|
oh l=E0 l=E0, know what I mean, know what I mean?
|
|
|
|
""").encode('ascii')
|
|
|
|
s = io.BytesIO()
|
|
|
|
g = BytesGenerator(s, policy=self.policy.clone(cte_type='7bit',
|
|
|
|
linesep='\n'))
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), expected)
|
2011-04-18 14:59:37 -03:00
|
|
|
|
2015-05-17 12:29:21 -03:00
|
|
|
def test_smtputf8_policy(self):
|
|
|
|
msg = EmailMessage()
|
|
|
|
msg['From'] = "Páolo <főo@bar.com>"
|
|
|
|
msg['To'] = 'Dinsdale'
|
|
|
|
msg['Subject'] = 'Nudge nudge, wink, wink \u1F609'
|
|
|
|
msg.set_content("oh là là, know what I mean, know what I mean?")
|
|
|
|
expected = textwrap.dedent("""\
|
|
|
|
From: Páolo <főo@bar.com>
|
|
|
|
To: Dinsdale
|
|
|
|
Subject: Nudge nudge, wink, wink \u1F609
|
|
|
|
Content-Type: text/plain; charset="utf-8"
|
|
|
|
Content-Transfer-Encoding: 8bit
|
|
|
|
MIME-Version: 1.0
|
|
|
|
|
|
|
|
oh là là, know what I mean, know what I mean?
|
|
|
|
""").encode('utf-8').replace(b'\n', b'\r\n')
|
|
|
|
s = io.BytesIO()
|
|
|
|
g = BytesGenerator(s, policy=policy.SMTPUTF8)
|
|
|
|
g.flatten(msg)
|
|
|
|
self.assertEqual(s.getvalue(), expected)
|
|
|
|
|
2011-04-18 14:59:37 -03:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|