mirror of
https://github.com/python/cpython.git
synced 2026-01-30 03:02:30 +00:00
gh-143935: Email preserve parens when folding comments (#143936)
Fix a bug in the folding of comments when flattening an email message using a modern email policy. Comments consisting of a very long sequence of non-foldable characters could trigger a forced line wrap that omitted the required leading space on the continuation line, causing the remainder of the comment to be interpreted as a new header field. This enabled header injection with carefully crafted inputs. Co-authored-by: Denis Ledoux <dle@odoo.com>
This commit is contained in:
parent
bb2b9ba49d
commit
17d1490aa9
3 changed files with 43 additions and 1 deletions
|
|
@ -101,6 +101,12 @@ def make_quoted_pairs(value):
|
|||
return str(value).replace('\\', '\\\\').replace('"', '\\"')
|
||||
|
||||
|
||||
def make_parenthesis_pairs(value):
|
||||
"""Escape parenthesis and backslash for use within a comment."""
|
||||
return str(value).replace('\\', '\\\\') \
|
||||
.replace('(', '\\(').replace(')', '\\)')
|
||||
|
||||
|
||||
def quote_string(value):
|
||||
escaped = make_quoted_pairs(value)
|
||||
return f'"{escaped}"'
|
||||
|
|
@ -943,7 +949,7 @@ def value(self):
|
|||
return ' '
|
||||
|
||||
def startswith_fws(self):
|
||||
return True
|
||||
return self and self[0] in WSP
|
||||
|
||||
|
||||
class ValueTerminal(Terminal):
|
||||
|
|
@ -2963,6 +2969,13 @@ def _refold_parse_tree(parse_tree, *, policy):
|
|||
[ValueTerminal(make_quoted_pairs(p), 'ptext')
|
||||
for p in newparts] +
|
||||
[ValueTerminal('"', 'ptext')])
|
||||
if part.token_type == 'comment':
|
||||
newparts = (
|
||||
[ValueTerminal('(', 'ptext')] +
|
||||
[ValueTerminal(make_parenthesis_pairs(p), 'ptext')
|
||||
if p.token_type == 'ptext' else p
|
||||
for p in newparts] +
|
||||
[ValueTerminal(')', 'ptext')])
|
||||
if not part.as_ew_allowed:
|
||||
wrap_as_ew_blocked += 1
|
||||
newparts.append(end_ew_not_allowed)
|
||||
|
|
|
|||
|
|
@ -3294,6 +3294,29 @@ def test_address_list_with_specials_in_long_quoted_string(self):
|
|||
with self.subTest(to=to):
|
||||
self._test(parser.get_address_list(to)[0], folded, policy=policy)
|
||||
|
||||
def test_address_list_with_long_unwrapable_comment(self):
|
||||
policy = self.policy.clone(max_line_length=40)
|
||||
cases = [
|
||||
# (to, folded)
|
||||
('(loremipsumdolorsitametconsecteturadipi)<spy@example.org>',
|
||||
'(loremipsumdolorsitametconsecteturadipi)<spy@example.org>\n'),
|
||||
('<spy@example.org>(loremipsumdolorsitametconsecteturadipi)',
|
||||
'<spy@example.org>(loremipsumdolorsitametconsecteturadipi)\n'),
|
||||
('(loremipsum dolorsitametconsecteturadipi)<spy@example.org>',
|
||||
'(loremipsum dolorsitametconsecteturadipi)<spy@example.org>\n'),
|
||||
('<spy@example.org>(loremipsum dolorsitametconsecteturadipi)',
|
||||
'<spy@example.org>(loremipsum\n dolorsitametconsecteturadipi)\n'),
|
||||
('(Escaped \\( \\) chars \\\\ in comments stay escaped)<spy@example.org>',
|
||||
'(Escaped \\( \\) chars \\\\ in comments stay\n escaped)<spy@example.org>\n'),
|
||||
('((loremipsum)(loremipsum)(loremipsum)(loremipsum))<spy@example.org>',
|
||||
'((loremipsum)(loremipsum)(loremipsum)(loremipsum))<spy@example.org>\n'),
|
||||
('((loremipsum)(loremipsum)(loremipsum) (loremipsum))<spy@example.org>',
|
||||
'((loremipsum)(loremipsum)(loremipsum)\n (loremipsum))<spy@example.org>\n'),
|
||||
]
|
||||
for (to, folded) in cases:
|
||||
with self.subTest(to=to):
|
||||
self._test(parser.get_address_list(to)[0], folded, policy=policy)
|
||||
|
||||
# XXX Need tests with comments on various sides of a unicode token,
|
||||
# and with unicode tokens in the comments. Spaces inside the quotes
|
||||
# currently don't do the right thing.
|
||||
|
|
|
|||
|
|
@ -0,0 +1,6 @@
|
|||
Fixed a bug in the folding of comments when flattening an email message
|
||||
using a modern email policy. Comments consisting of a very long sequence of
|
||||
non-foldable characters could trigger a forced line wrap that omitted the
|
||||
required leading space on the continuation line, causing the remainder of
|
||||
the comment to be interpreted as a new header field. This enabled header
|
||||
injection with carefully crafted inputs.
|
||||
Loading…
Add table
Add a link
Reference in a new issue