#12586: add provisional email policy with new header parsing and folding.
When the new policies are used (and only when the new policies are explicitly used) headers turn into objects that have attributes based on their parsed values, and can be set using objects that encapsulate the values, as well as set directly from unicode strings. The folding algorithm then takes care of encoding unicode where needed, and folding according to the highest level syntactic objects. With this patch only date and time headers are parsed as anything other than unstructured, but that is all the helper methods in the existing API handle. I do plan to add more parsers, and complete the set specified in the RFC before the package becomes stable.
This commit is contained in:
parent
0fa2edd08f
commit
0b6f6c82b5
|
@ -306,3 +306,326 @@ added matters. To illustrate::
|
|||
``7bit``, non-ascii binary data is CTE encoded using the ``unknown-8bit``
|
||||
charset. Otherwise the original source header is used, with its existing
|
||||
line breaks and and any (RFC invalid) binary data it may contain.
|
||||
|
||||
|
||||
.. note::
|
||||
|
||||
The remainder of the classes documented below are included in the standard
|
||||
library on a :term:`provisional basis <provisional package>`. Backwards
|
||||
incompatible changes (up to and including removal of the feature) may occur
|
||||
if deemed necessary by the core developers.
|
||||
|
||||
|
||||
.. class:: EmailPolicy(**kw)
|
||||
|
||||
This concrete :class:`Policy` provides behavior that is intended to be fully
|
||||
compliant with the current email RFCs. These include (but are not limited
|
||||
to) :rfc:`5322`, :rfc:`2047`, and the current MIME RFCs.
|
||||
|
||||
This policy adds new header parsing and folding algorithms. Instead of
|
||||
simple strings, headers are custom objects with custom attributes depending
|
||||
on the type of the field. The parsing and folding algorithm fully implement
|
||||
:rfc:`2047` and :rfc:`5322`.
|
||||
|
||||
In addition to the settable attributes listed above that apply to all
|
||||
policies, this policy adds the following additional attributes:
|
||||
|
||||
.. attribute:: refold_source
|
||||
|
||||
If the value for a header in the ``Message`` object originated from a
|
||||
:mod:`~email.parser` (as opposed to being set by a program), this
|
||||
attribute indicates whether or not a generator should refold that value
|
||||
when transforming the message back into stream form. The possible values
|
||||
are:
|
||||
|
||||
======== ===============================================================
|
||||
``none`` all source values use original folding
|
||||
|
||||
``long`` source values that have any line that is longer than
|
||||
``max_line_length`` will be refolded
|
||||
|
||||
``all`` all values are refolded.
|
||||
======== ===============================================================
|
||||
|
||||
The default is ``long``.
|
||||
|
||||
.. attribute:: header_factory
|
||||
|
||||
A callable that takes two arguments, ``name`` and ``value``, where
|
||||
``name`` is a header field name and ``value`` is an unfolded header field
|
||||
value, and returns a string-like object that represents that header. A
|
||||
default ``header_factory`` is provided that understands some of the
|
||||
:RFC:`5322` header field types. (Currently address fields and date
|
||||
fields have special treatment, while all other fields are treated as
|
||||
unstructured. This list will be completed before the extension is marked
|
||||
stable.)
|
||||
|
||||
The class provides the following concrete implementations of the abstract
|
||||
methods of :class:`Policy`:
|
||||
|
||||
.. method:: header_source_parse(sourcelines)
|
||||
|
||||
The implementation of this method is the same as that for the
|
||||
:class:`Compat32` policy.
|
||||
|
||||
.. method:: header_store_parse(name, value)
|
||||
|
||||
The name is returned unchanged. If the input value has a ``name``
|
||||
attribute and it matches *name* ignoring case, the value is returned
|
||||
unchanged. Otherwise the *name* and *value* are passed to
|
||||
``header_factory``, and the resulting custom header object is returned as
|
||||
the value. In this case a ``ValueError`` is raised if the input value
|
||||
contains CR or LF characters.
|
||||
|
||||
.. method:: header_fetch_parse(name, value)
|
||||
|
||||
If the value has a ``name`` attribute, it is returned to unmodified.
|
||||
Otherwise the *name*, and the *value* with any CR or LF characters
|
||||
removed, are passed to the ``header_factory``, and the resulting custom
|
||||
header object is returned. Any surrogateescaped bytes get turned into
|
||||
the unicode unknown-character glyph.
|
||||
|
||||
.. method:: fold(name, value)
|
||||
|
||||
Header folding is controlled by the :attr:`refold_source` policy setting.
|
||||
A value is considered to be a 'source value' if and only if it does not
|
||||
have a ``name`` attribute (having a ``name`` attribute means it is a
|
||||
header object of some sort). If a source value needs to be refolded
|
||||
according to the policy, it is converted into a custom header object by
|
||||
passing the *name* and the *value* with any CR and LF characters removed
|
||||
to the ``header_factory``. Folding of a custom header object is done by
|
||||
calling its ``fold`` method with the current policy.
|
||||
|
||||
Source values are split into lines using :meth:`~str.splitlines`. If
|
||||
the value is not to be refolded, the lines are rejoined using the
|
||||
``linesep`` from the policy and returned. The exception is lines
|
||||
containing non-ascii binary data. In that case the value is refolded
|
||||
regardless of the ``refold_source`` setting, which causes the binary data
|
||||
to be CTE encoded using the ``unknown-8bit`` charset.
|
||||
|
||||
.. method:: fold_binary(name, value)
|
||||
|
||||
The same as :meth:`fold` if :attr:`cte_type` is ``7bit``, except that
|
||||
the returned value is bytes.
|
||||
|
||||
If :attr:`cte_type` is ``8bit``, non-ASCII binary data is converted back
|
||||
into bytes. Headers with binary data are not refolded, regardless of the
|
||||
``refold_header`` setting, since there is no way to know whether the
|
||||
binary data consists of single byte characters or multibyte characters.
|
||||
|
||||
The following instances of :class:`EmailPolicy` provide defaults suitable for
|
||||
specific application domains. Note that in the future the behavior of these
|
||||
instances (in particular the ``HTTP` instance) may be adjusted to conform even
|
||||
more closely to the RFCs relevant to their domains.
|
||||
|
||||
.. data:: default
|
||||
|
||||
An instance of ``EmailPolicy`` with all defaults unchanged. This policy
|
||||
uses the standard Python ``\n`` line endings rather than the RFC-correct
|
||||
``\r\n``.
|
||||
|
||||
.. data:: SMTP
|
||||
|
||||
Suitable for serializing messages in conformance with the email RFCs.
|
||||
Like ``default``, but with ``linesep`` set to ``\r\n``, which is RFC
|
||||
compliant.
|
||||
|
||||
.. data:: HTTP
|
||||
|
||||
Suitable for serializing headers with for use in HTTP traffic. Like
|
||||
``SMTP`` except that ``max_line_length`` is set to ``None`` (unlimited).
|
||||
|
||||
.. data:: strict
|
||||
|
||||
Convenience instance. The same as ``default`` except that
|
||||
``raise_on_defect`` is set to ``True``. This allows any policy to be made
|
||||
strict by writing::
|
||||
|
||||
somepolicy + policy.strict
|
||||
|
||||
With all of these :class:`EmailPolicies <.EmailPolicy>`, the effective API of
|
||||
the email package is changed from the Python 3.2 API in the following ways:
|
||||
|
||||
* Setting a header on a :class:`~email.message.Message` results in that
|
||||
header being parsed and a custom header object created.
|
||||
|
||||
* Fetching a header value from a :class:`~email.message.Message` results
|
||||
in that header being parsed and a custom header object created and
|
||||
returned.
|
||||
|
||||
* Any custom header object, or any header that is refolded due to the
|
||||
policy settings, is folded using an algorithm that fully implements the
|
||||
RFC folding algorithms, including knowing where encoded words are required
|
||||
and allowed.
|
||||
|
||||
From the application view, this means that any header obtained through the
|
||||
:class:`~email.message.Message` is a custom header object with custom
|
||||
attributes, whose string value is the fully decoded unicode value of the
|
||||
header. Likewise, a header may be assigned a new value, or a new header
|
||||
created, using a unicode string, and the policy will take care of converting
|
||||
the unicode string into the correct RFC encoded form.
|
||||
|
||||
The custom header objects and their attributes are described below. All custom
|
||||
header objects are string subclasses, and their string value is the fully
|
||||
decoded value of the header field (the part of the field after the ``:``)
|
||||
|
||||
|
||||
.. class:: BaseHeader
|
||||
|
||||
This is the base class for all custom header objects. It provides the
|
||||
following attributes:
|
||||
|
||||
.. attribute:: name
|
||||
|
||||
The header field name (the portion of the field before the ':').
|
||||
|
||||
.. attribute:: defects
|
||||
|
||||
A possibly empty list of :class:`~email.errors.MessageDefect` objects
|
||||
that record any RFC violations found while parsing the header field.
|
||||
|
||||
.. method:: fold(*, policy)
|
||||
|
||||
Return a string containing :attr:`~email.policy.Policy.linesep`
|
||||
characters as required to correctly fold the header according
|
||||
to *policy*. A :attr:`~email.policy.Policy.cte_type` of
|
||||
``8bit`` will be treated as if it were ``7bit``, since strings
|
||||
may not contain binary data.
|
||||
|
||||
|
||||
.. class:: UnstructuredHeader
|
||||
|
||||
The class used for any header that does not have a more specific
|
||||
type. (The :mailheader:`Subject` header is an example of an
|
||||
unstructured header.) It does not have any additional attributes.
|
||||
|
||||
|
||||
.. class:: DateHeader
|
||||
|
||||
The value of this type of header is a single date and time value. The
|
||||
primary example of this type of header is the :mailheader:`Date` header.
|
||||
|
||||
.. attribute:: datetime
|
||||
|
||||
A :class:`~datetime.datetime` encoding the date and time from the
|
||||
header value.
|
||||
|
||||
The ``datetime`` will be a naive ``datetime`` if the value either does
|
||||
not have a specified timezone (which would be a violation of the RFC) or
|
||||
if the timezone is specified as ``-0000``. This timezone value indicates
|
||||
that the date and time is to be considered to be in UTC, but with no
|
||||
indication of the local timezone in which it was generated. (This
|
||||
contrasts to ``+0000``, which indicates a date and time that really is in
|
||||
the UTC ``0000`` timezone.)
|
||||
|
||||
If the header value contains a valid timezone that is not ``-0000``, the
|
||||
``datetime`` will be an aware ``datetime`` having a
|
||||
:class:`~datetime.tzinfo` set to the :class:`~datetime.timezone`
|
||||
indicated by the header value.
|
||||
|
||||
A ``datetime`` may also be assigned to a :mailheader:`Date` type header.
|
||||
The resulting string value will use a timezone of ``-0000`` if the
|
||||
``datetime`` is naive, and the appropriate UTC offset if the ``datetime`` is
|
||||
aware.
|
||||
|
||||
|
||||
.. class:: AddressHeader
|
||||
|
||||
This class is used for all headers that can contain addresses, whether they
|
||||
are supposed to be singleton addresses or a list.
|
||||
|
||||
.. attribute:: addresses
|
||||
|
||||
A list of :class:`.Address` objects listing all of the addresses that
|
||||
could be parsed out of the field value.
|
||||
|
||||
.. attribute:: groups
|
||||
|
||||
A list of :class:`.Group` objects. Every address in :attr:`.addresses`
|
||||
appears in one of the group objects in the tuple. Addresses that are not
|
||||
syntactically part of a group are represented by ``Group`` objects whose
|
||||
``name`` is ``None``.
|
||||
|
||||
In addition to addresses in string form, any combination of
|
||||
:class:`.Address` and :class:`.Group` objects, singly or in a list, may be
|
||||
assigned to an address header.
|
||||
|
||||
|
||||
.. class:: Address(display_name='', username='', domain='', addr_spec=None):
|
||||
|
||||
The class used to represent an email address. The general form of an
|
||||
address is::
|
||||
|
||||
[display_name] <username@domain>
|
||||
|
||||
or::
|
||||
|
||||
username@domain
|
||||
|
||||
where each part must conform to specific syntax rules spelled out in
|
||||
:rfc:`5322`.
|
||||
|
||||
As a convenience *addr_spec* can be specified instead of *username* and
|
||||
*domain*, in which case *username* and *domain* will be parsed from the
|
||||
*addr_spec*. An *addr_spec* must be a properly RFC quoted string; if it is
|
||||
not ``Address`` will raise an error. Unicode characters are allowed and
|
||||
will be property encoded when serialized. However, per the RFCs, unicode is
|
||||
*not* allowed in the username portion of the address.
|
||||
|
||||
.. attribute:: display_name
|
||||
|
||||
The display name portion of the address, if any, with all quoting
|
||||
removed. If the address does not have a display name, this attribute
|
||||
will be an empty string.
|
||||
|
||||
.. attribute:: username
|
||||
|
||||
The ``username`` portion of the address, with all quoting removed.
|
||||
|
||||
.. attribute:: domain
|
||||
|
||||
The ``domain`` portion of the address.
|
||||
|
||||
.. attribute:: addr_spec
|
||||
|
||||
The ``username@domain`` portion of the address, correctly quoted
|
||||
for use as a bare address (the second form shown above). This
|
||||
attribute is not mutable.
|
||||
|
||||
.. method:: __str__()
|
||||
|
||||
The ``str`` value of the object is the address quoted according to
|
||||
:rfc:`5322` rules, but with no Content Transfer Encoding of any non-ASCII
|
||||
characters.
|
||||
|
||||
|
||||
.. class:: Group(display_name=None, addresses=None)
|
||||
|
||||
The class used to represent an address group. The general form of an
|
||||
address group is::
|
||||
|
||||
display_name: [address-list];
|
||||
|
||||
As a convenience for processing lists of addresses that consist of a mixture
|
||||
of groups and single addresses, a ``Group`` may also be used to represent
|
||||
single addresses that are not part of a group by setting *display_name* to
|
||||
``None`` and providing a list of the single address as *addresses*.
|
||||
|
||||
.. attribute:: display_name
|
||||
|
||||
The ``display_name`` of the group. If it is ``None`` and there is
|
||||
exactly one ``Address`` in ``addresses``, then the ``Group`` represents a
|
||||
single address that is not in a group.
|
||||
|
||||
.. attribute:: addresses
|
||||
|
||||
A possibly empty tuple of :class:`.Address` objects representing the
|
||||
addresses in the group.
|
||||
|
||||
.. method:: __str__()
|
||||
|
||||
The ``str`` value of a ``Group`` is formatted according to :rfc:`5322`,
|
||||
but with no Content Transfer Encoding of any non-ASCII characters. If
|
||||
``display_name`` is none and there is a single ``Address`` in the
|
||||
``addresses` list, the ``str`` value will be the same as the ``str`` of
|
||||
that single ``Address``.
|
||||
|
|
|
@ -0,0 +1,211 @@
|
|||
""" Routines for manipulating RFC2047 encoded words.
|
||||
|
||||
This is currently a package-private API, but will be considered for promotion
|
||||
to a public API if there is demand.
|
||||
|
||||
"""
|
||||
|
||||
# An ecoded word looks like this:
|
||||
#
|
||||
# =?charset[*lang]?cte?encoded_string?=
|
||||
#
|
||||
# for more information about charset see the charset module. Here it is one
|
||||
# of the preferred MIME charset names (hopefully; you never know when parsing).
|
||||
# cte (Content Transfer Encoding) is either 'q' or 'b' (ignoring case). In
|
||||
# theory other letters could be used for other encodings, but in practice this
|
||||
# (almost?) never happens. There could be a public API for adding entries
|
||||
# to to the CTE tables, but YAGNI for now. 'q' is Quoted Printable, 'b' is
|
||||
# Base64. The meaning of encoded_string should be obvious. 'lang' is optional
|
||||
# as indicated by the brackets (they are not part of the syntax) but is almost
|
||||
# never encountered in practice.
|
||||
#
|
||||
# The general interface for a CTE decoder is that it takes the encoded_string
|
||||
# as its argument, and returns a tuple (cte_decoded_string, defects). The
|
||||
# cte_decoded_string is the original binary that was encoded using the
|
||||
# specified cte. 'defects' is a list of MessageDefect instances indicating any
|
||||
# problems encountered during conversion. 'charset' and 'lang' are the
|
||||
# corresponding strings extracted from the EW, case preserved.
|
||||
#
|
||||
# The general interface for a CTE encoder is that it takes a binary sequence
|
||||
# as input and returns the cte_encoded_string, which is an ascii-only string.
|
||||
#
|
||||
# Each decoder must also supply a length function that takes the binary
|
||||
# sequence as its argument and returns the length of the resulting encoded
|
||||
# string.
|
||||
#
|
||||
# The main API functions for the module are decode, which calls the decoder
|
||||
# referenced by the cte specifier, and encode, which adds the appropriate
|
||||
# RFC 2047 "chrome" to the encoded string, and can optionally automatically
|
||||
# select the shortest possible encoding. See their docstrings below for
|
||||
# details.
|
||||
|
||||
import re
|
||||
import base64
|
||||
import binascii
|
||||
import functools
|
||||
from string import ascii_letters, digits
|
||||
from email import errors
|
||||
|
||||
#
|
||||
# Quoted Printable
|
||||
#
|
||||
|
||||
# regex based decoder.
|
||||
_q_byte_subber = functools.partial(re.compile(br'=([a-fA-F0-9]{2})').sub,
|
||||
lambda m: bytes([int(m.group(1), 16)]))
|
||||
|
||||
def decode_q(encoded):
|
||||
encoded = encoded.replace(b'_', b' ')
|
||||
return _q_byte_subber(encoded), []
|
||||
|
||||
|
||||
# dict mapping bytes to their encoded form
|
||||
class QByteMap(dict):
|
||||
|
||||
safe = b'-!*+/' + ascii_letters.encode('ascii') + digits.encode('ascii')
|
||||
|
||||
def __missing__(self, key):
|
||||
if key in self.safe:
|
||||
self[key] = chr(key)
|
||||
else:
|
||||
self[key] = "={:02X}".format(key)
|
||||
return self[key]
|
||||
|
||||
_q_byte_map = QByteMap()
|
||||
|
||||
# In headers spaces are mapped to '_'.
|
||||
_q_byte_map[ord(' ')] = '_'
|
||||
|
||||
def encode_q(bstring):
|
||||
return ''.join(_q_byte_map[x] for x in bstring)
|
||||
|
||||
def len_q(bstring):
|
||||
return sum(len(_q_byte_map[x]) for x in bstring)
|
||||
|
||||
|
||||
#
|
||||
# Base64
|
||||
#
|
||||
|
||||
def decode_b(encoded):
|
||||
defects = []
|
||||
pad_err = len(encoded) % 4
|
||||
if pad_err:
|
||||
defects.append(errors.InvalidBase64PaddingDefect())
|
||||
padded_encoded = encoded + b'==='[:4-pad_err]
|
||||
else:
|
||||
padded_encoded = encoded
|
||||
try:
|
||||
return base64.b64decode(padded_encoded, validate=True), defects
|
||||
except binascii.Error:
|
||||
# Since we had correct padding, this must an invalid char error.
|
||||
defects = [errors.InvalidBase64CharactersDefect()]
|
||||
# The non-alphabet characters are ignored as far as padding
|
||||
# goes, but we don't know how many there are. So we'll just
|
||||
# try various padding lengths until something works.
|
||||
for i in 0, 1, 2, 3:
|
||||
try:
|
||||
return base64.b64decode(encoded+b'='*i, validate=False), defects
|
||||
except binascii.Error:
|
||||
if i==0:
|
||||
defects.append(errors.InvalidBase64PaddingDefect())
|
||||
else:
|
||||
# This should never happen.
|
||||
raise AssertionError("unexpected binascii.Error")
|
||||
|
||||
def encode_b(bstring):
|
||||
return base64.b64encode(bstring).decode('ascii')
|
||||
|
||||
def len_b(bstring):
|
||||
groups_of_3, leftover = divmod(len(bstring), 3)
|
||||
# 4 bytes out for each 3 bytes (or nonzero fraction thereof) in.
|
||||
return groups_of_3 * 4 + (4 if leftover else 0)
|
||||
|
||||
|
||||
_cte_decoders = {
|
||||
'q': decode_q,
|
||||
'b': decode_b,
|
||||
}
|
||||
|
||||
def decode(ew):
|
||||
"""Decode encoded word and return (string, charset, lang, defects) tuple.
|
||||
|
||||
An RFC 2047/2243 encoded word has the form:
|
||||
|
||||
=?charset*lang?cte?encoded_string?=
|
||||
|
||||
where '*lang' may be omitted but the other parts may not be.
|
||||
|
||||
This function expects exactly such a string (that is, it does not check the
|
||||
syntax and may raise errors if the string is not well formed), and returns
|
||||
the encoded_string decoded first from its Content Transfer Encoding and
|
||||
then from the resulting bytes into unicode using the specified charset. If
|
||||
the cte-decoded string does not successfully decode using the specified
|
||||
character set, a defect is added to the defects list and the unknown octets
|
||||
are replaced by the unicode 'unknown' character \uFDFF.
|
||||
|
||||
The specified charset and language are returned. The default for language,
|
||||
which is rarely if ever encountered, is the empty string.
|
||||
|
||||
"""
|
||||
_, charset, cte, cte_string, _ = ew.split('?')
|
||||
charset, _, lang = charset.partition('*')
|
||||
cte = cte.lower()
|
||||
# Recover the original bytes and do CTE decoding.
|
||||
bstring = cte_string.encode('ascii', 'surrogateescape')
|
||||
bstring, defects = _cte_decoders[cte](bstring)
|
||||
# Turn the CTE decoded bytes into unicode.
|
||||
try:
|
||||
string = bstring.decode(charset)
|
||||
except UnicodeError:
|
||||
defects.append(errors.UndecodableBytesDefect("Encoded word "
|
||||
"contains bytes not decodable using {} charset".format(charset)))
|
||||
string = bstring.decode(charset, 'surrogateescape')
|
||||
except LookupError:
|
||||
string = bstring.decode('ascii', 'surrogateescape')
|
||||
if charset.lower() != 'unknown-8bit':
|
||||
defects.append(errors.CharsetError("Unknown charset {} "
|
||||
"in encoded word; decoded as unknown bytes".format(charset)))
|
||||
return string, charset, lang, defects
|
||||
|
||||
|
||||
_cte_encoders = {
|
||||
'q': encode_q,
|
||||
'b': encode_b,
|
||||
}
|
||||
|
||||
_cte_encode_length = {
|
||||
'q': len_q,
|
||||
'b': len_b,
|
||||
}
|
||||
|
||||
def encode(string, charset='utf-8', encoding=None, lang=''):
|
||||
"""Encode string using the CTE encoding that produces the shorter result.
|
||||
|
||||
Produces an RFC 2047/2243 encoded word of the form:
|
||||
|
||||
=?charset*lang?cte?encoded_string?=
|
||||
|
||||
where '*lang' is omitted unless the 'lang' parameter is given a value.
|
||||
Optional argument charset (defaults to utf-8) specifies the charset to use
|
||||
to encode the string to binary before CTE encoding it. Optional argument
|
||||
'encoding' is the cte specifier for the encoding that should be used ('q'
|
||||
or 'b'); if it is None (the default) the encoding which produces the
|
||||
shortest encoded sequence is used, except that 'q' is preferred if it is up
|
||||
to five characters longer. Optional argument 'lang' (default '') gives the
|
||||
RFC 2243 language string to specify in the encoded word.
|
||||
|
||||
"""
|
||||
if charset == 'unknown-8bit':
|
||||
bstring = string.encode('ascii', 'surrogateescape')
|
||||
else:
|
||||
bstring = string.encode(charset)
|
||||
if encoding is None:
|
||||
qlen = _cte_encode_length['q'](bstring)
|
||||
blen = _cte_encode_length['b'](bstring)
|
||||
# Bias toward q. 5 is arbitrary.
|
||||
encoding = 'q' if qlen - blen < 5 else 'b'
|
||||
encoded = _cte_encoders[encoding](bstring)
|
||||
if lang:
|
||||
lang = '*' + lang
|
||||
return "=?{}{}?{}?{}?=".format(charset, lang, encoding, encoded)
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,456 @@
|
|||
"""Representing and manipulating email headers via custom objects.
|
||||
|
||||
This module provides an implementation of the HeaderRegistry API.
|
||||
The implementation is designed to flexibly follow RFC5322 rules.
|
||||
|
||||
Eventually HeaderRegistry will be a public API, but it isn't yet,
|
||||
and will probably change some before that happens.
|
||||
|
||||
"""
|
||||
|
||||
from email import utils
|
||||
from email import errors
|
||||
from email import _header_value_parser as parser
|
||||
|
||||
class Address:
|
||||
|
||||
def __init__(self, display_name='', username='', domain='', addr_spec=None):
|
||||
"""Create an object represeting a full email address.
|
||||
|
||||
An address can have a 'display_name', a 'username', and a 'domain'. In
|
||||
addition to specifying the username and domain separately, they may be
|
||||
specified together by using the addr_spec keyword *instead of* the
|
||||
username and domain keywords. If an addr_spec string is specified it
|
||||
must be properly quoted according to RFC 5322 rules; an error will be
|
||||
raised if it is not.
|
||||
|
||||
An Address object has display_name, username, domain, and addr_spec
|
||||
attributes, all of which are read-only. The addr_spec and the string
|
||||
value of the object are both quoted according to RFC5322 rules, but
|
||||
without any Content Transfer Encoding.
|
||||
|
||||
"""
|
||||
# This clause with its potential 'raise' may only happen when an
|
||||
# application program creates an Address object using an addr_spec
|
||||
# keyword. The email library code itself must always supply username
|
||||
# and domain.
|
||||
if addr_spec is not None:
|
||||
if username or domain:
|
||||
raise TypeError("addrspec specified when username and/or "
|
||||
"domain also specified")
|
||||
a_s, rest = parser.get_addr_spec(addr_spec)
|
||||
if rest:
|
||||
raise ValueError("Invalid addr_spec; only '{}' "
|
||||
"could be parsed from '{}'".format(
|
||||
a_s, addr_spec))
|
||||
if a_s.all_defects:
|
||||
raise a_s.all_defects[0]
|
||||
username = a_s.local_part
|
||||
domain = a_s.domain
|
||||
self._display_name = display_name
|
||||
self._username = username
|
||||
self._domain = domain
|
||||
|
||||
@property
|
||||
def display_name(self):
|
||||
return self._display_name
|
||||
|
||||
@property
|
||||
def username(self):
|
||||
return self._username
|
||||
|
||||
@property
|
||||
def domain(self):
|
||||
return self._domain
|
||||
|
||||
@property
|
||||
def addr_spec(self):
|
||||
"""The addr_spec (username@domain) portion of the address, quoted
|
||||
according to RFC 5322 rules, but with no Content Transfer Encoding.
|
||||
"""
|
||||
nameset = set(self.username)
|
||||
if len(nameset) > len(nameset-parser.DOT_ATOM_ENDS):
|
||||
lp = parser.quote_string(self.username)
|
||||
else:
|
||||
lp = self.username
|
||||
if self.domain:
|
||||
return lp + '@' + self.domain
|
||||
if not lp:
|
||||
return '<>'
|
||||
return lp
|
||||
|
||||
def __repr__(self):
|
||||
return "Address(display_name={!r}, username={!r}, domain={!r})".format(
|
||||
self.display_name, self.username, self.domain)
|
||||
|
||||
def __str__(self):
|
||||
nameset = set(self.display_name)
|
||||
if len(nameset) > len(nameset-parser.SPECIALS):
|
||||
disp = parser.quote_string(self.display_name)
|
||||
else:
|
||||
disp = self.display_name
|
||||
if disp:
|
||||
addr_spec = '' if self.addr_spec=='<>' else self.addr_spec
|
||||
return "{} <{}>".format(disp, addr_spec)
|
||||
return self.addr_spec
|
||||
|
||||
def __eq__(self, other):
|
||||
if type(other) != type(self):
|
||||
return False
|
||||
return (self.display_name == other.display_name and
|
||||
self.username == other.username and
|
||||
self.domain == other.domain)
|
||||
|
||||
|
||||
class Group:
|
||||
|
||||
def __init__(self, display_name=None, addresses=None):
|
||||
"""Create an object representing an address group.
|
||||
|
||||
An address group consists of a display_name followed by colon and an
|
||||
list of addresses (see Address) terminated by a semi-colon. The Group
|
||||
is created by specifying a display_name and a possibly empty list of
|
||||
Address objects. A Group can also be used to represent a single
|
||||
address that is not in a group, which is convenient when manipulating
|
||||
lists that are a combination of Groups and individual Addresses. In
|
||||
this case the display_name should be set to None. In particular, the
|
||||
string representation of a Group whose display_name is None is the same
|
||||
as the Address object, if there is one and only one Address object in
|
||||
the addresses list.
|
||||
|
||||
"""
|
||||
self._display_name = display_name
|
||||
self._addresses = tuple(addresses) if addresses else tuple()
|
||||
|
||||
@property
|
||||
def display_name(self):
|
||||
return self._display_name
|
||||
|
||||
@property
|
||||
def addresses(self):
|
||||
return self._addresses
|
||||
|
||||
def __repr__(self):
|
||||
return "Group(display_name={!r}, addresses={!r}".format(
|
||||
self.display_name, self.addresses)
|
||||
|
||||
def __str__(self):
|
||||
if self.display_name is None and len(self.addresses)==1:
|
||||
return str(self.addresses[0])
|
||||
disp = self.display_name
|
||||
if disp is not None:
|
||||
nameset = set(disp)
|
||||
if len(nameset) > len(nameset-parser.SPECIALS):
|
||||
disp = parser.quote_string(disp)
|
||||
adrstr = ", ".join(str(x) for x in self.addresses)
|
||||
adrstr = ' ' + adrstr if adrstr else adrstr
|
||||
return "{}:{};".format(disp, adrstr)
|
||||
|
||||
def __eq__(self, other):
|
||||
if type(other) != type(self):
|
||||
return False
|
||||
return (self.display_name == other.display_name and
|
||||
self.addresses == other.addresses)
|
||||
|
||||
|
||||
# Header Classes #
|
||||
|
||||
class BaseHeader(str):
|
||||
|
||||
"""Base class for message headers.
|
||||
|
||||
Implements generic behavior and provides tools for subclasses.
|
||||
|
||||
A subclass must define a classmethod named 'parse' that takes an unfolded
|
||||
value string and a dictionary as its arguments. The dictionary will
|
||||
contain one key, 'defects', initialized to an empty list. After the call
|
||||
the dictionary must contain two additional keys: parse_tree, set to the
|
||||
parse tree obtained from parsing the header, and 'decoded', set to the
|
||||
string value of the idealized representation of the data from the value.
|
||||
(That is, encoded words are decoded, and values that have canonical
|
||||
representations are so represented.)
|
||||
|
||||
The defects key is intended to collect parsing defects, which the message
|
||||
parser will subsequently dispose of as appropriate. The parser should not,
|
||||
insofar as practical, raise any errors. Defects should be added to the
|
||||
list instead. The standard header parsers register defects for RFC
|
||||
compliance issues, for obsolete RFC syntax, and for unrecoverable parsing
|
||||
errors.
|
||||
|
||||
The parse method may add additional keys to the dictionary. In this case
|
||||
the subclass must define an 'init' method, which will be passed the
|
||||
dictionary as its keyword arguments. The method should use (usually by
|
||||
setting them as the value of similarly named attributes) and remove all the
|
||||
extra keys added by its parse method, and then use super to call its parent
|
||||
class with the remaining arguments and keywords.
|
||||
|
||||
The subclass should also make sure that a 'max_count' attribute is defined
|
||||
that is either None or 1. XXX: need to better define this API.
|
||||
|
||||
"""
|
||||
|
||||
def __new__(cls, name, value):
|
||||
kwds = {'defects': []}
|
||||
cls.parse(value, kwds)
|
||||
if utils._has_surrogates(kwds['decoded']):
|
||||
kwds['decoded'] = utils._sanitize(kwds['decoded'])
|
||||
self = str.__new__(cls, kwds['decoded'])
|
||||
del kwds['decoded']
|
||||
self.init(name, **kwds)
|
||||
return self
|
||||
|
||||
def init(self, name, *, parse_tree, defects):
|
||||
self._name = name
|
||||
self._parse_tree = parse_tree
|
||||
self._defects = defects
|
||||
|
||||
@property
|
||||
def name(self):
|
||||
return self._name
|
||||
|
||||
@property
|
||||
def defects(self):
|
||||
return tuple(self._defects)
|
||||
|
||||
def __reduce__(self):
|
||||
return (
|
||||
_reconstruct_header,
|
||||
(
|
||||
self.__class__.__name__,
|
||||
self.__class__.__bases__,
|
||||
str(self),
|
||||
),
|
||||
self.__dict__)
|
||||
|
||||
@classmethod
|
||||
def _reconstruct(cls, value):
|
||||
return str.__new__(cls, value)
|
||||
|
||||
def fold(self, *, policy):
|
||||
"""Fold header according to policy.
|
||||
|
||||
The parsed representation of the header is folded according to
|
||||
RFC5322 rules, as modified by the policy. If the parse tree
|
||||
contains surrogateescaped bytes, the bytes are CTE encoded using
|
||||
the charset 'unknown-8bit".
|
||||
|
||||
Any non-ASCII characters in the parse tree are CTE encoded using
|
||||
charset utf-8. XXX: make this a policy setting.
|
||||
|
||||
The returned value is an ASCII-only string possibly containing linesep
|
||||
characters, and ending with a linesep character. The string includes
|
||||
the header name and the ': ' separator.
|
||||
|
||||
"""
|
||||
# At some point we need to only put fws here if it was in the source.
|
||||
header = parser.Header([
|
||||
parser.HeaderLabel([
|
||||
parser.ValueTerminal(self.name, 'header-name'),
|
||||
parser.ValueTerminal(':', 'header-sep')]),
|
||||
parser.CFWSList([parser.WhiteSpaceTerminal(' ', 'fws')]),
|
||||
self._parse_tree])
|
||||
return header.fold(policy=policy)
|
||||
|
||||
|
||||
def _reconstruct_header(cls_name, bases, value):
|
||||
return type(cls_name, bases, {})._reconstruct(value)
|
||||
|
||||
|
||||
class UnstructuredHeader:
|
||||
|
||||
max_count = None
|
||||
value_parser = staticmethod(parser.get_unstructured)
|
||||
|
||||
@classmethod
|
||||
def parse(cls, value, kwds):
|
||||
kwds['parse_tree'] = cls.value_parser(value)
|
||||
kwds['decoded'] = str(kwds['parse_tree'])
|
||||
|
||||
|
||||
class UniqueUnstructuredHeader(UnstructuredHeader):
|
||||
|
||||
max_count = 1
|
||||
|
||||
|
||||
class DateHeader:
|
||||
|
||||
"""Header whose value consists of a single timestamp.
|
||||
|
||||
Provides an additional attribute, datetime, which is either an aware
|
||||
datetime using a timezone, or a naive datetime if the timezone
|
||||
in the input string is -0000. Also accepts a datetime as input.
|
||||
The 'value' attribute is the normalized form of the timestamp,
|
||||
which means it is the output of format_datetime on the datetime.
|
||||
"""
|
||||
|
||||
max_count = None
|
||||
|
||||
# This is used only for folding, not for creating 'decoded'.
|
||||
value_parser = staticmethod(parser.get_unstructured)
|
||||
|
||||
@classmethod
|
||||
def parse(cls, value, kwds):
|
||||
if not value:
|
||||
kwds['defects'].append(errors.HeaderMissingRequiredValue())
|
||||
kwds['datetime'] = None
|
||||
kwds['decoded'] = ''
|
||||
kwds['parse_tree'] = parser.TokenList()
|
||||
return
|
||||
if isinstance(value, str):
|
||||
value = utils.parsedate_to_datetime(value)
|
||||
kwds['datetime'] = value
|
||||
kwds['decoded'] = utils.format_datetime(kwds['datetime'])
|
||||
kwds['parse_tree'] = cls.value_parser(kwds['decoded'])
|
||||
|
||||
def init(self, *args, **kw):
|
||||
self._datetime = kw.pop('datetime')
|
||||
super().init(*args, **kw)
|
||||
|
||||
@property
|
||||
def datetime(self):
|
||||
return self._datetime
|
||||
|
||||
|
||||
class UniqueDateHeader(DateHeader):
|
||||
|
||||
max_count = 1
|
||||
|
||||
|
||||
class AddressHeader:
|
||||
|
||||
max_count = None
|
||||
|
||||
@staticmethod
|
||||
def value_parser(value):
|
||||
address_list, value = parser.get_address_list(value)
|
||||
assert not value, 'this should not happen'
|
||||
return address_list
|
||||
|
||||
@classmethod
|
||||
def parse(cls, value, kwds):
|
||||
if isinstance(value, str):
|
||||
# We are translating here from the RFC language (address/mailbox)
|
||||
# to our API language (group/address).
|
||||
kwds['parse_tree'] = address_list = cls.value_parser(value)
|
||||
groups = []
|
||||
for addr in address_list.addresses:
|
||||
groups.append(Group(addr.display_name,
|
||||
[Address(mb.display_name or '',
|
||||
mb.local_part or '',
|
||||
mb.domain or '')
|
||||
for mb in addr.all_mailboxes]))
|
||||
defects = list(address_list.all_defects)
|
||||
else:
|
||||
# Assume it is Address/Group stuff
|
||||
if not hasattr(value, '__iter__'):
|
||||
value = [value]
|
||||
groups = [Group(None, [item]) if not hasattr(item, 'addresses')
|
||||
else item
|
||||
for item in value]
|
||||
defects = []
|
||||
kwds['groups'] = groups
|
||||
kwds['defects'] = defects
|
||||
kwds['decoded'] = ', '.join([str(item) for item in groups])
|
||||
if 'parse_tree' not in kwds:
|
||||
kwds['parse_tree'] = cls.value_parser(kwds['decoded'])
|
||||
|
||||
def init(self, *args, **kw):
|
||||
self._groups = tuple(kw.pop('groups'))
|
||||
self._addresses = None
|
||||
super().init(*args, **kw)
|
||||
|
||||
@property
|
||||
def groups(self):
|
||||
return self._groups
|
||||
|
||||
@property
|
||||
def addresses(self):
|
||||
if self._addresses is None:
|
||||
self._addresses = tuple([address for group in self._groups
|
||||
for address in group.addresses])
|
||||
return self._addresses
|
||||
|
||||
|
||||
class UniqueAddressHeader(AddressHeader):
|
||||
|
||||
max_count = 1
|
||||
|
||||
|
||||
class SingleAddressHeader(AddressHeader):
|
||||
|
||||
@property
|
||||
def address(self):
|
||||
if len(self.addresses)!=1:
|
||||
raise ValueError(("value of single address header {} is not "
|
||||
"a single address").format(self.name))
|
||||
return self.addresses[0]
|
||||
|
||||
|
||||
class UniqueSingleAddressHeader(SingleAddressHeader):
|
||||
|
||||
max_count = 1
|
||||
|
||||
|
||||
# The header factory #
|
||||
|
||||
_default_header_map = {
|
||||
'subject': UniqueUnstructuredHeader,
|
||||
'date': UniqueDateHeader,
|
||||
'resent-date': DateHeader,
|
||||
'orig-date': UniqueDateHeader,
|
||||
'sender': UniqueSingleAddressHeader,
|
||||
'resent-sender': SingleAddressHeader,
|
||||
'to': UniqueAddressHeader,
|
||||
'resent-to': AddressHeader,
|
||||
'cc': UniqueAddressHeader,
|
||||
'resent-cc': AddressHeader,
|
||||
'bcc': UniqueAddressHeader,
|
||||
'resent-bcc': AddressHeader,
|
||||
'from': UniqueAddressHeader,
|
||||
'resent-from': AddressHeader,
|
||||
'reply-to': UniqueAddressHeader,
|
||||
}
|
||||
|
||||
class HeaderRegistry:
|
||||
|
||||
"""A header_factory and header registry."""
|
||||
|
||||
def __init__(self, base_class=BaseHeader, default_class=UnstructuredHeader,
|
||||
use_default_map=True):
|
||||
"""Create a header_factory that works with the Policy API.
|
||||
|
||||
base_class is the class that will be the last class in the created
|
||||
header class's __bases__ list. default_class is the class that will be
|
||||
used if "name" (see __call__) does not appear in the registry.
|
||||
use_default_map controls whether or not the default mapping of names to
|
||||
specialized classes is copied in to the registry when the factory is
|
||||
created. The default is True.
|
||||
|
||||
"""
|
||||
self.registry = {}
|
||||
self.base_class = base_class
|
||||
self.default_class = default_class
|
||||
if use_default_map:
|
||||
self.registry.update(_default_header_map)
|
||||
|
||||
def map_to_type(self, name, cls):
|
||||
"""Register cls as the specialized class for handling "name" headers.
|
||||
|
||||
"""
|
||||
self.registry[name.lower()] = cls
|
||||
|
||||
def __getitem__(self, name):
|
||||
cls = self.registry.get(name.lower(), self.default_class)
|
||||
return type('_'+cls.__name__, (cls, self.base_class), {})
|
||||
|
||||
def __call__(self, name, value):
|
||||
"""Create a header instance for header 'name' from 'value'.
|
||||
|
||||
Creates a header instance by creating a specialized class for parsing
|
||||
and representing the specified header by combining the factory
|
||||
base_class with a specialized class from the registry or the
|
||||
default_class, and passing the name and value to the constructed
|
||||
class's constructor.
|
||||
|
||||
"""
|
||||
return self[name](name, value)
|
|
@ -64,10 +64,16 @@ class _PolicyBase:
|
|||
except for the changes passed in as keyword arguments.
|
||||
|
||||
"""
|
||||
newpolicy = self.__class__.__new__(self.__class__)
|
||||
for attr, value in self.__dict__.items():
|
||||
if attr not in kw:
|
||||
kw[attr] = value
|
||||
return self.__class__(**kw)
|
||||
object.__setattr__(newpolicy, attr, value)
|
||||
for attr, value in kw.items():
|
||||
if not hasattr(self, attr):
|
||||
raise TypeError(
|
||||
"{!r} is an invalid keyword argument for {}".format(
|
||||
attr, self.__class__.__name__))
|
||||
object.__setattr__(newpolicy, attr, value)
|
||||
return newpolicy
|
||||
|
||||
def __setattr__(self, name, value):
|
||||
if hasattr(self, name):
|
||||
|
|
|
@ -5,7 +5,6 @@
|
|||
"""email package exception classes."""
|
||||
|
||||
|
||||
|
||||
class MessageError(Exception):
|
||||
"""Base class for errors in the email package."""
|
||||
|
||||
|
@ -30,9 +29,8 @@ class CharsetError(MessageError):
|
|||
"""An illegal charset was given."""
|
||||
|
||||
|
||||
|
||||
# These are parsing defects which the parser was able to work around.
|
||||
class MessageDefect(Exception):
|
||||
class MessageDefect(ValueError):
|
||||
"""Base class for a message defect."""
|
||||
|
||||
def __init__(self, line=None):
|
||||
|
@ -58,3 +56,42 @@ class MultipartInvariantViolationDefect(MessageDefect):
|
|||
|
||||
class InvalidMultipartContentTransferEncodingDefect(MessageDefect):
|
||||
"""An invalid content transfer encoding was set on the multipart itself."""
|
||||
|
||||
class UndecodableBytesDefect(MessageDefect):
|
||||
"""Header contained bytes that could not be decoded"""
|
||||
|
||||
class InvalidBase64PaddingDefect(MessageDefect):
|
||||
"""base64 encoded sequence had an incorrect length"""
|
||||
|
||||
class InvalidBase64CharactersDefect(MessageDefect):
|
||||
"""base64 encoded sequence had characters not in base64 alphabet"""
|
||||
|
||||
# These errors are specific to header parsing.
|
||||
|
||||
class HeaderDefect(MessageDefect):
|
||||
"""Base class for a header defect."""
|
||||
|
||||
class InvalidHeaderDefect(HeaderDefect):
|
||||
"""Header is not valid, message gives details."""
|
||||
|
||||
class HeaderMissingRequiredValue(HeaderDefect):
|
||||
"""A header that must have a value had none"""
|
||||
|
||||
class NonPrintableDefect(HeaderDefect):
|
||||
"""ASCII characters outside the ascii-printable range found"""
|
||||
|
||||
def __init__(self, non_printables):
|
||||
super().__init__(non_printables)
|
||||
self.non_printables = non_printables
|
||||
|
||||
def __str__(self):
|
||||
return ("the following ASCII non-printables found in header: "
|
||||
"{}".format(self.non_printables))
|
||||
|
||||
class ObsoleteHeaderDefect(HeaderDefect):
|
||||
"""Header uses syntax declared obsolete by RFC 5322"""
|
||||
|
||||
class NonASCIILocalPartDefect(HeaderDefect):
|
||||
"""local_part contains non-ASCII characters"""
|
||||
# This defect only occurs during unicode parsing, not when
|
||||
# parsing messages decoded from binary.
|
||||
|
|
|
@ -95,9 +95,15 @@ class Generator:
|
|||
self._encoded_NL = self._encode(self._NL)
|
||||
self._EMPTY = ''
|
||||
self._encoded_EMTPY = self._encode('')
|
||||
p = self.policy
|
||||
# Because we use clone (below) when we recursively process message
|
||||
# subparts, and because clone uses the computed policy (not None),
|
||||
# submessages will automatically get set to the computed policy when
|
||||
# they are processed by this code.
|
||||
old_gen_policy = self.policy
|
||||
old_msg_policy = msg.policy
|
||||
try:
|
||||
self.policy = policy
|
||||
msg.policy = policy
|
||||
if unixfrom:
|
||||
ufrom = msg.get_unixfrom()
|
||||
if not ufrom:
|
||||
|
@ -105,7 +111,8 @@ class Generator:
|
|||
self.write(ufrom + self._NL)
|
||||
self._write(msg)
|
||||
finally:
|
||||
self.policy = p
|
||||
self.policy = old_gen_policy
|
||||
msg.policy = old_msg_policy
|
||||
|
||||
def clone(self, fp):
|
||||
"""Clone this generator with the exact same options."""
|
||||
|
|
|
@ -2,11 +2,178 @@
|
|||
code that adds all the email6 features.
|
||||
"""
|
||||
|
||||
from email._policybase import Policy, compat32, Compat32
|
||||
from email._policybase import Policy, Compat32, compat32
|
||||
from email.utils import _has_surrogates
|
||||
from email._headerregistry import HeaderRegistry as _HeaderRegistry
|
||||
|
||||
# XXX: temporarily derive everything from compat32.
|
||||
__all__ = [
|
||||
'Compat32',
|
||||
'compat32',
|
||||
'Policy',
|
||||
'EmailPolicy',
|
||||
'default',
|
||||
'strict',
|
||||
'SMTP',
|
||||
'HTTP',
|
||||
]
|
||||
|
||||
default = compat32
|
||||
class EmailPolicy(Policy):
|
||||
|
||||
"""+
|
||||
PROVISIONAL
|
||||
|
||||
The API extensions enabled by this this policy are currently provisional.
|
||||
Refer to the documentation for details.
|
||||
|
||||
This policy adds new header parsing and folding algorithms. Instead of
|
||||
simple strings, headers are custom objects with custom attributes
|
||||
depending on the type of the field. The folding algorithm fully
|
||||
implements RFCs 2047 and 5322.
|
||||
|
||||
In addition to the settable attributes listed above that apply to
|
||||
all Policies, this policy adds the following additional attributes:
|
||||
|
||||
refold_source -- if the value for a header in the Message object
|
||||
came from the parsing of some source, this attribute
|
||||
indicates whether or not a generator should refold
|
||||
that value when transforming the message back into
|
||||
stream form. The possible values are:
|
||||
|
||||
none -- all source values use original folding
|
||||
long -- source values that have any line that is
|
||||
longer than max_line_length will be
|
||||
refolded
|
||||
all -- all values are refolded.
|
||||
|
||||
The default is 'long'.
|
||||
|
||||
header_factory -- a callable that takes two arguments, 'name' and
|
||||
'value', where 'name' is a header field name and
|
||||
'value' is an unfolded header field value, and
|
||||
returns a string-like object that represents that
|
||||
header. A default header_factory is provided that
|
||||
understands some of the RFC5322 header field types.
|
||||
(Currently address fields and date fields have
|
||||
special treatment, while all other fields are
|
||||
treated as unstructured. This list will be
|
||||
completed before the extension is marked stable.)
|
||||
"""
|
||||
|
||||
refold_source = 'long'
|
||||
header_factory = _HeaderRegistry()
|
||||
|
||||
def __init__(self, **kw):
|
||||
# Ensure that each new instance gets a unique header factory
|
||||
# (as opposed to clones, which share the factory).
|
||||
if 'header_factory' not in kw:
|
||||
object.__setattr__(self, 'header_factory', _HeaderRegistry())
|
||||
super().__init__(**kw)
|
||||
|
||||
# The logic of the next three methods is chosen such that it is possible to
|
||||
# switch a Message object between a Compat32 policy and a policy derived
|
||||
# from this class and have the results stay consistent. This allows a
|
||||
# Message object constructed with this policy to be passed to a library
|
||||
# that only handles Compat32 objects, or to receive such an object and
|
||||
# convert it to use the newer style by just changing its policy. It is
|
||||
# also chosen because it postpones the relatively expensive full rfc5322
|
||||
# parse until as late as possible when parsing from source, since in many
|
||||
# applications only a few headers will actually be inspected.
|
||||
|
||||
def header_source_parse(self, sourcelines):
|
||||
"""+
|
||||
The name is parsed as everything up to the ':' and returned unmodified.
|
||||
The value is determined by stripping leading whitespace off the
|
||||
remainder of the first line, joining all subsequent lines together, and
|
||||
stripping any trailing carriage return or linefeed characters. (This
|
||||
is the same as Compat32).
|
||||
|
||||
"""
|
||||
name, value = sourcelines[0].split(':', 1)
|
||||
value = value.lstrip(' \t') + ''.join(sourcelines[1:])
|
||||
return (name, value.rstrip('\r\n'))
|
||||
|
||||
def header_store_parse(self, name, value):
|
||||
"""+
|
||||
The name is returned unchanged. If the input value has a 'name'
|
||||
attribute and it matches the name ignoring case, the value is returned
|
||||
unchanged. Otherwise the name and value are passed to header_factory
|
||||
method, and the resulting custom header object is returned as the
|
||||
value. In this case a ValueError is raised if the input value contains
|
||||
CR or LF characters.
|
||||
|
||||
"""
|
||||
if hasattr(value, 'name') and value.name.lower() == name.lower():
|
||||
return (name, value)
|
||||
if len(value.splitlines())>1:
|
||||
raise ValueError("Header values may not contain linefeed "
|
||||
"or carriage return characters")
|
||||
return (name, self.header_factory(name, value))
|
||||
|
||||
def header_fetch_parse(self, name, value):
|
||||
"""+
|
||||
If the value has a 'name' attribute, it is returned to unmodified.
|
||||
Otherwise the name and the value with any linesep characters removed
|
||||
are passed to the header_factory method, and the resulting custom
|
||||
header object is returned. Any surrogateescaped bytes get turned
|
||||
into the unicode unknown-character glyph.
|
||||
|
||||
"""
|
||||
if hasattr(value, 'name'):
|
||||
return value
|
||||
return self.header_factory(name, ''.join(value.splitlines()))
|
||||
|
||||
def fold(self, name, value):
|
||||
"""+
|
||||
Header folding is controlled by the refold_source policy setting. A
|
||||
value is considered to be a 'source value' if and only if it does not
|
||||
have a 'name' attribute (having a 'name' attribute means it is a header
|
||||
object of some sort). If a source value needs to be refolded according
|
||||
to the policy, it is converted into a custom header object by passing
|
||||
the name and the value with any linesep characters removed to the
|
||||
header_factory method. Folding of a custom header object is done by
|
||||
calling its fold method with the current policy.
|
||||
|
||||
Source values are split into lines using splitlines. If the value is
|
||||
not to be refolded, the lines are rejoined using the linesep from the
|
||||
policy and returned. The exception is lines containing non-ascii
|
||||
binary data. In that case the value is refolded regardless of the
|
||||
refold_source setting, which causes the binary data to be CTE encoded
|
||||
using the unknown-8bit charset.
|
||||
|
||||
"""
|
||||
return self._fold(name, value, refold_binary=True)
|
||||
|
||||
def fold_binary(self, name, value):
|
||||
"""+
|
||||
The same as fold if cte_type is 7bit, except that the returned value is
|
||||
bytes.
|
||||
|
||||
If cte_type is 8bit, non-ASCII binary data is converted back into
|
||||
bytes. Headers with binary data are not refolded, regardless of the
|
||||
refold_header setting, since there is no way to know whether the binary
|
||||
data consists of single byte characters or multibyte characters.
|
||||
|
||||
"""
|
||||
folded = self._fold(name, value, refold_binary=self.cte_type=='7bit')
|
||||
return folded.encode('ascii', 'surrogateescape')
|
||||
|
||||
def _fold(self, name, value, refold_binary=False):
|
||||
if hasattr(value, 'name'):
|
||||
return value.fold(policy=self)
|
||||
maxlen = self.max_line_length if self.max_line_length else float('inf')
|
||||
lines = value.splitlines()
|
||||
refold = (self.refold_source == 'all' or
|
||||
self.refold_source == 'long' and
|
||||
(len(lines[0])+len(name)+2 > maxlen or
|
||||
any(len(x) > maxlen for x in lines[1:])))
|
||||
if refold or refold_binary and _has_surrogates(value):
|
||||
return self.header_factory(name, ''.join(lines)).fold(policy=self)
|
||||
return name + ': ' + self.linesep.join(lines) + self.linesep
|
||||
|
||||
|
||||
default = EmailPolicy()
|
||||
# Make the default policy use the class default header_factory
|
||||
del default.header_factory
|
||||
strict = default.clone(raise_on_defect=True)
|
||||
SMTP = default.clone(linesep='\r\n')
|
||||
HTTP = default.clone(linesep='\r\n', max_line_length=None)
|
||||
|
|
|
@ -62,6 +62,13 @@ escapesre = re.compile(r'[\\"]')
|
|||
_has_surrogates = re.compile(
|
||||
'([^\ud800-\udbff]|\A)[\udc00-\udfff]([^\udc00-\udfff]|\Z)').search
|
||||
|
||||
# How to deal with a string containing bytes before handing it to the
|
||||
# application through the 'normal' interface.
|
||||
def _sanitize(string):
|
||||
# Turn any escaped bytes into unicode 'unknown' char.
|
||||
original_bytes = string.encode('ascii', 'surrogateescape')
|
||||
return original_bytes.decode('ascii', 'replace')
|
||||
|
||||
|
||||
# Helpers
|
||||
|
||||
|
|
|
@ -65,3 +65,9 @@ class TestEmailBase(unittest.TestCase):
|
|||
def assertBytesEqual(self, first, second, msg):
|
||||
"""Our byte strings are really encoded strings; improve diff output"""
|
||||
self.assertEqual(self._bytes_repr(first), self._bytes_repr(second))
|
||||
|
||||
def assertDefectsEqual(self, actual, expected):
|
||||
self.assertEqual(len(actual), len(expected), actual)
|
||||
for i in range(len(actual)):
|
||||
self.assertIsInstance(actual[i], expected[i],
|
||||
'item {}'.format(i))
|
||||
|
|
|
@ -0,0 +1,187 @@
|
|||
import unittest
|
||||
from email import _encoded_words as _ew
|
||||
from email import errors
|
||||
from test.test_email import TestEmailBase
|
||||
|
||||
|
||||
class TestDecodeQ(TestEmailBase):
|
||||
|
||||
def _test(self, source, ex_result, ex_defects=[]):
|
||||
result, defects = _ew.decode_q(source)
|
||||
self.assertEqual(result, ex_result)
|
||||
self.assertDefectsEqual(defects, ex_defects)
|
||||
|
||||
def test_no_encoded(self):
|
||||
self._test(b'foobar', b'foobar')
|
||||
|
||||
def test_spaces(self):
|
||||
self._test(b'foo=20bar=20', b'foo bar ')
|
||||
self._test(b'foo_bar_', b'foo bar ')
|
||||
|
||||
def test_run_of_encoded(self):
|
||||
self._test(b'foo=20=20=21=2Cbar', b'foo !,bar')
|
||||
|
||||
|
||||
class TestDecodeB(TestEmailBase):
|
||||
|
||||
def _test(self, source, ex_result, ex_defects=[]):
|
||||
result, defects = _ew.decode_b(source)
|
||||
self.assertEqual(result, ex_result)
|
||||
self.assertDefectsEqual(defects, ex_defects)
|
||||
|
||||
def test_simple(self):
|
||||
self._test(b'Zm9v', b'foo')
|
||||
|
||||
def test_missing_padding(self):
|
||||
self._test(b'dmk', b'vi', [errors.InvalidBase64PaddingDefect])
|
||||
|
||||
def test_invalid_character(self):
|
||||
self._test(b'dm\x01k===', b'vi', [errors.InvalidBase64CharactersDefect])
|
||||
|
||||
def test_invalid_character_and_bad_padding(self):
|
||||
self._test(b'dm\x01k', b'vi', [errors.InvalidBase64CharactersDefect,
|
||||
errors.InvalidBase64PaddingDefect])
|
||||
|
||||
|
||||
class TestDecode(TestEmailBase):
|
||||
|
||||
def test_wrong_format_input_raises(self):
|
||||
with self.assertRaises(ValueError):
|
||||
_ew.decode('=?badone?=')
|
||||
with self.assertRaises(ValueError):
|
||||
_ew.decode('=?')
|
||||
with self.assertRaises(ValueError):
|
||||
_ew.decode('')
|
||||
|
||||
def _test(self, source, result, charset='us-ascii', lang='', defects=[]):
|
||||
res, char, l, d = _ew.decode(source)
|
||||
self.assertEqual(res, result)
|
||||
self.assertEqual(char, charset)
|
||||
self.assertEqual(l, lang)
|
||||
self.assertDefectsEqual(d, defects)
|
||||
|
||||
def test_simple_q(self):
|
||||
self._test('=?us-ascii?q?foo?=', 'foo')
|
||||
|
||||
def test_simple_b(self):
|
||||
self._test('=?us-ascii?b?dmk=?=', 'vi')
|
||||
|
||||
def test_q_case_ignored(self):
|
||||
self._test('=?us-ascii?Q?foo?=', 'foo')
|
||||
|
||||
def test_b_case_ignored(self):
|
||||
self._test('=?us-ascii?B?dmk=?=', 'vi')
|
||||
|
||||
def test_non_trivial_q(self):
|
||||
self._test('=?latin-1?q?=20F=fcr=20Elise=20?=', ' Für Elise ', 'latin-1')
|
||||
|
||||
def test_q_escpaed_bytes_preserved(self):
|
||||
self._test(b'=?us-ascii?q?=20\xACfoo?='.decode('us-ascii',
|
||||
'surrogateescape'),
|
||||
' \uDCACfoo',
|
||||
defects = [errors.UndecodableBytesDefect])
|
||||
|
||||
def test_b_undecodable_bytes_ignored_with_defect(self):
|
||||
self._test(b'=?us-ascii?b?dm\xACk?='.decode('us-ascii',
|
||||
'surrogateescape'),
|
||||
'vi',
|
||||
defects = [
|
||||
errors.InvalidBase64CharactersDefect,
|
||||
errors.InvalidBase64PaddingDefect])
|
||||
|
||||
def test_b_invalid_bytes_ignored_with_defect(self):
|
||||
self._test('=?us-ascii?b?dm\x01k===?=',
|
||||
'vi',
|
||||
defects = [errors.InvalidBase64CharactersDefect])
|
||||
|
||||
def test_b_invalid_bytes_incorrect_padding(self):
|
||||
self._test('=?us-ascii?b?dm\x01k?=',
|
||||
'vi',
|
||||
defects = [
|
||||
errors.InvalidBase64CharactersDefect,
|
||||
errors.InvalidBase64PaddingDefect])
|
||||
|
||||
def test_b_padding_defect(self):
|
||||
self._test('=?us-ascii?b?dmk?=',
|
||||
'vi',
|
||||
defects = [errors.InvalidBase64PaddingDefect])
|
||||
|
||||
def test_nonnull_lang(self):
|
||||
self._test('=?us-ascii*jive?q?test?=', 'test', lang='jive')
|
||||
|
||||
def test_unknown_8bit_charset(self):
|
||||
self._test('=?unknown-8bit?q?foo=ACbar?=',
|
||||
b'foo\xacbar'.decode('ascii', 'surrogateescape'),
|
||||
charset = 'unknown-8bit',
|
||||
defects = [])
|
||||
|
||||
def test_unknown_charset(self):
|
||||
self._test('=?foobar?q?foo=ACbar?=',
|
||||
b'foo\xacbar'.decode('ascii', 'surrogateescape'),
|
||||
charset = 'foobar',
|
||||
# XXX Should this be a new Defect instead?
|
||||
defects = [errors.CharsetError])
|
||||
|
||||
|
||||
class TestEncodeQ(TestEmailBase):
|
||||
|
||||
def _test(self, src, expected):
|
||||
self.assertEqual(_ew.encode_q(src), expected)
|
||||
|
||||
def test_all_safe(self):
|
||||
self._test(b'foobar', 'foobar')
|
||||
|
||||
def test_spaces(self):
|
||||
self._test(b'foo bar ', 'foo_bar_')
|
||||
|
||||
def test_run_of_encodables(self):
|
||||
self._test(b'foo ,,bar', 'foo__=2C=2Cbar')
|
||||
|
||||
|
||||
class TestEncodeB(TestEmailBase):
|
||||
|
||||
def test_simple(self):
|
||||
self.assertEqual(_ew.encode_b(b'foo'), 'Zm9v')
|
||||
|
||||
def test_padding(self):
|
||||
self.assertEqual(_ew.encode_b(b'vi'), 'dmk=')
|
||||
|
||||
|
||||
class TestEncode(TestEmailBase):
|
||||
|
||||
def test_q(self):
|
||||
self.assertEqual(_ew.encode('foo', 'utf-8', 'q'), '=?utf-8?q?foo?=')
|
||||
|
||||
def test_b(self):
|
||||
self.assertEqual(_ew.encode('foo', 'utf-8', 'b'), '=?utf-8?b?Zm9v?=')
|
||||
|
||||
def test_auto_q(self):
|
||||
self.assertEqual(_ew.encode('foo', 'utf-8'), '=?utf-8?q?foo?=')
|
||||
|
||||
def test_auto_q_if_short_mostly_safe(self):
|
||||
self.assertEqual(_ew.encode('vi.', 'utf-8'), '=?utf-8?q?vi=2E?=')
|
||||
|
||||
def test_auto_b_if_enough_unsafe(self):
|
||||
self.assertEqual(_ew.encode('.....', 'utf-8'), '=?utf-8?b?Li4uLi4=?=')
|
||||
|
||||
def test_auto_b_if_long_unsafe(self):
|
||||
self.assertEqual(_ew.encode('vi.vi.vi.vi.vi.', 'utf-8'),
|
||||
'=?utf-8?b?dmkudmkudmkudmkudmku?=')
|
||||
|
||||
def test_auto_q_if_long_mostly_safe(self):
|
||||
self.assertEqual(_ew.encode('vi vi vi.vi ', 'utf-8'),
|
||||
'=?utf-8?q?vi_vi_vi=2Evi_?=')
|
||||
|
||||
def test_utf8_default(self):
|
||||
self.assertEqual(_ew.encode('foo'), '=?utf-8?q?foo?=')
|
||||
|
||||
def test_lang(self):
|
||||
self.assertEqual(_ew.encode('foo', lang='jive'), '=?utf-8*jive?q?foo?=')
|
||||
|
||||
def test_unknown_8bit(self):
|
||||
self.assertEqual(_ew.encode('foo\uDCACbar', charset='unknown-8bit'),
|
||||
'=?unknown-8bit?q?foo=ACbar?=')
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
unittest.main()
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,717 @@
|
|||
import datetime
|
||||
import textwrap
|
||||
import unittest
|
||||
from email import errors
|
||||
from email import policy
|
||||
from test.test_email import TestEmailBase
|
||||
from email import _headerregistry
|
||||
# Address and Group are public but I'm not sure where to put them yet.
|
||||
from email._headerregistry import Address, Group
|
||||
|
||||
|
||||
class TestHeaderRegistry(TestEmailBase):
|
||||
|
||||
def test_arbitrary_name_unstructured(self):
|
||||
factory = _headerregistry.HeaderRegistry()
|
||||
h = factory('foobar', 'test')
|
||||
self.assertIsInstance(h, _headerregistry.BaseHeader)
|
||||
self.assertIsInstance(h, _headerregistry.UnstructuredHeader)
|
||||
|
||||
def test_name_case_ignored(self):
|
||||
factory = _headerregistry.HeaderRegistry()
|
||||
# Whitebox check that test is valid
|
||||
self.assertNotIn('Subject', factory.registry)
|
||||
h = factory('Subject', 'test')
|
||||
self.assertIsInstance(h, _headerregistry.BaseHeader)
|
||||
self.assertIsInstance(h, _headerregistry.UniqueUnstructuredHeader)
|
||||
|
||||
class FooBase:
|
||||
def __init__(self, *args, **kw):
|
||||
pass
|
||||
|
||||
def test_override_default_base_class(self):
|
||||
factory = _headerregistry.HeaderRegistry(base_class=self.FooBase)
|
||||
h = factory('foobar', 'test')
|
||||
self.assertIsInstance(h, self.FooBase)
|
||||
self.assertIsInstance(h, _headerregistry.UnstructuredHeader)
|
||||
|
||||
class FooDefault:
|
||||
parse = _headerregistry.UnstructuredHeader.parse
|
||||
|
||||
def test_override_default_class(self):
|
||||
factory = _headerregistry.HeaderRegistry(default_class=self.FooDefault)
|
||||
h = factory('foobar', 'test')
|
||||
self.assertIsInstance(h, _headerregistry.BaseHeader)
|
||||
self.assertIsInstance(h, self.FooDefault)
|
||||
|
||||
def test_override_default_class_only_overrides_default(self):
|
||||
factory = _headerregistry.HeaderRegistry(default_class=self.FooDefault)
|
||||
h = factory('subject', 'test')
|
||||
self.assertIsInstance(h, _headerregistry.BaseHeader)
|
||||
self.assertIsInstance(h, _headerregistry.UniqueUnstructuredHeader)
|
||||
|
||||
def test_dont_use_default_map(self):
|
||||
factory = _headerregistry.HeaderRegistry(use_default_map=False)
|
||||
h = factory('subject', 'test')
|
||||
self.assertIsInstance(h, _headerregistry.BaseHeader)
|
||||
self.assertIsInstance(h, _headerregistry.UnstructuredHeader)
|
||||
|
||||
def test_map_to_type(self):
|
||||
factory = _headerregistry.HeaderRegistry()
|
||||
h1 = factory('foobar', 'test')
|
||||
factory.map_to_type('foobar', _headerregistry.UniqueUnstructuredHeader)
|
||||
h2 = factory('foobar', 'test')
|
||||
self.assertIsInstance(h1, _headerregistry.BaseHeader)
|
||||
self.assertIsInstance(h1, _headerregistry.UnstructuredHeader)
|
||||
self.assertIsInstance(h2, _headerregistry.BaseHeader)
|
||||
self.assertIsInstance(h2, _headerregistry.UniqueUnstructuredHeader)
|
||||
|
||||
|
||||
class TestHeaderBase(TestEmailBase):
|
||||
|
||||
factory = _headerregistry.HeaderRegistry()
|
||||
|
||||
def make_header(self, name, value):
|
||||
return self.factory(name, value)
|
||||
|
||||
|
||||
class TestBaseHeaderFeatures(TestHeaderBase):
|
||||
|
||||
def test_str(self):
|
||||
h = self.make_header('subject', 'this is a test')
|
||||
self.assertIsInstance(h, str)
|
||||
self.assertEqual(h, 'this is a test')
|
||||
self.assertEqual(str(h), 'this is a test')
|
||||
|
||||
def test_substr(self):
|
||||
h = self.make_header('subject', 'this is a test')
|
||||
self.assertEqual(h[5:7], 'is')
|
||||
|
||||
def test_has_name(self):
|
||||
h = self.make_header('subject', 'this is a test')
|
||||
self.assertEqual(h.name, 'subject')
|
||||
|
||||
def _test_attr_ro(self, attr):
|
||||
h = self.make_header('subject', 'this is a test')
|
||||
with self.assertRaises(AttributeError):
|
||||
setattr(h, attr, 'foo')
|
||||
|
||||
def test_name_read_only(self):
|
||||
self._test_attr_ro('name')
|
||||
|
||||
def test_defects_read_only(self):
|
||||
self._test_attr_ro('defects')
|
||||
|
||||
def test_defects_is_tuple(self):
|
||||
h = self.make_header('subject', 'this is a test')
|
||||
self.assertEqual(len(h.defects), 0)
|
||||
self.assertIsInstance(h.defects, tuple)
|
||||
# Make sure it is still true when there are defects.
|
||||
h = self.make_header('date', '')
|
||||
self.assertEqual(len(h.defects), 1)
|
||||
self.assertIsInstance(h.defects, tuple)
|
||||
|
||||
# XXX: FIXME
|
||||
#def test_CR_in_value(self):
|
||||
# # XXX: this also re-raises the issue of embedded headers,
|
||||
# # need test and solution for that.
|
||||
# value = '\r'.join(['this is', ' a test'])
|
||||
# h = self.make_header('subject', value)
|
||||
# self.assertEqual(h, value)
|
||||
# self.assertDefectsEqual(h.defects, [errors.ObsoleteHeaderDefect])
|
||||
|
||||
def test_RFC2047_value_decoded(self):
|
||||
value = '=?utf-8?q?this_is_a_test?='
|
||||
h = self.make_header('subject', value)
|
||||
self.assertEqual(h, 'this is a test')
|
||||
|
||||
|
||||
class TestDateHeader(TestHeaderBase):
|
||||
|
||||
datestring = 'Sun, 23 Sep 2001 20:10:55 -0700'
|
||||
utcoffset = datetime.timedelta(hours=-7)
|
||||
tz = datetime.timezone(utcoffset)
|
||||
dt = datetime.datetime(2001, 9, 23, 20, 10, 55, tzinfo=tz)
|
||||
|
||||
def test_parse_date(self):
|
||||
h = self.make_header('date', self.datestring)
|
||||
self.assertEqual(h, self.datestring)
|
||||
self.assertEqual(h.datetime, self.dt)
|
||||
self.assertEqual(h.datetime.utcoffset(), self.utcoffset)
|
||||
self.assertEqual(h.defects, ())
|
||||
|
||||
def test_set_from_datetime(self):
|
||||
h = self.make_header('date', self.dt)
|
||||
self.assertEqual(h, self.datestring)
|
||||
self.assertEqual(h.datetime, self.dt)
|
||||
self.assertEqual(h.defects, ())
|
||||
|
||||
def test_date_header_properties(self):
|
||||
h = self.make_header('date', self.datestring)
|
||||
self.assertIsInstance(h, _headerregistry.UniqueDateHeader)
|
||||
self.assertEqual(h.max_count, 1)
|
||||
self.assertEqual(h.defects, ())
|
||||
|
||||
def test_resent_date_header_properties(self):
|
||||
h = self.make_header('resent-date', self.datestring)
|
||||
self.assertIsInstance(h, _headerregistry.DateHeader)
|
||||
self.assertEqual(h.max_count, None)
|
||||
self.assertEqual(h.defects, ())
|
||||
|
||||
def test_no_value_is_defect(self):
|
||||
h = self.make_header('date', '')
|
||||
self.assertEqual(len(h.defects), 1)
|
||||
self.assertIsInstance(h.defects[0], errors.HeaderMissingRequiredValue)
|
||||
|
||||
def test_datetime_read_only(self):
|
||||
h = self.make_header('date', self.datestring)
|
||||
with self.assertRaises(AttributeError):
|
||||
h.datetime = 'foo'
|
||||
|
||||
|
||||
class TestAddressHeader(TestHeaderBase):
|
||||
|
||||
examples = {
|
||||
|
||||
'empty':
|
||||
('<>',
|
||||
[errors.InvalidHeaderDefect],
|
||||
'<>',
|
||||
'',
|
||||
'<>',
|
||||
'',
|
||||
'',
|
||||
None),
|
||||
|
||||
'address_only':
|
||||
('zippy@pinhead.com',
|
||||
[],
|
||||
'zippy@pinhead.com',
|
||||
'',
|
||||
'zippy@pinhead.com',
|
||||
'zippy',
|
||||
'pinhead.com',
|
||||
None),
|
||||
|
||||
'name_and_address':
|
||||
('Zaphrod Beblebrux <zippy@pinhead.com>',
|
||||
[],
|
||||
'Zaphrod Beblebrux <zippy@pinhead.com>',
|
||||
'Zaphrod Beblebrux',
|
||||
'zippy@pinhead.com',
|
||||
'zippy',
|
||||
'pinhead.com',
|
||||
None),
|
||||
|
||||
'quoted_local_part':
|
||||
('Zaphrod Beblebrux <"foo bar"@pinhead.com>',
|
||||
[],
|
||||
'Zaphrod Beblebrux <"foo bar"@pinhead.com>',
|
||||
'Zaphrod Beblebrux',
|
||||
'"foo bar"@pinhead.com',
|
||||
'foo bar',
|
||||
'pinhead.com',
|
||||
None),
|
||||
|
||||
'quoted_parens_in_name':
|
||||
(r'"A \(Special\) Person" <person@dom.ain>',
|
||||
[],
|
||||
'"A (Special) Person" <person@dom.ain>',
|
||||
'A (Special) Person',
|
||||
'person@dom.ain',
|
||||
'person',
|
||||
'dom.ain',
|
||||
None),
|
||||
|
||||
'quoted_backslashes_in_name':
|
||||
(r'"Arthur \\Backslash\\ Foobar" <person@dom.ain>',
|
||||
[],
|
||||
r'"Arthur \\Backslash\\ Foobar" <person@dom.ain>',
|
||||
r'Arthur \Backslash\ Foobar',
|
||||
'person@dom.ain',
|
||||
'person',
|
||||
'dom.ain',
|
||||
None),
|
||||
|
||||
'name_with_dot':
|
||||
('John X. Doe <jxd@example.com>',
|
||||
[errors.ObsoleteHeaderDefect],
|
||||
'"John X. Doe" <jxd@example.com>',
|
||||
'John X. Doe',
|
||||
'jxd@example.com',
|
||||
'jxd',
|
||||
'example.com',
|
||||
None),
|
||||
|
||||
'quoted_strings_in_local_part':
|
||||
('""example" example"@example.com',
|
||||
[errors.InvalidHeaderDefect]*3,
|
||||
'"example example"@example.com',
|
||||
'',
|
||||
'"example example"@example.com',
|
||||
'example example',
|
||||
'example.com',
|
||||
None),
|
||||
|
||||
'escaped_quoted_strings_in_local_part':
|
||||
(r'"\"example\" example"@example.com',
|
||||
[],
|
||||
r'"\"example\" example"@example.com',
|
||||
'',
|
||||
r'"\"example\" example"@example.com',
|
||||
r'"example" example',
|
||||
'example.com',
|
||||
None),
|
||||
|
||||
'escaped_escapes_in_local_part':
|
||||
(r'"\\"example\\" example"@example.com',
|
||||
[errors.InvalidHeaderDefect]*5,
|
||||
r'"\\example\\\\ example"@example.com',
|
||||
'',
|
||||
r'"\\example\\\\ example"@example.com',
|
||||
r'\example\\ example',
|
||||
'example.com',
|
||||
None),
|
||||
|
||||
'spaces_in_unquoted_local_part_collapsed':
|
||||
('merwok wok @example.com',
|
||||
[errors.InvalidHeaderDefect]*2,
|
||||
'"merwok wok"@example.com',
|
||||
'',
|
||||
'"merwok wok"@example.com',
|
||||
'merwok wok',
|
||||
'example.com',
|
||||
None),
|
||||
|
||||
'spaces_around_dots_in_local_part_removed':
|
||||
('merwok. wok . wok@example.com',
|
||||
[errors.ObsoleteHeaderDefect],
|
||||
'merwok.wok.wok@example.com',
|
||||
'',
|
||||
'merwok.wok.wok@example.com',
|
||||
'merwok.wok.wok',
|
||||
'example.com',
|
||||
None),
|
||||
|
||||
}
|
||||
|
||||
# XXX: Need many more examples, and in particular some with names in
|
||||
# trailing comments, which aren't currently handled. comments in
|
||||
# general are not handled yet.
|
||||
|
||||
def _test_single_addr(self, source, defects, decoded, display_name,
|
||||
addr_spec, username, domain, comment):
|
||||
h = self.make_header('sender', source)
|
||||
self.assertEqual(h, decoded)
|
||||
self.assertDefectsEqual(h.defects, defects)
|
||||
a = h.address
|
||||
self.assertEqual(str(a), decoded)
|
||||
self.assertEqual(len(h.groups), 1)
|
||||
self.assertEqual([a], list(h.groups[0].addresses))
|
||||
self.assertEqual([a], list(h.addresses))
|
||||
self.assertEqual(a.display_name, display_name)
|
||||
self.assertEqual(a.addr_spec, addr_spec)
|
||||
self.assertEqual(a.username, username)
|
||||
self.assertEqual(a.domain, domain)
|
||||
# XXX: we have no comment support yet.
|
||||
#self.assertEqual(a.comment, comment)
|
||||
|
||||
for name in examples:
|
||||
locals()['test_'+name] = (
|
||||
lambda self, name=name:
|
||||
self._test_single_addr(*self.examples[name]))
|
||||
|
||||
def _test_group_single_addr(self, source, defects, decoded, display_name,
|
||||
addr_spec, username, domain, comment):
|
||||
source = 'foo: {};'.format(source)
|
||||
gdecoded = 'foo: {};'.format(decoded) if decoded else 'foo:;'
|
||||
h = self.make_header('to', source)
|
||||
self.assertEqual(h, gdecoded)
|
||||
self.assertDefectsEqual(h.defects, defects)
|
||||
self.assertEqual(h.groups[0].addresses, h.addresses)
|
||||
self.assertEqual(len(h.groups), 1)
|
||||
self.assertEqual(len(h.addresses), 1)
|
||||
a = h.addresses[0]
|
||||
self.assertEqual(str(a), decoded)
|
||||
self.assertEqual(a.display_name, display_name)
|
||||
self.assertEqual(a.addr_spec, addr_spec)
|
||||
self.assertEqual(a.username, username)
|
||||
self.assertEqual(a.domain, domain)
|
||||
|
||||
for name in examples:
|
||||
locals()['test_group_'+name] = (
|
||||
lambda self, name=name:
|
||||
self._test_group_single_addr(*self.examples[name]))
|
||||
|
||||
def test_simple_address_list(self):
|
||||
value = ('Fred <dinsdale@python.org>, foo@example.com, '
|
||||
'"Harry W. Hastings" <hasty@example.com>')
|
||||
h = self.make_header('to', value)
|
||||
self.assertEqual(h, value)
|
||||
self.assertEqual(len(h.groups), 3)
|
||||
self.assertEqual(len(h.addresses), 3)
|
||||
for i in range(3):
|
||||
self.assertEqual(h.groups[i].addresses[0], h.addresses[i])
|
||||
self.assertEqual(str(h.addresses[0]), 'Fred <dinsdale@python.org>')
|
||||
self.assertEqual(str(h.addresses[1]), 'foo@example.com')
|
||||
self.assertEqual(str(h.addresses[2]),
|
||||
'"Harry W. Hastings" <hasty@example.com>')
|
||||
self.assertEqual(h.addresses[2].display_name,
|
||||
'Harry W. Hastings')
|
||||
|
||||
def test_complex_address_list(self):
|
||||
examples = list(self.examples.values())
|
||||
source = ('dummy list:;, another: (empty);,' +
|
||||
', '.join([x[0] for x in examples[:4]]) + ', ' +
|
||||
r'"A \"list\"": ' +
|
||||
', '.join([x[0] for x in examples[4:6]]) + ';,' +
|
||||
', '.join([x[0] for x in examples[6:]])
|
||||
)
|
||||
# XXX: the fact that (empty) disappears here is a potential API design
|
||||
# bug. We don't currently have a way to preserve comments.
|
||||
expected = ('dummy list:;, another:;, ' +
|
||||
', '.join([x[2] for x in examples[:4]]) + ', ' +
|
||||
r'"A \"list\"": ' +
|
||||
', '.join([x[2] for x in examples[4:6]]) + ';, ' +
|
||||
', '.join([x[2] for x in examples[6:]])
|
||||
)
|
||||
|
||||
h = self.make_header('to', source)
|
||||
self.assertEqual(h.split(','), expected.split(','))
|
||||
self.assertEqual(h, expected)
|
||||
self.assertEqual(len(h.groups), 7 + len(examples) - 6)
|
||||
self.assertEqual(h.groups[0].display_name, 'dummy list')
|
||||
self.assertEqual(h.groups[1].display_name, 'another')
|
||||
self.assertEqual(h.groups[6].display_name, 'A "list"')
|
||||
self.assertEqual(len(h.addresses), len(examples))
|
||||
for i in range(4):
|
||||
self.assertIsNone(h.groups[i+2].display_name)
|
||||
self.assertEqual(str(h.groups[i+2].addresses[0]), examples[i][2])
|
||||
for i in range(7, 7 + len(examples) - 6):
|
||||
self.assertIsNone(h.groups[i].display_name)
|
||||
self.assertEqual(str(h.groups[i].addresses[0]), examples[i-1][2])
|
||||
for i in range(len(examples)):
|
||||
self.assertEqual(str(h.addresses[i]), examples[i][2])
|
||||
self.assertEqual(h.addresses[i].addr_spec, examples[i][4])
|
||||
|
||||
def test_address_read_only(self):
|
||||
h = self.make_header('sender', 'abc@xyz.com')
|
||||
with self.assertRaises(AttributeError):
|
||||
h.address = 'foo'
|
||||
|
||||
def test_addresses_read_only(self):
|
||||
h = self.make_header('sender', 'abc@xyz.com')
|
||||
with self.assertRaises(AttributeError):
|
||||
h.addresses = 'foo'
|
||||
|
||||
def test_groups_read_only(self):
|
||||
h = self.make_header('sender', 'abc@xyz.com')
|
||||
with self.assertRaises(AttributeError):
|
||||
h.groups = 'foo'
|
||||
|
||||
def test_addresses_types(self):
|
||||
source = 'me <who@example.com>'
|
||||
h = self.make_header('to', source)
|
||||
self.assertIsInstance(h.addresses, tuple)
|
||||
self.assertIsInstance(h.addresses[0], Address)
|
||||
|
||||
def test_groups_types(self):
|
||||
source = 'me <who@example.com>'
|
||||
h = self.make_header('to', source)
|
||||
self.assertIsInstance(h.groups, tuple)
|
||||
self.assertIsInstance(h.groups[0], Group)
|
||||
|
||||
def test_set_from_Address(self):
|
||||
h = self.make_header('to', Address('me', 'foo', 'example.com'))
|
||||
self.assertEqual(h, 'me <foo@example.com>')
|
||||
|
||||
def test_set_from_Address_list(self):
|
||||
h = self.make_header('to', [Address('me', 'foo', 'example.com'),
|
||||
Address('you', 'bar', 'example.com')])
|
||||
self.assertEqual(h, 'me <foo@example.com>, you <bar@example.com>')
|
||||
|
||||
def test_set_from_Address_and_Group_list(self):
|
||||
h = self.make_header('to', [Address('me', 'foo', 'example.com'),
|
||||
Group('bing', [Address('fiz', 'z', 'b.com'),
|
||||
Address('zif', 'f', 'c.com')]),
|
||||
Address('you', 'bar', 'example.com')])
|
||||
self.assertEqual(h, 'me <foo@example.com>, bing: fiz <z@b.com>, '
|
||||
'zif <f@c.com>;, you <bar@example.com>')
|
||||
self.assertEqual(h.fold(policy=policy.default.clone(max_line_length=40)),
|
||||
'to: me <foo@example.com>,\n'
|
||||
' bing: fiz <z@b.com>, zif <f@c.com>;,\n'
|
||||
' you <bar@example.com>\n')
|
||||
|
||||
def test_set_from_Group_list(self):
|
||||
h = self.make_header('to', [Group('bing', [Address('fiz', 'z', 'b.com'),
|
||||
Address('zif', 'f', 'c.com')])])
|
||||
self.assertEqual(h, 'bing: fiz <z@b.com>, zif <f@c.com>;')
|
||||
|
||||
|
||||
class TestAddressAndGroup(TestEmailBase):
|
||||
|
||||
def _test_attr_ro(self, obj, attr):
|
||||
with self.assertRaises(AttributeError):
|
||||
setattr(obj, attr, 'foo')
|
||||
|
||||
def test_address_display_name_ro(self):
|
||||
self._test_attr_ro(Address('foo', 'bar', 'baz'), 'display_name')
|
||||
|
||||
def test_address_username_ro(self):
|
||||
self._test_attr_ro(Address('foo', 'bar', 'baz'), 'username')
|
||||
|
||||
def test_address_domain_ro(self):
|
||||
self._test_attr_ro(Address('foo', 'bar', 'baz'), 'domain')
|
||||
|
||||
def test_group_display_name_ro(self):
|
||||
self._test_attr_ro(Group('foo'), 'display_name')
|
||||
|
||||
def test_group_addresses_ro(self):
|
||||
self._test_attr_ro(Group('foo'), 'addresses')
|
||||
|
||||
def test_address_from_username_domain(self):
|
||||
a = Address('foo', 'bar', 'baz')
|
||||
self.assertEqual(a.display_name, 'foo')
|
||||
self.assertEqual(a.username, 'bar')
|
||||
self.assertEqual(a.domain, 'baz')
|
||||
self.assertEqual(a.addr_spec, 'bar@baz')
|
||||
self.assertEqual(str(a), 'foo <bar@baz>')
|
||||
|
||||
def test_address_from_addr_spec(self):
|
||||
a = Address('foo', addr_spec='bar@baz')
|
||||
self.assertEqual(a.display_name, 'foo')
|
||||
self.assertEqual(a.username, 'bar')
|
||||
self.assertEqual(a.domain, 'baz')
|
||||
self.assertEqual(a.addr_spec, 'bar@baz')
|
||||
self.assertEqual(str(a), 'foo <bar@baz>')
|
||||
|
||||
def test_address_with_no_display_name(self):
|
||||
a = Address(addr_spec='bar@baz')
|
||||
self.assertEqual(a.display_name, '')
|
||||
self.assertEqual(a.username, 'bar')
|
||||
self.assertEqual(a.domain, 'baz')
|
||||
self.assertEqual(a.addr_spec, 'bar@baz')
|
||||
self.assertEqual(str(a), 'bar@baz')
|
||||
|
||||
def test_null_address(self):
|
||||
a = Address()
|
||||
self.assertEqual(a.display_name, '')
|
||||
self.assertEqual(a.username, '')
|
||||
self.assertEqual(a.domain, '')
|
||||
self.assertEqual(a.addr_spec, '<>')
|
||||
self.assertEqual(str(a), '<>')
|
||||
|
||||
def test_domain_only(self):
|
||||
# This isn't really a valid address.
|
||||
a = Address(domain='buzz')
|
||||
self.assertEqual(a.display_name, '')
|
||||
self.assertEqual(a.username, '')
|
||||
self.assertEqual(a.domain, 'buzz')
|
||||
self.assertEqual(a.addr_spec, '@buzz')
|
||||
self.assertEqual(str(a), '@buzz')
|
||||
|
||||
def test_username_only(self):
|
||||
# This isn't really a valid address.
|
||||
a = Address(username='buzz')
|
||||
self.assertEqual(a.display_name, '')
|
||||
self.assertEqual(a.username, 'buzz')
|
||||
self.assertEqual(a.domain, '')
|
||||
self.assertEqual(a.addr_spec, 'buzz')
|
||||
self.assertEqual(str(a), 'buzz')
|
||||
|
||||
def test_display_name_only(self):
|
||||
a = Address('buzz')
|
||||
self.assertEqual(a.display_name, 'buzz')
|
||||
self.assertEqual(a.username, '')
|
||||
self.assertEqual(a.domain, '')
|
||||
self.assertEqual(a.addr_spec, '<>')
|
||||
self.assertEqual(str(a), 'buzz <>')
|
||||
|
||||
def test_quoting(self):
|
||||
# Ideally we'd check every special individually, but I'm not up for
|
||||
# writing that many tests.
|
||||
a = Address('Sara J.', 'bad name', 'example.com')
|
||||
self.assertEqual(a.display_name, 'Sara J.')
|
||||
self.assertEqual(a.username, 'bad name')
|
||||
self.assertEqual(a.domain, 'example.com')
|
||||
self.assertEqual(a.addr_spec, '"bad name"@example.com')
|
||||
self.assertEqual(str(a), '"Sara J." <"bad name"@example.com>')
|
||||
|
||||
def test_il8n(self):
|
||||
a = Address('Éric', 'wok', 'exàmple.com')
|
||||
self.assertEqual(a.display_name, 'Éric')
|
||||
self.assertEqual(a.username, 'wok')
|
||||
self.assertEqual(a.domain, 'exàmple.com')
|
||||
self.assertEqual(a.addr_spec, 'wok@exàmple.com')
|
||||
self.assertEqual(str(a), 'Éric <wok@exàmple.com>')
|
||||
|
||||
# XXX: there is an API design issue that needs to be solved here.
|
||||
#def test_non_ascii_username_raises(self):
|
||||
# with self.assertRaises(ValueError):
|
||||
# Address('foo', 'wők', 'example.com')
|
||||
|
||||
def test_non_ascii_username_in_addr_spec_raises(self):
|
||||
with self.assertRaises(ValueError):
|
||||
Address('foo', addr_spec='wők@example.com')
|
||||
|
||||
def test_address_addr_spec_and_username_raises(self):
|
||||
with self.assertRaises(TypeError):
|
||||
Address('foo', username='bing', addr_spec='bar@baz')
|
||||
|
||||
def test_address_addr_spec_and_domain_raises(self):
|
||||
with self.assertRaises(TypeError):
|
||||
Address('foo', domain='bing', addr_spec='bar@baz')
|
||||
|
||||
def test_address_addr_spec_and_username_and_domain_raises(self):
|
||||
with self.assertRaises(TypeError):
|
||||
Address('foo', username='bong', domain='bing', addr_spec='bar@baz')
|
||||
|
||||
def test_space_in_addr_spec_username_raises(self):
|
||||
with self.assertRaises(ValueError):
|
||||
Address('foo', addr_spec="bad name@example.com")
|
||||
|
||||
def test_bad_addr_sepc_raises(self):
|
||||
with self.assertRaises(ValueError):
|
||||
Address('foo', addr_spec="name@ex[]ample.com")
|
||||
|
||||
def test_empty_group(self):
|
||||
g = Group('foo')
|
||||
self.assertEqual(g.display_name, 'foo')
|
||||
self.assertEqual(g.addresses, tuple())
|
||||
self.assertEqual(str(g), 'foo:;')
|
||||
|
||||
def test_empty_group_list(self):
|
||||
g = Group('foo', addresses=[])
|
||||
self.assertEqual(g.display_name, 'foo')
|
||||
self.assertEqual(g.addresses, tuple())
|
||||
self.assertEqual(str(g), 'foo:;')
|
||||
|
||||
def test_null_group(self):
|
||||
g = Group()
|
||||
self.assertIsNone(g.display_name)
|
||||
self.assertEqual(g.addresses, tuple())
|
||||
self.assertEqual(str(g), 'None:;')
|
||||
|
||||
def test_group_with_addresses(self):
|
||||
addrs = [Address('b', 'b', 'c'), Address('a', 'b','c')]
|
||||
g = Group('foo', addrs)
|
||||
self.assertEqual(g.display_name, 'foo')
|
||||
self.assertEqual(g.addresses, tuple(addrs))
|
||||
self.assertEqual(str(g), 'foo: b <b@c>, a <b@c>;')
|
||||
|
||||
def test_group_with_addresses_no_display_name(self):
|
||||
addrs = [Address('b', 'b', 'c'), Address('a', 'b','c')]
|
||||
g = Group(addresses=addrs)
|
||||
self.assertIsNone(g.display_name)
|
||||
self.assertEqual(g.addresses, tuple(addrs))
|
||||
self.assertEqual(str(g), 'None: b <b@c>, a <b@c>;')
|
||||
|
||||
def test_group_with_one_address_no_display_name(self):
|
||||
addrs = [Address('b', 'b', 'c')]
|
||||
g = Group(addresses=addrs)
|
||||
self.assertIsNone(g.display_name)
|
||||
self.assertEqual(g.addresses, tuple(addrs))
|
||||
self.assertEqual(str(g), 'b <b@c>')
|
||||
|
||||
def test_display_name_quoting(self):
|
||||
g = Group('foo.bar')
|
||||
self.assertEqual(g.display_name, 'foo.bar')
|
||||
self.assertEqual(g.addresses, tuple())
|
||||
self.assertEqual(str(g), '"foo.bar":;')
|
||||
|
||||
def test_display_name_blanks_not_quoted(self):
|
||||
g = Group('foo bar')
|
||||
self.assertEqual(g.display_name, 'foo bar')
|
||||
self.assertEqual(g.addresses, tuple())
|
||||
self.assertEqual(str(g), 'foo bar:;')
|
||||
|
||||
|
||||
class TestFolding(TestHeaderBase):
|
||||
|
||||
def test_short_unstructured(self):
|
||||
h = self.make_header('subject', 'this is a test')
|
||||
self.assertEqual(h.fold(policy=self.policy),
|
||||
'subject: this is a test\n')
|
||||
|
||||
def test_long_unstructured(self):
|
||||
h = self.make_header('Subject', 'This is a long header '
|
||||
'line that will need to be folded into two lines '
|
||||
'and will demonstrate basic folding')
|
||||
self.assertEqual(h.fold(policy=self.policy),
|
||||
'Subject: This is a long header line that will '
|
||||
'need to be folded into two lines\n'
|
||||
' and will demonstrate basic folding\n')
|
||||
|
||||
def test_unstructured_short_max_line_length(self):
|
||||
h = self.make_header('Subject', 'this is a short header '
|
||||
'that will be folded anyway')
|
||||
self.assertEqual(
|
||||
h.fold(policy=policy.default.clone(max_line_length=20)),
|
||||
textwrap.dedent("""\
|
||||
Subject: this is a
|
||||
short header that
|
||||
will be folded
|
||||
anyway
|
||||
"""))
|
||||
|
||||
def test_fold_unstructured_single_word(self):
|
||||
h = self.make_header('Subject', 'test')
|
||||
self.assertEqual(h.fold(policy=self.policy), 'Subject: test\n')
|
||||
|
||||
def test_fold_unstructured_short(self):
|
||||
h = self.make_header('Subject', 'test test test')
|
||||
self.assertEqual(h.fold(policy=self.policy),
|
||||
'Subject: test test test\n')
|
||||
|
||||
def test_fold_unstructured_with_overlong_word(self):
|
||||
h = self.make_header('Subject', 'thisisaverylonglineconsistingofa'
|
||||
'singlewordthatwontfit')
|
||||
self.assertEqual(
|
||||
h.fold(policy=policy.default.clone(max_line_length=20)),
|
||||
'Subject: thisisaverylonglineconsistingofasinglewordthatwontfit\n')
|
||||
|
||||
def test_fold_unstructured_with_two_overlong_words(self):
|
||||
h = self.make_header('Subject', 'thisisaverylonglineconsistingofa'
|
||||
'singlewordthatwontfit plusanotherverylongwordthatwontfit')
|
||||
self.assertEqual(
|
||||
h.fold(policy=policy.default.clone(max_line_length=20)),
|
||||
'Subject: thisisaverylonglineconsistingofasinglewordthatwontfit\n'
|
||||
' plusanotherverylongwordthatwontfit\n')
|
||||
|
||||
def test_fold_unstructured_with_slightly_long_word(self):
|
||||
h = self.make_header('Subject', 'thislongwordislessthanmaxlinelen')
|
||||
self.assertEqual(
|
||||
h.fold(policy=policy.default.clone(max_line_length=35)),
|
||||
'Subject:\n thislongwordislessthanmaxlinelen\n')
|
||||
|
||||
def test_fold_unstructured_with_commas(self):
|
||||
# The old wrapper would fold this at the commas.
|
||||
h = self.make_header('Subject', "This header is intended to "
|
||||
"demonstrate, in a fairly susinct way, that we now do "
|
||||
"not give a , special treatment in unstructured headers.")
|
||||
self.assertEqual(
|
||||
h.fold(policy=policy.default.clone(max_line_length=60)),
|
||||
textwrap.dedent("""\
|
||||
Subject: This header is intended to demonstrate, in a fairly
|
||||
susinct way, that we now do not give a , special treatment
|
||||
in unstructured headers.
|
||||
"""))
|
||||
|
||||
def test_fold_address_list(self):
|
||||
h = self.make_header('To', '"Theodore H. Perfect" <yes@man.com>, '
|
||||
'"My address is very long because my name is long" <foo@bar.com>, '
|
||||
'"Only A. Friend" <no@yes.com>')
|
||||
self.assertEqual(h.fold(policy=self.policy), textwrap.dedent("""\
|
||||
To: "Theodore H. Perfect" <yes@man.com>,
|
||||
"My address is very long because my name is long" <foo@bar.com>,
|
||||
"Only A. Friend" <no@yes.com>
|
||||
"""))
|
||||
|
||||
def test_fold_date_header(self):
|
||||
h = self.make_header('Date', 'Sat, 2 Feb 2002 17:00:06 -0800')
|
||||
self.assertEqual(h.fold(policy=self.policy),
|
||||
'Date: Sat, 02 Feb 2002 17:00:06 -0800\n')
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
unittest.main()
|
|
@ -6,14 +6,16 @@ from email.generator import Generator, BytesGenerator
|
|||
from email import policy
|
||||
from test.test_email import TestEmailBase
|
||||
|
||||
# XXX: move generator tests from test_email into here at some point.
|
||||
|
||||
class TestGeneratorBase:
|
||||
|
||||
class TestGeneratorBase():
|
||||
policy = policy.default
|
||||
|
||||
policy = policy.compat32
|
||||
def msgmaker(self, msg, policy=None):
|
||||
policy = self.policy if policy is None else policy
|
||||
return self.msgfunc(msg, policy=policy)
|
||||
|
||||
long_subject = {
|
||||
refold_long_expected = {
|
||||
0: textwrap.dedent("""\
|
||||
To: whom_it_may_concern@example.com
|
||||
From: nobody_you_want_to_know@example.com
|
||||
|
@ -23,33 +25,32 @@ class TestGeneratorBase():
|
|||
|
||||
None
|
||||
"""),
|
||||
# From is wrapped because wrapped it fits in 40.
|
||||
40: textwrap.dedent("""\
|
||||
To: whom_it_may_concern@example.com
|
||||
From:\x20
|
||||
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.
|
||||
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
|
||||
"""),
|
||||
# Neither to nor from fit even if put on a new line,
|
||||
# so we leave them sticking out on the first line.
|
||||
20: textwrap.dedent("""\
|
||||
To:\x20
|
||||
whom_it_may_concern@example.com
|
||||
From:\x20
|
||||
nobody_you_want_to_know@example.com
|
||||
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
|
||||
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.
|
||||
|
@ -57,65 +58,90 @@ class TestGeneratorBase():
|
|||
None
|
||||
"""),
|
||||
}
|
||||
long_subject[100] = long_subject[0]
|
||||
refold_long_expected[100] = refold_long_expected[0]
|
||||
|
||||
def maxheaderlen_parameter_test(self, n):
|
||||
msg = self.msgmaker(self.typ(self.long_subject[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")
|
||||
|
||||
def _test_maxheaderlen_parameter(self, n):
|
||||
msg = self.msgmaker(self.typ(self.refold_long_expected[0]))
|
||||
s = self.ioclass()
|
||||
g = self.genclass(s, maxheaderlen=n)
|
||||
g = self.genclass(s, maxheaderlen=n, policy=self.policy)
|
||||
g.flatten(msg)
|
||||
self.assertEqual(s.getvalue(), self.typ(self.long_subject[n]))
|
||||
self.assertEqual(s.getvalue(), self.typ(self.refold_long_expected[n]))
|
||||
|
||||
def test_maxheaderlen_parameter_0(self):
|
||||
self.maxheaderlen_parameter_test(0)
|
||||
for n in refold_long_expected:
|
||||
locals()['test_maxheaderlen_parameter_' + str(n)] = (
|
||||
lambda self, n=n:
|
||||
self._test_maxheaderlen_parameter(n))
|
||||
|
||||
def test_maxheaderlen_parameter_100(self):
|
||||
self.maxheaderlen_parameter_test(100)
|
||||
|
||||
def test_maxheaderlen_parameter_40(self):
|
||||
self.maxheaderlen_parameter_test(40)
|
||||
|
||||
def test_maxheaderlen_parameter_20(self):
|
||||
self.maxheaderlen_parameter_test(20)
|
||||
|
||||
def maxheaderlen_policy_test(self, n):
|
||||
msg = self.msgmaker(self.typ(self.long_subject[0]))
|
||||
def _test_max_line_length_policy(self, n):
|
||||
msg = self.msgmaker(self.typ(self.refold_long_expected[0]))
|
||||
s = self.ioclass()
|
||||
g = self.genclass(s, policy=policy.default.clone(max_line_length=n))
|
||||
g = self.genclass(s, policy=self.policy.clone(max_line_length=n))
|
||||
g.flatten(msg)
|
||||
self.assertEqual(s.getvalue(), self.typ(self.long_subject[n]))
|
||||
self.assertEqual(s.getvalue(), self.typ(self.refold_long_expected[n]))
|
||||
|
||||
def test_maxheaderlen_policy_0(self):
|
||||
self.maxheaderlen_policy_test(0)
|
||||
for n in refold_long_expected:
|
||||
locals()['test_max_line_length_policy' + str(n)] = (
|
||||
lambda self, n=n:
|
||||
self._test_max_line_length_policy(n))
|
||||
|
||||
def test_maxheaderlen_policy_100(self):
|
||||
self.maxheaderlen_policy_test(100)
|
||||
|
||||
def test_maxheaderlen_policy_40(self):
|
||||
self.maxheaderlen_policy_test(40)
|
||||
|
||||
def test_maxheaderlen_policy_20(self):
|
||||
self.maxheaderlen_policy_test(20)
|
||||
|
||||
def maxheaderlen_parm_overrides_policy_test(self, n):
|
||||
msg = self.msgmaker(self.typ(self.long_subject[0]))
|
||||
def _test_maxheaderlen_parm_overrides_policy(self, n):
|
||||
msg = self.msgmaker(self.typ(self.refold_long_expected[0]))
|
||||
s = self.ioclass()
|
||||
g = self.genclass(s, maxheaderlen=n,
|
||||
policy=policy.default.clone(max_line_length=10))
|
||||
policy=self.policy.clone(max_line_length=10))
|
||||
g.flatten(msg)
|
||||
self.assertEqual(s.getvalue(), self.typ(self.long_subject[n]))
|
||||
self.assertEqual(s.getvalue(), self.typ(self.refold_long_expected[n]))
|
||||
|
||||
def test_maxheaderlen_parm_overrides_policy_0(self):
|
||||
self.maxheaderlen_parm_overrides_policy_test(0)
|
||||
for n in refold_long_expected:
|
||||
locals()['test_maxheaderlen_parm_overrides_policy' + str(n)] = (
|
||||
lambda self, n=n:
|
||||
self._test_maxheaderlen_parm_overrides_policy(n))
|
||||
|
||||
def test_maxheaderlen_parm_overrides_policy_100(self):
|
||||
self.maxheaderlen_parm_overrides_policy_test(100)
|
||||
def _test_refold_none_does_not_fold(self, n):
|
||||
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]))
|
||||
|
||||
def test_maxheaderlen_parm_overrides_policy_40(self):
|
||||
self.maxheaderlen_parm_overrides_policy_test(40)
|
||||
for n in refold_long_expected:
|
||||
locals()['test_refold_none_does_not_fold' + str(n)] = (
|
||||
lambda self, n=n:
|
||||
self._test_refold_none_does_not_fold(n))
|
||||
|
||||
def test_maxheaderlen_parm_overrides_policy_20(self):
|
||||
self.maxheaderlen_parm_overrides_policy_test(20)
|
||||
def _test_refold_all(self, n):
|
||||
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]))
|
||||
|
||||
for n in refold_long_expected:
|
||||
locals()['test_refold_all' + str(n)] = (
|
||||
lambda self, n=n:
|
||||
self._test_refold_all(n))
|
||||
|
||||
def test_crlf_control_via_policy(self):
|
||||
source = "Subject: test\r\n\r\ntest body\r\n"
|
||||
|
@ -138,30 +164,24 @@ class TestGeneratorBase():
|
|||
|
||||
class TestGenerator(TestGeneratorBase, TestEmailBase):
|
||||
|
||||
msgfunc = staticmethod(message_from_string)
|
||||
genclass = Generator
|
||||
ioclass = io.StringIO
|
||||
typ = str
|
||||
|
||||
def msgmaker(self, msg, policy=None):
|
||||
policy = self.policy if policy is None else policy
|
||||
return message_from_string(msg, policy=policy)
|
||||
|
||||
|
||||
class TestBytesGenerator(TestGeneratorBase, TestEmailBase):
|
||||
|
||||
msgfunc = staticmethod(message_from_bytes)
|
||||
genclass = BytesGenerator
|
||||
ioclass = io.BytesIO
|
||||
typ = lambda self, x: x.encode('ascii')
|
||||
|
||||
def msgmaker(self, msg, policy=None):
|
||||
policy = self.policy if policy is None else policy
|
||||
return message_from_bytes(msg, policy=policy)
|
||||
|
||||
def test_cte_type_7bit_handles_unknown_8bit(self):
|
||||
source = ("Subject: Maintenant je vous présente mon "
|
||||
"collègue\n\n").encode('utf-8')
|
||||
expected = ('Subject: =?unknown-8bit?q?Maintenant_je_vous_pr=C3=A9sente_mon_'
|
||||
'coll=C3=A8gue?=\n\n').encode('ascii')
|
||||
expected = ('Subject: Maintenant je vous =?unknown-8bit?q?'
|
||||
'pr=C3=A9sente_mon_coll=C3=A8gue?=\n\n').encode('ascii')
|
||||
msg = message_from_bytes(source)
|
||||
s = io.BytesIO()
|
||||
g = BytesGenerator(s, policy=self.policy.clone(cte_type='7bit'))
|
||||
|
|
|
@ -0,0 +1,57 @@
|
|||
import unittest
|
||||
import textwrap
|
||||
import copy
|
||||
import pickle
|
||||
from email import policy
|
||||
from email import message_from_string
|
||||
from email._headerregistry import HeaderRegistry
|
||||
from test.test_email import TestEmailBase
|
||||
|
||||
class TestPickleCopyHeader(TestEmailBase):
|
||||
|
||||
unstructured = HeaderRegistry()('subject', 'this is a test')
|
||||
|
||||
def test_deepcopy_unstructured(self):
|
||||
h = copy.deepcopy(self.unstructured)
|
||||
self.assertEqual(str(h), str(self.unstructured))
|
||||
|
||||
def test_pickle_unstructured(self):
|
||||
p = pickle.dumps(self.unstructured)
|
||||
h = pickle.loads(p)
|
||||
self.assertEqual(str(h), str(self.unstructured))
|
||||
|
||||
address = HeaderRegistry()('from', 'frodo@mordor.net')
|
||||
|
||||
def test_deepcopy_address(self):
|
||||
h = copy.deepcopy(self.address)
|
||||
self.assertEqual(str(h), str(self.address))
|
||||
|
||||
def test_pickle_address(self):
|
||||
p = pickle.dumps(self.address)
|
||||
h = pickle.loads(p)
|
||||
self.assertEqual(str(h), str(self.address))
|
||||
|
||||
|
||||
class TestPickleCopyMessage(TestEmailBase):
|
||||
|
||||
testmsg = message_from_string(textwrap.dedent("""\
|
||||
From: frodo@mordor.net
|
||||
To: bilbo@underhill.org
|
||||
Subject: help
|
||||
|
||||
I think I forgot the ring.
|
||||
"""), policy=policy.default)
|
||||
|
||||
def test_deepcopy(self):
|
||||
msg2 = copy.deepcopy(self.testmsg)
|
||||
self.assertEqual(msg2.as_string(), self.testmsg.as_string())
|
||||
|
||||
def test_pickle(self):
|
||||
p = pickle.dumps(self.testmsg)
|
||||
msg2 = pickle.loads(p)
|
||||
self.assertEqual(msg2.as_string(), self.testmsg.as_string())
|
||||
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
unittest.main()
|
|
@ -5,49 +5,70 @@ import unittest
|
|||
import email.policy
|
||||
import email.parser
|
||||
import email.generator
|
||||
from email import _headerregistry
|
||||
|
||||
def make_defaults(base_defaults, differences):
|
||||
defaults = base_defaults.copy()
|
||||
defaults.update(differences)
|
||||
return defaults
|
||||
|
||||
class PolicyAPITests(unittest.TestCase):
|
||||
|
||||
longMessage = True
|
||||
|
||||
# These default values are the ones set on email.policy.default.
|
||||
# If any of these defaults change, the docs must be updated.
|
||||
policy_defaults = {
|
||||
# Base default values.
|
||||
compat32_defaults = {
|
||||
'max_line_length': 78,
|
||||
'linesep': '\n',
|
||||
'cte_type': '8bit',
|
||||
'raise_on_defect': False,
|
||||
}
|
||||
# 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({
|
||||
'raise_on_defect': False,
|
||||
'header_factory': email.policy.EmailPolicy.header_factory,
|
||||
'refold_source': 'long',
|
||||
})
|
||||
|
||||
# For each policy under test, we give here the values of the attributes
|
||||
# that are different from the defaults for that policy.
|
||||
# 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()
|
||||
policies = {
|
||||
email.policy.Compat32(): {},
|
||||
email.policy.compat32: {},
|
||||
email.policy.default: {},
|
||||
email.policy.SMTP: {'linesep': '\r\n'},
|
||||
email.policy.HTTP: {'linesep': '\r\n', 'max_line_length': None},
|
||||
email.policy.strict: {'raise_on_defect': True},
|
||||
email.policy.compat32: make_defaults(compat32_defaults, {}),
|
||||
email.policy.default: make_defaults(policy_defaults, {}),
|
||||
email.policy.SMTP: make_defaults(policy_defaults,
|
||||
{'linesep': '\r\n'}),
|
||||
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, {}),
|
||||
}
|
||||
# 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
|
||||
|
||||
def test_defaults(self):
|
||||
for policy, changed_defaults in self.policies.items():
|
||||
expected = self.policy_defaults.copy()
|
||||
expected.update(changed_defaults)
|
||||
for policy, expected in self.policies.items():
|
||||
for attr, value in expected.items():
|
||||
self.assertEqual(getattr(policy, attr), value,
|
||||
("change {} docs/docstrings if defaults have "
|
||||
"changed").format(policy))
|
||||
|
||||
def test_all_attributes_covered(self):
|
||||
for attr in dir(email.policy.default):
|
||||
if (attr.startswith('_') or
|
||||
isinstance(getattr(email.policy.Policy, attr),
|
||||
types.FunctionType)):
|
||||
continue
|
||||
else:
|
||||
self.assertIn(attr, self.policy_defaults,
|
||||
"{} is not fully tested".format(attr))
|
||||
for policy, expected in self.policies.items():
|
||||
for attr in dir(policy):
|
||||
if (attr.startswith('_') or
|
||||
isinstance(getattr(email.policy.EmailPolicy, attr),
|
||||
types.FunctionType)):
|
||||
continue
|
||||
else:
|
||||
self.assertIn(attr, expected,
|
||||
"{} is not fully tested".format(attr))
|
||||
|
||||
def test_abc(self):
|
||||
with self.assertRaises(TypeError) as cm:
|
||||
|
@ -62,18 +83,20 @@ class PolicyAPITests(unittest.TestCase):
|
|||
self.assertIn(method, msg)
|
||||
|
||||
def test_policy_is_immutable(self):
|
||||
for policy in self.policies:
|
||||
for attr in self.policy_defaults:
|
||||
for policy, defaults in self.policies.items():
|
||||
for attr in defaults:
|
||||
with self.assertRaisesRegex(AttributeError, attr+".*read-only"):
|
||||
setattr(policy, attr, None)
|
||||
with self.assertRaisesRegex(AttributeError, 'no attribute.*foo'):
|
||||
policy.foo = None
|
||||
|
||||
def test_set_policy_attrs_when_calledl(self):
|
||||
testattrdict = { attr: None for attr in self.policy_defaults }
|
||||
for policyclass in self.policies:
|
||||
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}
|
||||
policy = policyclass.clone(**testattrdict)
|
||||
for attr in self.policy_defaults:
|
||||
for attr in defaults:
|
||||
self.assertIsNone(getattr(policy, attr))
|
||||
|
||||
def test_reject_non_policy_keyword_when_called(self):
|
||||
|
@ -105,7 +128,7 @@ class PolicyAPITests(unittest.TestCase):
|
|||
self.defects = []
|
||||
obj = Dummy()
|
||||
defect = object()
|
||||
policy = email.policy.Compat32()
|
||||
policy = email.policy.EmailPolicy()
|
||||
policy.register_defect(obj, defect)
|
||||
self.assertEqual(obj.defects, [defect])
|
||||
defect2 = object()
|
||||
|
@ -134,7 +157,7 @@ class PolicyAPITests(unittest.TestCase):
|
|||
email.policy.default.handle_defect(foo, defect2)
|
||||
self.assertEqual(foo.defects, [defect1, defect2])
|
||||
|
||||
class MyPolicy(email.policy.Compat32):
|
||||
class MyPolicy(email.policy.EmailPolicy):
|
||||
defects = None
|
||||
def __init__(self, *args, **kw):
|
||||
super().__init__(*args, defects=[], **kw)
|
||||
|
@ -159,6 +182,49 @@ class PolicyAPITests(unittest.TestCase):
|
|||
self.assertEqual(my_policy.defects, [defect1, defect2])
|
||||
self.assertEqual(foo.defects, [])
|
||||
|
||||
def test_default_header_factory(self):
|
||||
h = email.policy.default.header_factory('Test', 'test')
|
||||
self.assertEqual(h.name, 'Test')
|
||||
self.assertIsInstance(h, _headerregistry.UnstructuredHeader)
|
||||
self.assertIsInstance(h, _headerregistry.BaseHeader)
|
||||
|
||||
class Foo:
|
||||
parse = _headerregistry.UnstructuredHeader.parse
|
||||
|
||||
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)
|
||||
self.assertNotIsInstance(h, _headerregistry.UnstructuredHeader)
|
||||
h = policy2.header_factory('foo', 'test')
|
||||
self.assertNotIsInstance(h, self.Foo)
|
||||
self.assertIsInstance(h, _headerregistry.UnstructuredHeader)
|
||||
|
||||
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})
|
||||
|
||||
# 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).
|
||||
|
|
Loading…
Reference in New Issue