summaryrefslogtreecommitdiffstats
path: root/Lib/email/header.py
diff options
context:
space:
mode:
authorR David Murray <rdmurray@bitdance.com>2011-04-18 14:04:34 (GMT)
committerR David Murray <rdmurray@bitdance.com>2011-04-18 14:04:34 (GMT)
commit01581ee0b7c968adb987a36495af7ce5eb794d0d (patch)
tree4bd6ec577ce781c19418b62463f2f8fd758715dd /Lib/email/header.py
parent74c0031066c365563cd0b4e5ad8da3687ed60656 (diff)
downloadcpython-01581ee0b7c968adb987a36495af7ce5eb794d0d.zip
cpython-01581ee0b7c968adb987a36495af7ce5eb794d0d.tar.gz
cpython-01581ee0b7c968adb987a36495af7ce5eb794d0d.tar.bz2
#11492: rewrite header folding algorithm. Less code, more passing tests.
Diffstat (limited to 'Lib/email/header.py')
-rw-r--r--Lib/email/header.py289
1 files changed, 109 insertions, 180 deletions
diff --git a/Lib/email/header.py b/Lib/email/header.py
index 7e235df..0a66df5 100644
--- a/Lib/email/header.py
+++ b/Lib/email/header.py
@@ -26,6 +26,7 @@ BSPACE = b' '
SPACE8 = ' ' * 8
EMPTYSTRING = ''
MAXLINELEN = 78
+FWS = ' \t'
USASCII = Charset('us-ascii')
UTF8 = Charset('utf-8')
@@ -299,9 +300,15 @@ class Header:
name was specified at Header construction time. The default value for
maxlinelen is determined at header construction time.
- Optional splitchars is a string containing characters to split long
- ASCII lines on, in rough support of RFC 2822's `highest level
- syntactic breaks'. This doesn't affect RFC 2047 encoded lines.
+ Optional splitchars is a string containing characters which should be
+ given extra weight by the splitting algorithm during normal header
+ wrapping. This is in very rough support of RFC 2822's `higher level
+ syntactic breaks': split points preceded by a splitchar are preferred
+ during line splitting, with the characters preferred in the order in
+ which they appear in the string. Space and tab may be included in the
+ string to indicate whether preference should be given to one over the
+ other as a split point when other split chars do not appear in the line
+ being split. Splitchars does not affect RFC 2047 encoded lines.
Optional linesep is a string to be used to separate the lines of
the value. The default value is the most useful for typical
@@ -320,13 +327,19 @@ class Header:
self._continuation_ws, splitchars)
for string, charset in self._chunks:
lines = string.splitlines()
- formatter.feed(lines[0] if lines else '', charset)
+ if lines:
+ formatter.feed('', lines[0], charset)
+ else:
+ formatter.feed('', '', charset)
for line in lines[1:]:
formatter.newline()
if charset.header_encoding is not None:
- formatter.feed(self._continuation_ws, USASCII)
- line = ' ' + line.lstrip()
- formatter.feed(line, charset)
+ formatter.feed(self._continuation_ws, ' ' + line.lstrip(),
+ charset)
+ else:
+ sline = line.lstrip()
+ fws = line[:len(line)-len(sline)]
+ formatter.feed(fws, sline, charset)
if len(lines) > 1:
formatter.newline()
formatter.add_transition()
@@ -360,7 +373,7 @@ class _ValueFormatter:
def __init__(self, headerlen, maxlen, continuation_ws, splitchars):
self._maxlen = maxlen
self._continuation_ws = continuation_ws
- self._continuation_ws_len = len(continuation_ws.replace('\t', SPACE8))
+ self._continuation_ws_len = len(continuation_ws)
self._splitchars = splitchars
self._lines = []
self._current_line = _Accumulator(headerlen)
@@ -374,43 +387,26 @@ class _ValueFormatter:
def newline(self):
end_of_line = self._current_line.pop()
- if end_of_line is not None:
- self._current_line.push(end_of_line)
+ if end_of_line != (' ', ''):
+ self._current_line.push(*end_of_line)
if len(self._current_line) > 0:
- self._lines.append(str(self._current_line))
+ if self._current_line.is_onlyws():
+ self._lines[-1] += str(self._current_line)
+ else:
+ self._lines.append(str(self._current_line))
self._current_line.reset()
def add_transition(self):
- self._current_line.push(None)
-
- def feed(self, string, charset):
- # If the string itself fits on the current line in its encoded format,
- # then add it now and be done with it.
- encoded_string = charset.header_encode(string)
- if len(encoded_string) + len(self._current_line) <= self._maxlen:
- self._current_line.push(encoded_string)
- return
+ self._current_line.push(' ', '')
+
+ def feed(self, fws, string, charset):
# If the charset has no header encoding (i.e. it is an ASCII encoding)
# then we must split the header at the "highest level syntactic break"
# possible. Note that we don't have a lot of smarts about field
# syntax; we just try to break on semi-colons, then commas, then
# whitespace. Eventually, this should be pluggable.
if charset.header_encoding is None:
- for ch in self._splitchars:
- if ch in string:
- break
- else:
- ch = None
- # If there's no available split character then regardless of
- # whether the string fits on the line, we have to put it on a line
- # by itself.
- if ch is None:
- if not self._current_line.is_onlyws():
- self._lines.append(str(self._current_line))
- self._current_line.reset(self._continuation_ws)
- self._current_line.push(encoded_string)
- else:
- self._ascii_split(string, ch)
+ self._ascii_split(fws, string, self._splitchars)
return
# Otherwise, we're doing either a Base64 or a quoted-printable
# encoding which means we don't need to split the line on syntactic
@@ -428,15 +424,14 @@ class _ValueFormatter:
# There are no encoded lines, so we're done.
return
if first_line is not None:
- self._current_line.push(first_line)
- self._lines.append(str(self._current_line))
- self._current_line.reset(self._continuation_ws)
+ self._append_chunk(fws, first_line)
try:
last_line = encoded_lines.pop()
except IndexError:
# There was only one line.
return
- self._current_line.push(last_line)
+ self.newline()
+ self._current_line.push(self._continuation_ws, last_line)
# Everything else are full lines in themselves.
for line in encoded_lines:
self._lines.append(self._continuation_ws + line)
@@ -447,162 +442,96 @@ class _ValueFormatter:
while True:
yield self._maxlen - self._continuation_ws_len
- def _ascii_split(self, string, ch):
- holding = _Accumulator()
- # Split the line on the split character, preserving it. If the split
- # character is whitespace RFC 2822 $2.2.3 requires us to fold on the
- # whitespace, so that the line leads with the original whitespace we
- # split on. However, if a higher syntactic break is used instead
- # (e.g. comma or semicolon), the folding should happen after the split
- # character. But then in that case, we need to add our own
- # continuation whitespace -- although won't that break unfolding?
- for part, splitpart, nextpart in _spliterator(ch, string):
- if not splitpart:
- # No splitpart means this is the last chunk. Put this part
- # either on the current line or the next line depending on
- # whether it fits.
- holding.push(part)
- if len(holding) + len(self._current_line) <= self._maxlen:
- # It fits, but we're done.
- self._current_line.push(str(holding))
- else:
- # It doesn't fit, but we're done. Before pushing a new
- # line, watch out for the current line containing only
- # whitespace.
- holding.pop()
- if self._current_line.is_onlyws() and holding.is_onlyws():
- # Don't start a new line.
- holding.push(part)
- part = None
- self._current_line.push(str(holding))
- self._lines.append(str(self._current_line))
- if part is None:
- self._current_line.reset()
- else:
- holding.reset(part)
- self._current_line.reset(str(holding))
- return
- elif not nextpart:
- # There must be some trailing or duplicated split characters
- # because we
- # found a split character but no next part. In this case we
- # must treat the thing to fit as the part + splitpart because
- # if splitpart is whitespace it's not allowed to be the only
- # thing on the line, and if it's not whitespace we must split
- # after the syntactic break.
- holding_prelen = len(holding)
- holding.push(part + splitpart)
- if len(holding) + len(self._current_line) <= self._maxlen:
- self._current_line.push(str(holding))
- elif holding_prelen == 0:
- # This is the only chunk left so it has to go on the
- # current line.
- self._current_line.push(str(holding))
- else:
- save_part = holding.pop()
- self._current_line.push(str(holding))
- self._lines.append(str(self._current_line))
- holding.reset(save_part)
- self._current_line.reset(str(holding))
- holding.reset()
- elif not part:
- # We're leading with a split character. See if the splitpart
- # and nextpart fits on the current line.
- holding.push(splitpart + nextpart)
- holding_len = len(holding)
- # We know we're not leaving the nextpart on the stack.
- holding.pop()
- if holding_len + len(self._current_line) <= self._maxlen:
- holding.push(splitpart)
+ def _ascii_split(self, fws, string, splitchars):
+ # The RFC 2822 header folding algorithm is simple in principle but
+ # complex in practice. Lines may be folded any place where "folding
+ # white space" appears by inserting a linesep character in front of the
+ # FWS. The complication is that not all spaces or tabs qualify as FWS,
+ # and we are also supposed to prefer to break at "higher level
+ # syntactic breaks". We can't do either of these without intimate
+ # knowledge of the structure of structured headers, which we don't have
+ # here. So the best we can do here is prefer to break at the specified
+ # splitchars, and hope that we don't choose any spaces or tabs that
+ # aren't legal FWS. (This is at least better than the old algorithm,
+ # where we would sometimes *introduce* FWS after a splitchar, or the
+ # algorithm before that, where we would turn all white space runs into
+ # single spaces or tabs.)
+ parts = re.split("(["+FWS+"]+)", fws+string)
+ if parts[0]:
+ parts[:0] = ['']
+ else:
+ parts.pop(0)
+ for fws, part in zip(*[iter(parts)]*2):
+ self._append_chunk(fws, part)
+
+ def _append_chunk(self, fws, string):
+ self._current_line.push(fws, string)
+ if len(self._current_line) > self._maxlen:
+ # Find the best split point, working backward from the end.
+ # There might be none, on a long first line.
+ for ch in self._splitchars:
+ for i in range(self._current_line.part_count()-1, 0, -1):
+ if ch.isspace():
+ fws = self._current_line[i][0]
+ if fws and fws[0]==ch:
+ break
+ prevpart = self._current_line[i-1][1]
+ if prevpart and prevpart[-1]==ch:
+ break
else:
- # It doesn't fit. Since there's no current part really
- # the best we can do is start a new line and push the
- # split part onto it.
- self._current_line.push(str(holding))
- holding.reset()
- if len(self._current_line) > 0 and self._lines:
- self._lines.append(str(self._current_line))
- self._current_line.reset()
- holding.push(splitpart)
+ continue
+ break
else:
- # All three parts are present. First let's see if all three
- # parts will fit on the current line. If so, we don't need to
- # split it.
- holding.push(part + splitpart + nextpart)
- holding_len = len(holding)
- # Pop the part because we'll push nextpart on the next
- # iteration through the loop.
- holding.pop()
- if holding_len + len(self._current_line) <= self._maxlen:
- holding.push(part + splitpart)
- else:
- # The entire thing doesn't fit. See if we need to split
- # before or after the split characters.
- if splitpart.isspace():
- # Split before whitespace. Remember that the
- # whitespace becomes the continuation whitespace of
- # the next line so it goes to current_line not holding.
- holding.push(part)
- self._current_line.push(str(holding))
- holding.reset()
- self._lines.append(str(self._current_line))
- self._current_line.reset(splitpart)
- else:
- # Split after non-whitespace. The continuation
- # whitespace comes from the instance variable.
- holding.push(part + splitpart)
- self._current_line.push(str(holding))
- holding.reset()
- self._lines.append(str(self._current_line))
- if nextpart[0].isspace():
- self._current_line.reset()
- else:
- self._current_line.reset(self._continuation_ws)
- # Get the last of the holding part
- self._current_line.push(str(holding))
+ fws, part = self._current_line.pop()
+ if self._current_line._initial_size > 0:
+ # There will be a header, so leave it on a line by itself.
+ self.newline()
+ if not fws:
+ # We don't use continuation_ws here because the whitespace
+ # after a header should always be a space.
+ fws = ' '
+ self._current_line.push(fws, part)
+ return
+ remainder = self._current_line.pop_from(i)
+ self._lines.append(str(self._current_line))
+ self._current_line.reset(remainder)
-
-def _spliterator(character, string):
- parts = list(reversed(re.split('(%s)' % character, string)))
- while parts:
- part = parts.pop()
- splitparts = (parts.pop() if parts else None)
- nextpart = (parts.pop() if parts else None)
- yield (part, splitparts, nextpart)
- if nextpart is not None:
- parts.append(nextpart)
-
-
-class _Accumulator:
+class _Accumulator(list):
+
def __init__(self, initial_size=0):
self._initial_size = initial_size
- self._current = []
+ super().__init__()
- def push(self, string):
- self._current.append(string)
+ def push(self, fws, string):
+ self.append((fws, string))
+
+ def pop_from(self, i=0):
+ popped = self[i:]
+ self[i:] = []
+ return popped
def pop(self):
- if not self._current:
- return None
- return self._current.pop()
+ if self.part_count()==0:
+ return ('', '')
+ return super().pop()
def __len__(self):
- return sum(((1 if string is None else len(string))
- for string in self._current),
+ return sum((len(fws)+len(part) for fws, part in self),
self._initial_size)
def __str__(self):
- if self._current and self._current[-1] is None:
- self._current.pop()
- return EMPTYSTRING.join((' ' if string is None else string)
- for string in self._current)
+ return EMPTYSTRING.join((EMPTYSTRING.join((fws, part))
+ for fws, part in self))
- def reset(self, string=None):
- self._current = []
+ def reset(self, startval=None):
+ if startval is None:
+ startval = []
+ self[:] = startval
self._initial_size = 0
- if string is not None:
- self.push(string)
def is_onlyws(self):
- return len(self) == 0 or str(self).isspace()
+ return self._initial_size==0 and (not self or str(self).isspace())
+
+ def part_count(self):
+ return super().__len__()