| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  | import io | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  | import types | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  | import textwrap | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  | import unittest | 
					
						
							| 
									
										
										
										
											2019-07-16 19:50:01 +02:00
										 |  |  | import email.errors | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  | import email.policy | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  | import email.parser | 
					
						
							|  |  |  | import email.generator | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  | import email.message | 
					
						
							| 
									
										
										
										
											2012-05-27 15:03:38 -04:00
										 |  |  | from email import headerregistry | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | def make_defaults(base_defaults, differences): | 
					
						
							|  |  |  |     defaults = base_defaults.copy() | 
					
						
							|  |  |  |     defaults.update(differences) | 
					
						
							|  |  |  |     return defaults | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | class PolicyAPITests(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     longMessage = True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |     # Base default values. | 
					
						
							|  |  |  |     compat32_defaults = { | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |         'max_line_length':          78, | 
					
						
							|  |  |  |         'linesep':                  '\n', | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  |         'cte_type':                 '8bit', | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |         'raise_on_defect':          False, | 
					
						
							| 
									
										
										
										
											2015-05-17 14:24:33 -04:00
										 |  |  |         'mangle_from_':             True, | 
					
						
							| 
									
										
										
										
											2016-09-10 00:22:25 -04:00
										 |  |  |         'message_factory':          None, | 
					
						
							| 
									
										
										
										
											2024-07-31 00:19:48 +02:00
										 |  |  |         'verify_generated_headers': True, | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |     # These default values are the ones set on email.policy.default. | 
					
						
							|  |  |  |     # If any of these defaults change, the docs must be updated. | 
					
						
							|  |  |  |     policy_defaults = compat32_defaults.copy() | 
					
						
							|  |  |  |     policy_defaults.update({ | 
					
						
							| 
									
										
										
										
											2015-05-17 11:29:21 -04:00
										 |  |  |         'utf8':                     False, | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |         'raise_on_defect':          False, | 
					
						
							|  |  |  |         'header_factory':           email.policy.EmailPolicy.header_factory, | 
					
						
							|  |  |  |         'refold_source':            'long', | 
					
						
							| 
									
										
										
										
											2013-10-16 22:48:40 -04:00
										 |  |  |         'content_manager':          email.policy.EmailPolicy.content_manager, | 
					
						
							| 
									
										
										
										
											2015-05-17 14:24:33 -04:00
										 |  |  |         'mangle_from_':             False, | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |         'message_factory':          email.message.EmailMessage, | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # For each policy under test, we give here what we expect the defaults to | 
					
						
							|  |  |  |     # be for that policy.  The second argument to make defaults is the | 
					
						
							|  |  |  |     # difference between the base defaults and that for the particular policy. | 
					
						
							|  |  |  |     new_policy = email.policy.EmailPolicy() | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |     policies = { | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |         email.policy.compat32: make_defaults(compat32_defaults, {}), | 
					
						
							|  |  |  |         email.policy.default: make_defaults(policy_defaults, {}), | 
					
						
							|  |  |  |         email.policy.SMTP: make_defaults(policy_defaults, | 
					
						
							|  |  |  |                                          {'linesep': '\r\n'}), | 
					
						
							| 
									
										
										
										
											2015-05-17 11:29:21 -04:00
										 |  |  |         email.policy.SMTPUTF8: make_defaults(policy_defaults, | 
					
						
							|  |  |  |                                              {'linesep': '\r\n', | 
					
						
							|  |  |  |                                               'utf8': True}), | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |         email.policy.HTTP: make_defaults(policy_defaults, | 
					
						
							|  |  |  |                                          {'linesep': '\r\n', | 
					
						
							|  |  |  |                                           'max_line_length': None}), | 
					
						
							|  |  |  |         email.policy.strict: make_defaults(policy_defaults, | 
					
						
							|  |  |  |                                            {'raise_on_defect': True}), | 
					
						
							|  |  |  |         new_policy: make_defaults(policy_defaults, {}), | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |     # Creating a new policy creates a new header factory.  There is a test | 
					
						
							|  |  |  |     # later that proves this. | 
					
						
							|  |  |  |     policies[new_policy]['header_factory'] = new_policy.header_factory | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_defaults(self): | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |         for policy, expected in self.policies.items(): | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |             for attr, value in expected.items(): | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |                 with self.subTest(policy=policy, attr=attr): | 
					
						
							|  |  |  |                     self.assertEqual(getattr(policy, attr), value, | 
					
						
							|  |  |  |                                     ("change {} docs/docstrings if defaults have " | 
					
						
							|  |  |  |                                     "changed").format(policy)) | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_all_attributes_covered(self): | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |         for policy, expected in self.policies.items(): | 
					
						
							|  |  |  |             for attr in dir(policy): | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |                 with self.subTest(policy=policy, attr=attr): | 
					
						
							|  |  |  |                     if (attr.startswith('_') or | 
					
						
							|  |  |  |                             isinstance(getattr(email.policy.EmailPolicy, attr), | 
					
						
							|  |  |  |                                   types.FunctionType)): | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         self.assertIn(attr, expected, | 
					
						
							|  |  |  |                                       "{} is not fully tested".format(attr)) | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  |     def test_abc(self): | 
					
						
							|  |  |  |         with self.assertRaises(TypeError) as cm: | 
					
						
							|  |  |  |             email.policy.Policy() | 
					
						
							|  |  |  |         msg = str(cm.exception) | 
					
						
							|  |  |  |         abstract_methods = ('fold', | 
					
						
							|  |  |  |                             'fold_binary', | 
					
						
							|  |  |  |                             'header_fetch_parse', | 
					
						
							|  |  |  |                             'header_source_parse', | 
					
						
							|  |  |  |                             'header_store_parse') | 
					
						
							|  |  |  |         for method in abstract_methods: | 
					
						
							|  |  |  |             self.assertIn(method, msg) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |     def test_policy_is_immutable(self): | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |         for policy, defaults in self.policies.items(): | 
					
						
							|  |  |  |             for attr in defaults: | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |                 with self.assertRaisesRegex(AttributeError, attr+".*read-only"): | 
					
						
							|  |  |  |                     setattr(policy, attr, None) | 
					
						
							|  |  |  |             with self.assertRaisesRegex(AttributeError, 'no attribute.*foo'): | 
					
						
							|  |  |  |                 policy.foo = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |     def test_set_policy_attrs_when_cloned(self): | 
					
						
							|  |  |  |         # None of the attributes has a default value of None, so we set them | 
					
						
							|  |  |  |         # all to None in the clone call and check that it worked. | 
					
						
							|  |  |  |         for policyclass, defaults in self.policies.items(): | 
					
						
							|  |  |  |             testattrdict = {attr: None for attr in defaults} | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |             policy = policyclass.clone(**testattrdict) | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |             for attr in defaults: | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |                 self.assertIsNone(getattr(policy, attr)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_reject_non_policy_keyword_when_called(self): | 
					
						
							|  |  |  |         for policyclass in self.policies: | 
					
						
							|  |  |  |             with self.assertRaises(TypeError): | 
					
						
							|  |  |  |                 policyclass(this_keyword_should_not_be_valid=None) | 
					
						
							|  |  |  |             with self.assertRaises(TypeError): | 
					
						
							|  |  |  |                 policyclass(newtline=None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_policy_addition(self): | 
					
						
							|  |  |  |         expected = self.policy_defaults.copy() | 
					
						
							|  |  |  |         p1 = email.policy.default.clone(max_line_length=100) | 
					
						
							|  |  |  |         p2 = email.policy.default.clone(max_line_length=50) | 
					
						
							|  |  |  |         added = p1 + p2 | 
					
						
							|  |  |  |         expected.update(max_line_length=50) | 
					
						
							|  |  |  |         for attr, value in expected.items(): | 
					
						
							|  |  |  |             self.assertEqual(getattr(added, attr), value) | 
					
						
							|  |  |  |         added = p2 + p1 | 
					
						
							|  |  |  |         expected.update(max_line_length=100) | 
					
						
							|  |  |  |         for attr, value in expected.items(): | 
					
						
							|  |  |  |             self.assertEqual(getattr(added, attr), value) | 
					
						
							|  |  |  |         added = added + email.policy.default | 
					
						
							|  |  |  |         for attr, value in expected.items(): | 
					
						
							|  |  |  |             self.assertEqual(getattr(added, attr), value) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-27 00:19:41 +09:00
										 |  |  |     def test_fold_utf8(self): | 
					
						
							|  |  |  |         expected_ascii = 'Subject: =?utf-8?q?=C3=A1?=\n' | 
					
						
							|  |  |  |         expected_utf8 = 'Subject: á\n' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         msg = email.message.EmailMessage() | 
					
						
							|  |  |  |         s = 'á' | 
					
						
							|  |  |  |         msg['Subject'] = s | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         p_ascii = email.policy.default.clone() | 
					
						
							|  |  |  |         p_utf8 = email.policy.default.clone(utf8=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(p_ascii.fold('Subject', msg['Subject']), expected_ascii) | 
					
						
							|  |  |  |         self.assertEqual(p_utf8.fold('Subject', msg['Subject']), expected_utf8) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(p_ascii.fold('Subject', s), expected_ascii) | 
					
						
							|  |  |  |         self.assertEqual(p_utf8.fold('Subject', s), expected_utf8) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 12:28:44 -07:00
										 |  |  |     def test_fold_zero_max_line_length(self): | 
					
						
							|  |  |  |         expected = 'Subject: =?utf-8?q?=C3=A1?=\n' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         msg = email.message.EmailMessage() | 
					
						
							|  |  |  |         msg['Subject'] = 'á' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         p1 = email.policy.default.clone(max_line_length=0) | 
					
						
							|  |  |  |         p2 = email.policy.default.clone(max_line_length=None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(p1.fold('Subject', msg['Subject']), expected) | 
					
						
							|  |  |  |         self.assertEqual(p2.fold('Subject', msg['Subject']), expected) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |     def test_register_defect(self): | 
					
						
							|  |  |  |         class Dummy: | 
					
						
							|  |  |  |             def __init__(self): | 
					
						
							|  |  |  |                 self.defects = [] | 
					
						
							|  |  |  |         obj = Dummy() | 
					
						
							|  |  |  |         defect = object() | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |         policy = email.policy.EmailPolicy() | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |         policy.register_defect(obj, defect) | 
					
						
							|  |  |  |         self.assertEqual(obj.defects, [defect]) | 
					
						
							|  |  |  |         defect2 = object() | 
					
						
							|  |  |  |         policy.register_defect(obj, defect2) | 
					
						
							|  |  |  |         self.assertEqual(obj.defects, [defect, defect2]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     class MyObj: | 
					
						
							|  |  |  |         def __init__(self): | 
					
						
							|  |  |  |             self.defects = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     class MyDefect(Exception): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_handle_defect_raises_on_strict(self): | 
					
						
							|  |  |  |         foo = self.MyObj() | 
					
						
							|  |  |  |         defect = self.MyDefect("the telly is broken") | 
					
						
							|  |  |  |         with self.assertRaisesRegex(self.MyDefect, "the telly is broken"): | 
					
						
							|  |  |  |             email.policy.strict.handle_defect(foo, defect) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_handle_defect_registers_defect(self): | 
					
						
							|  |  |  |         foo = self.MyObj() | 
					
						
							|  |  |  |         defect1 = self.MyDefect("one") | 
					
						
							|  |  |  |         email.policy.default.handle_defect(foo, defect1) | 
					
						
							|  |  |  |         self.assertEqual(foo.defects, [defect1]) | 
					
						
							|  |  |  |         defect2 = self.MyDefect("two") | 
					
						
							|  |  |  |         email.policy.default.handle_defect(foo, defect2) | 
					
						
							|  |  |  |         self.assertEqual(foo.defects, [defect1, defect2]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |     class MyPolicy(email.policy.EmailPolicy): | 
					
						
							| 
									
										
										
										
											2012-03-16 21:39:57 -04:00
										 |  |  |         defects = None | 
					
						
							|  |  |  |         def __init__(self, *args, **kw): | 
					
						
							|  |  |  |             super().__init__(*args, defects=[], **kw) | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |         def register_defect(self, obj, defect): | 
					
						
							|  |  |  |             self.defects.append(defect) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_overridden_register_defect_still_raises(self): | 
					
						
							|  |  |  |         foo = self.MyObj() | 
					
						
							|  |  |  |         defect = self.MyDefect("the telly is broken") | 
					
						
							|  |  |  |         with self.assertRaisesRegex(self.MyDefect, "the telly is broken"): | 
					
						
							|  |  |  |             self.MyPolicy(raise_on_defect=True).handle_defect(foo, defect) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-02 10:07:09 +00:00
										 |  |  |     def test_overridden_register_defect_works(self): | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |         foo = self.MyObj() | 
					
						
							|  |  |  |         defect1 = self.MyDefect("one") | 
					
						
							|  |  |  |         my_policy = self.MyPolicy() | 
					
						
							|  |  |  |         my_policy.handle_defect(foo, defect1) | 
					
						
							|  |  |  |         self.assertEqual(my_policy.defects, [defect1]) | 
					
						
							|  |  |  |         self.assertEqual(foo.defects, []) | 
					
						
							|  |  |  |         defect2 = self.MyDefect("two") | 
					
						
							|  |  |  |         my_policy.handle_defect(foo, defect2) | 
					
						
							|  |  |  |         self.assertEqual(my_policy.defects, [defect1, defect2]) | 
					
						
							|  |  |  |         self.assertEqual(foo.defects, []) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |     def test_default_header_factory(self): | 
					
						
							|  |  |  |         h = email.policy.default.header_factory('Test', 'test') | 
					
						
							|  |  |  |         self.assertEqual(h.name, 'Test') | 
					
						
							| 
									
										
										
										
											2012-05-27 15:03:38 -04:00
										 |  |  |         self.assertIsInstance(h, headerregistry.UnstructuredHeader) | 
					
						
							|  |  |  |         self.assertIsInstance(h, headerregistry.BaseHeader) | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     class Foo: | 
					
						
							| 
									
										
										
										
											2012-05-27 15:03:38 -04:00
										 |  |  |         parse = headerregistry.UnstructuredHeader.parse | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_each_Policy_gets_unique_factory(self): | 
					
						
							|  |  |  |         policy1 = email.policy.EmailPolicy() | 
					
						
							|  |  |  |         policy2 = email.policy.EmailPolicy() | 
					
						
							|  |  |  |         policy1.header_factory.map_to_type('foo', self.Foo) | 
					
						
							|  |  |  |         h = policy1.header_factory('foo', 'test') | 
					
						
							|  |  |  |         self.assertIsInstance(h, self.Foo) | 
					
						
							| 
									
										
										
										
											2012-05-27 15:03:38 -04:00
										 |  |  |         self.assertNotIsInstance(h, headerregistry.UnstructuredHeader) | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  |         h = policy2.header_factory('foo', 'test') | 
					
						
							|  |  |  |         self.assertNotIsInstance(h, self.Foo) | 
					
						
							| 
									
										
										
										
											2012-05-27 15:03:38 -04:00
										 |  |  |         self.assertIsInstance(h, headerregistry.UnstructuredHeader) | 
					
						
							| 
									
										
										
										
											2012-05-25 18:42:14 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_clone_copies_factory(self): | 
					
						
							|  |  |  |         policy1 = email.policy.EmailPolicy() | 
					
						
							|  |  |  |         policy2 = policy1.clone() | 
					
						
							|  |  |  |         policy1.header_factory.map_to_type('foo', self.Foo) | 
					
						
							|  |  |  |         h = policy1.header_factory('foo', 'test') | 
					
						
							|  |  |  |         self.assertIsInstance(h, self.Foo) | 
					
						
							|  |  |  |         h = policy2.header_factory('foo', 'test') | 
					
						
							|  |  |  |         self.assertIsInstance(h, self.Foo) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_new_factory_overrides_default(self): | 
					
						
							|  |  |  |         mypolicy = email.policy.EmailPolicy() | 
					
						
							|  |  |  |         myfactory = mypolicy.header_factory | 
					
						
							|  |  |  |         newpolicy = mypolicy + email.policy.strict | 
					
						
							|  |  |  |         self.assertEqual(newpolicy.header_factory, myfactory) | 
					
						
							|  |  |  |         newpolicy = email.policy.strict + mypolicy | 
					
						
							|  |  |  |         self.assertEqual(newpolicy.header_factory, myfactory) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_adding_default_policies_preserves_default_factory(self): | 
					
						
							|  |  |  |         newpolicy = email.policy.default + email.policy.strict | 
					
						
							|  |  |  |         self.assertEqual(newpolicy.header_factory, | 
					
						
							|  |  |  |                          email.policy.EmailPolicy.header_factory) | 
					
						
							|  |  |  |         self.assertEqual(newpolicy.__dict__, {'raise_on_defect': True}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-14 18:55:23 +02:00
										 |  |  |     def test_non_ascii_chars_do_not_cause_inf_loop(self): | 
					
						
							|  |  |  |         policy = email.policy.default.clone(max_line_length=20) | 
					
						
							|  |  |  |         actual = policy.fold('Subject', 'ą' * 12) | 
					
						
							|  |  |  |         self.assertEqual( | 
					
						
							|  |  |  |             actual, | 
					
						
							|  |  |  |             'Subject: \n' + | 
					
						
							|  |  |  |             12 * ' =?utf-8?q?=C4=85?=\n') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 19:50:01 +02:00
										 |  |  |     def test_short_maxlen_error(self): | 
					
						
							|  |  |  |         # RFC 2047 chrome takes up 7 characters, plus the length of the charset | 
					
						
							|  |  |  |         # name, so folding should fail if maxlen is lower than the minimum | 
					
						
							|  |  |  |         # required length for a line. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Note: This is only triggered when there is a single word longer than | 
					
						
							|  |  |  |         # max_line_length, hence the 1234567890 at the end of this whimsical | 
					
						
							|  |  |  |         # subject. This is because when we encounter a word longer than | 
					
						
							|  |  |  |         # max_line_length, it is broken down into encoded words to fit | 
					
						
							|  |  |  |         # max_line_length. If the max_line_length isn't large enough to even | 
					
						
							|  |  |  |         # contain the RFC 2047 chrome (`?=<charset>?q??=`), we fail. | 
					
						
							|  |  |  |         subject = "Melt away the pounds with this one simple trick! 1234567890" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for maxlen in [3, 7, 9]: | 
					
						
							|  |  |  |             with self.subTest(maxlen=maxlen): | 
					
						
							|  |  |  |                 policy = email.policy.default.clone(max_line_length=maxlen) | 
					
						
							|  |  |  |                 with self.assertRaises(email.errors.HeaderParseError): | 
					
						
							|  |  |  |                     policy.fold("Subject", subject) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-31 00:19:48 +02:00
										 |  |  |     def test_verify_generated_headers(self): | 
					
						
							|  |  |  |         """Turning protection off allows header injection""" | 
					
						
							|  |  |  |         policy = email.policy.default.clone(verify_generated_headers=False) | 
					
						
							|  |  |  |         for text in ( | 
					
						
							|  |  |  |             'Header: Value\r\nBad: Injection\r\n', | 
					
						
							|  |  |  |             'Header: NoNewLine' | 
					
						
							|  |  |  |         ): | 
					
						
							|  |  |  |             with self.subTest(text=text): | 
					
						
							|  |  |  |                 message = email.message_from_string( | 
					
						
							|  |  |  |                     "Header: Value\r\n\r\nBody", | 
					
						
							|  |  |  |                     policy=policy, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 class LiteralHeader(str): | 
					
						
							|  |  |  |                     name = 'Header' | 
					
						
							|  |  |  |                     def fold(self, **kwargs): | 
					
						
							|  |  |  |                         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 del message['Header'] | 
					
						
							|  |  |  |                 message['Header'] = LiteralHeader(text) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 self.assertEqual( | 
					
						
							|  |  |  |                     message.as_string(), | 
					
						
							|  |  |  |                     f"{text}\nBody", | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  |     # XXX: Need subclassing tests. | 
					
						
							|  |  |  |     # For adding subclassed objects, make sure the usual rules apply (subclass | 
					
						
							|  |  |  |     # wins), but that the order still works (right overrides left). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  | class TestException(Exception): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  | class TestPolicyPropagation(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # The abstract methods are used by the parser but not by the wrapper | 
					
						
							|  |  |  |     # functions that call it, so if the exception gets raised we know that the | 
					
						
							|  |  |  |     # policy was actually propagated all the way to feedparser. | 
					
						
							|  |  |  |     class MyPolicy(email.policy.Policy): | 
					
						
							|  |  |  |         def badmethod(self, *args, **kw): | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |             raise TestException("test") | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  |         fold = fold_binary = header_fetch_parser = badmethod | 
					
						
							|  |  |  |         header_source_parse = header_store_parse = badmethod | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_message_from_string(self): | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |         with self.assertRaisesRegex(TestException, "^test$"): | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  |             email.message_from_string("Subject: test\n\n", | 
					
						
							|  |  |  |                                       policy=self.MyPolicy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_message_from_bytes(self): | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |         with self.assertRaisesRegex(TestException, "^test$"): | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  |             email.message_from_bytes(b"Subject: test\n\n", | 
					
						
							|  |  |  |                                      policy=self.MyPolicy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_message_from_file(self): | 
					
						
							|  |  |  |         f = io.StringIO('Subject: test\n\n') | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |         with self.assertRaisesRegex(TestException, "^test$"): | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  |             email.message_from_file(f, policy=self.MyPolicy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_message_from_binary_file(self): | 
					
						
							|  |  |  |         f = io.BytesIO(b'Subject: test\n\n') | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |         with self.assertRaisesRegex(TestException, "^test$"): | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  |             email.message_from_binary_file(f, policy=self.MyPolicy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # These are redundant, but we need them for black-box completeness. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_parser(self): | 
					
						
							|  |  |  |         p = email.parser.Parser(policy=self.MyPolicy) | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |         with self.assertRaisesRegex(TestException, "^test$"): | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  |             p.parsestr('Subject: test\n\n') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_bytes_parser(self): | 
					
						
							|  |  |  |         p = email.parser.BytesParser(policy=self.MyPolicy) | 
					
						
							| 
									
										
										
										
											2016-09-09 18:39:18 -04:00
										 |  |  |         with self.assertRaisesRegex(TestException, "^test$"): | 
					
						
							| 
									
										
										
										
											2012-05-25 15:01:48 -04:00
										 |  |  |             p.parsebytes(b'Subject: test\n\n') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Now that we've established that all the parse methods get the | 
					
						
							|  |  |  |     # policy in to feedparser, we can use message_from_string for | 
					
						
							|  |  |  |     # the rest of the propagation tests. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _make_msg(self, source='Subject: test\n\n', policy=None): | 
					
						
							|  |  |  |         self.policy = email.policy.default.clone() if policy is None else policy | 
					
						
							|  |  |  |         return email.message_from_string(source, policy=self.policy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_parser_propagates_policy_to_message(self): | 
					
						
							|  |  |  |         msg = self._make_msg() | 
					
						
							|  |  |  |         self.assertIs(msg.policy, self.policy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_parser_propagates_policy_to_sub_messages(self): | 
					
						
							|  |  |  |         msg = self._make_msg(textwrap.dedent("""\
 | 
					
						
							|  |  |  |             Subject: mime test | 
					
						
							|  |  |  |             MIME-Version: 1.0 | 
					
						
							|  |  |  |             Content-Type: multipart/mixed, boundary="XXX" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             --XXX | 
					
						
							|  |  |  |             Content-Type: text/plain | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             test | 
					
						
							|  |  |  |             --XXX | 
					
						
							|  |  |  |             Content-Type: text/plain | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             test2 | 
					
						
							|  |  |  |             --XXX-- | 
					
						
							|  |  |  |             """))
 | 
					
						
							|  |  |  |         for part in msg.walk(): | 
					
						
							|  |  |  |             self.assertIs(part.policy, self.policy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_message_policy_propagates_to_generator(self): | 
					
						
							|  |  |  |         msg = self._make_msg("Subject: test\nTo: foo\n\n", | 
					
						
							|  |  |  |                              policy=email.policy.default.clone(linesep='X')) | 
					
						
							|  |  |  |         s = io.StringIO() | 
					
						
							|  |  |  |         g = email.generator.Generator(s) | 
					
						
							|  |  |  |         g.flatten(msg) | 
					
						
							|  |  |  |         self.assertEqual(s.getvalue(), "Subject: testXTo: fooXX") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_message_policy_used_by_as_string(self): | 
					
						
							|  |  |  |         msg = self._make_msg("Subject: test\nTo: foo\n\n", | 
					
						
							|  |  |  |                              policy=email.policy.default.clone(linesep='X')) | 
					
						
							|  |  |  |         self.assertEqual(msg.as_string(), "Subject: testXTo: fooXX") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-09 15:29:24 -04:00
										 |  |  | class TestConcretePolicies(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_header_store_parse_rejects_newlines(self): | 
					
						
							|  |  |  |         instance = email.policy.EmailPolicy() | 
					
						
							|  |  |  |         self.assertRaises(ValueError, | 
					
						
							|  |  |  |                           instance.header_store_parse, | 
					
						
							|  |  |  |                           'From', 'spam\negg@foo.py') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 13:59:37 -04:00
										 |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     unittest.main() |