mail.py
527 lines
| 16.8 KiB
| text/x-python
|
PythonLexer
/ mercurial / mail.py
Matt Mackall
|
r2889 | # mail.py - mail sending bits for mercurial | ||
# | ||||
Raphaël Gomès
|
r47575 | # Copyright 2006 Olivia Mackall <olivia@selenic.com> | ||
Matt Mackall
|
r2889 | # | ||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Matt Mackall
|
r2889 | |||
Gregory Szorc
|
r25957 | |||
Augie Fackler
|
r19790 | import email | ||
Gábor Stefanik
|
r30089 | import email.charset | ||
Denis Laxalde
|
r43426 | import email.generator | ||
Pulkit Goyal
|
r30072 | import email.header | ||
Igor Ippolitov
|
r34311 | import email.message | ||
Yuya Nishihara
|
r38354 | import email.parser | ||
import io | ||||
Gregory Szorc
|
r25957 | import os | ||
import smtplib | ||||
import socket | ||||
import time | ||||
r52178 | from typing import ( | |||
Any, | ||||
List, | ||||
Tuple, | ||||
Union, | ||||
) | ||||
Gregory Szorc
|
r25957 | from .i18n import _ | ||
Gregory Szorc
|
r43359 | from .pycompat import ( | ||
open, | ||||
) | ||||
Gregory Szorc
|
r25957 | from . import ( | ||
encoding, | ||||
Pierre-Yves David
|
r26587 | error, | ||
Gregory Szorc
|
r36064 | pycompat, | ||
Gregory Szorc
|
r25957 | sslutil, | ||
util, | ||||
) | ||||
Yuya Nishihara
|
r37102 | from .utils import ( | ||
Yuya Nishihara
|
r37138 | procutil, | ||
Yuya Nishihara
|
r37102 | stringutil, | ||
r47669 | urlutil, | |||
Yuya Nishihara
|
r37102 | ) | ||
Matt Mackall
|
r2889 | |||
Denis Laxalde
|
r44024 | |||
r52178 | # keep pyflakes happy | |||
assert [ | ||||
Any, | ||||
List, | ||||
Tuple, | ||||
Union, | ||||
] | ||||
Denis Laxalde
|
r44024 | |||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r18885 | class STARTTLS(smtplib.SMTP): | ||
Augie Fackler
|
r46554 | """Derived class to verify the peer certificate for STARTTLS. | ||
FUJIWARA Katsunori
|
r18885 | |||
This class allows to pass any keyword arguments to SSL socket creation. | ||||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r29251 | def __init__(self, ui, host=None, **kwargs): | ||
FUJIWARA Katsunori
|
r18885 | smtplib.SMTP.__init__(self, **kwargs) | ||
Gregory Szorc
|
r29248 | self._ui = ui | ||
timeless
|
r28935 | self._host = host | ||
FUJIWARA Katsunori
|
r18885 | |||
Matt Harbison
|
r51171 | def starttls(self, keyfile=None, certfile=None, context=None): | ||
Denis Laxalde
|
r43439 | if not self.has_extn("starttls"): | ||
Matt Harbison
|
r51172 | msg = "STARTTLS extension not supported by server" | ||
FUJIWARA Katsunori
|
r18885 | raise smtplib.SMTPException(msg) | ||
Denis Laxalde
|
r43440 | (resp, reply) = self.docmd("STARTTLS") | ||
FUJIWARA Katsunori
|
r18885 | if resp == 220: | ||
Augie Fackler
|
r43346 | self.sock = sslutil.wrapsocket( | ||
self.sock, | ||||
keyfile, | ||||
certfile, | ||||
ui=self._ui, | ||||
serverhostname=self._host, | ||||
) | ||||
Denis Laxalde
|
r43441 | self.file = self.sock.makefile("rb") | ||
FUJIWARA Katsunori
|
r18885 | self.helo_resp = None | ||
self.ehlo_resp = None | ||||
self.esmtp_features = {} | ||||
self.does_esmtp = 0 | ||||
return (resp, reply) | ||||
Augie Fackler
|
r43346 | |||
timeless@mozdev.org
|
r26673 | class SMTPS(smtplib.SMTP): | ||
Augie Fackler
|
r46554 | """Derived class to verify the peer certificate for SMTPS. | ||
FUJIWARA Katsunori
|
r18886 | |||
timeless@mozdev.org
|
r26673 | This class allows to pass any keyword arguments to SSL socket creation. | ||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r43346 | |||
def __init__(self, ui, keyfile=None, certfile=None, host=None, **kwargs): | ||||
timeless@mozdev.org
|
r26673 | self.keyfile = keyfile | ||
self.certfile = certfile | ||||
smtplib.SMTP.__init__(self, **kwargs) | ||||
timeless
|
r28935 | self._host = host | ||
timeless@mozdev.org
|
r26673 | self.default_port = smtplib.SMTP_SSL_PORT | ||
Gregory Szorc
|
r29248 | self._ui = ui | ||
timeless@mozdev.org
|
r26673 | |||
def _get_socket(self, host, port, timeout): | ||||
if self.debuglevel > 0: | ||||
Augie Fackler
|
r43347 | self._ui.debug(b'connect: %r\n' % ((host, port),)) | ||
timeless@mozdev.org
|
r26673 | new_socket = socket.create_connection((host, port), timeout) | ||
Augie Fackler
|
r43346 | new_socket = sslutil.wrapsocket( | ||
new_socket, | ||||
self.keyfile, | ||||
self.certfile, | ||||
ui=self._ui, | ||||
serverhostname=self._host, | ||||
) | ||||
Augie Fackler
|
r43906 | self.file = new_socket.makefile('rb') | ||
timeless@mozdev.org
|
r26673 | return new_socket | ||
FUJIWARA Katsunori
|
r18886 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r39060 | def _pyhastls(): | ||
Denis Laxalde
|
r44024 | # type: () -> bool | ||
Augie Fackler
|
r39060 | """Returns true iff Python has TLS support, false otherwise.""" | ||
try: | ||||
import ssl | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r39060 | getattr(ssl, 'HAS_TLS', False) | ||
return True | ||||
except ImportError: | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r2889 | def _smtp(ui): | ||
Matt Mackall
|
r5973 | '''build an smtp connection and return a function to send mail''' | ||
Augie Fackler
|
r43347 | local_hostname = ui.config(b'smtp', b'local_hostname') | ||
tls = ui.config(b'smtp', b'tls') | ||||
Zhigang Wang
|
r13201 | # backward compatible: when tls = true, we use starttls. | ||
Augie Fackler
|
r43347 | starttls = tls == b'starttls' or stringutil.parsebool(tls) | ||
smtps = tls == b'smtps' | ||||
Augie Fackler
|
r39060 | if (starttls or smtps) and not _pyhastls(): | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b"can't use TLS: Python SSL support not installed")) | ||
mailhost = ui.config(b'smtp', b'host') | ||||
Matt Mackall
|
r2889 | if not mailhost: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b'smtp.host not configured - cannot send mail')) | ||
FUJIWARA Katsunori
|
r18888 | if smtps: | ||
Augie Fackler
|
r43347 | ui.note(_(b'(using smtps)\n')) | ||
Gregory Szorc
|
r29251 | s = SMTPS(ui, local_hostname=local_hostname, host=mailhost) | ||
FUJIWARA Katsunori
|
r18888 | elif starttls: | ||
Gregory Szorc
|
r29251 | s = STARTTLS(ui, local_hostname=local_hostname, host=mailhost) | ||
FUJIWARA Katsunori
|
r18888 | else: | ||
s = smtplib.SMTP(local_hostname=local_hostname) | ||||
FUJIWARA Katsunori
|
r19050 | if smtps: | ||
defaultport = 465 | ||||
else: | ||||
defaultport = 25 | ||||
r47669 | mailport = urlutil.getport(ui.config(b'smtp', b'port', defaultport)) | |||
Augie Fackler
|
r43347 | ui.note(_(b'sending mail: smtp host %s, port %d\n') % (mailhost, mailport)) | ||
Matt Mackall
|
r2889 | s.connect(host=mailhost, port=mailport) | ||
Zhigang Wang
|
r13201 | if starttls: | ||
Augie Fackler
|
r43347 | ui.note(_(b'(using starttls)\n')) | ||
Matt Mackall
|
r2889 | s.ehlo() | ||
s.starttls() | ||||
s.ehlo() | ||||
Gregory Szorc
|
r29285 | if starttls or smtps: | ||
Augie Fackler
|
r43347 | ui.note(_(b'(verifying remote certificate)\n')) | ||
Gregory Szorc
|
r29285 | sslutil.validatesocket(s.sock) | ||
Matt Harbison
|
r47753 | |||
try: | ||||
_smtp_login(ui, s, mailhost, mailport) | ||||
except smtplib.SMTPException as inst: | ||||
raise error.Abort(stringutil.forcebytestr(inst)) | ||||
def send(sender, recipients, msg): | ||||
try: | ||||
return s.sendmail(sender, recipients, msg) | ||||
except smtplib.SMTPRecipientsRefused as inst: | ||||
recipients = [r[1] for r in inst.recipients.values()] | ||||
raise error.Abort(b'\n' + b'\n'.join(recipients)) | ||||
except smtplib.SMTPException as inst: | ||||
Matt Harbison
|
r47754 | raise error.Abort(stringutil.forcebytestr(inst)) | ||
Matt Harbison
|
r47753 | |||
return send | ||||
def _smtp_login(ui, smtp, mailhost, mailport): | ||||
"""A hook for the keyring extension to perform the actual SMTP login. | ||||
An already connected SMTP object of the proper type is provided, based on | ||||
the current configuration. The host and port to which the connection was | ||||
established are provided for accessibility, since the SMTP object doesn't | ||||
provide an accessor. ``smtplib.SMTPException`` is raised on error. | ||||
""" | ||||
Augie Fackler
|
r43347 | username = ui.config(b'smtp', b'username') | ||
password = ui.config(b'smtp', b'password') | ||||
Denis Laxalde
|
r43442 | if username: | ||
if password: | ||||
password = encoding.strfromlocal(password) | ||||
else: | ||||
password = ui.getpass() | ||||
Matt Harbison
|
r46541 | if password is not None: | ||
password = encoding.strfromlocal(password) | ||||
Matt Mackall
|
r2889 | if username and password: | ||
Augie Fackler
|
r43347 | ui.note(_(b'(authenticating to mail server as %s)\n') % username) | ||
Denis Laxalde
|
r43442 | username = encoding.strfromlocal(username) | ||
Matt Harbison
|
r47753 | smtp.login(username, password) | ||
Bryan O'Sullivan
|
r5947 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r5973 | def _sendmail(ui, sender, recipients, msg): | ||
'''send mail using sendmail.''' | ||||
Augie Fackler
|
r43347 | program = ui.config(b'email', b'method') | ||
Yuya Nishihara
|
r43366 | |||
def stremail(x): | ||||
return procutil.shellquote(stringutil.email(encoding.strtolocal(x))) | ||||
Augie Fackler
|
r43347 | cmdline = b'%s -f %s %s' % ( | ||
Augie Fackler
|
r43346 | program, | ||
stremail(sender), | ||||
Augie Fackler
|
r43347 | b' '.join(map(stremail, recipients)), | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | ui.note(_(b'sending mail: %s\n') % cmdline) | ||
fp = procutil.popen(cmdline, b'wb') | ||||
Yuya Nishihara
|
r37476 | fp.write(util.tonativeeol(msg)) | ||
Matt Mackall
|
r5973 | ret = fp.close() | ||
if ret: | ||||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | b'%s %s' | ||
Augie Fackler
|
r43346 | % ( | ||
Julien Cristau
|
r44277 | os.path.basename(procutil.shellsplit(program)[0]), | ||
Augie Fackler
|
r43346 | procutil.explainexit(ret), | ||
) | ||||
) | ||||
Matt Mackall
|
r2889 | |||
Mads Kiilerich
|
r15560 | def _mbox(mbox, sender, recipients, msg): | ||
'''write mails to mbox''' | ||||
Matt Harbison
|
r47760 | # TODO: use python mbox library for proper locking | ||
Matt Harbison
|
r47744 | with open(mbox, b'ab+') as fp: | ||
# Should be time.asctime(), but Windows prints 2-characters day | ||||
# of month instead of one. Make them print the same thing. | ||||
date = time.strftime('%a %b %d %H:%M:%S %Y', time.localtime()) | ||||
fp.write( | ||||
b'From %s %s\n' | ||||
% (encoding.strtolocal(sender), encoding.strtolocal(date)) | ||||
) | ||||
fp.write(msg) | ||||
fp.write(b'\n\n') | ||||
Mads Kiilerich
|
r15560 | |||
Augie Fackler
|
r43346 | |||
Mads Kiilerich
|
r15560 | def connect(ui, mbox=None): | ||
Augie Fackler
|
r46554 | """make a mail connection. return a function to send mail. | ||
call as sendmail(sender, list-of-recipients, msg).""" | ||||
Mads Kiilerich
|
r15560 | if mbox: | ||
Augie Fackler
|
r43347 | open(mbox, b'wb').close() | ||
Mads Kiilerich
|
r15560 | return lambda s, r, m: _mbox(mbox, s, r, m) | ||
Augie Fackler
|
r43347 | if ui.config(b'email', b'method') == b'smtp': | ||
Bryan O'Sullivan
|
r5947 | return _smtp(ui) | ||
Matt Mackall
|
r5973 | return lambda s, r, m: _sendmail(ui, s, r, m) | ||
Matt Mackall
|
r2889 | |||
Augie Fackler
|
r43346 | |||
Mads Kiilerich
|
r15561 | def sendmail(ui, sender, recipients, msg, mbox=None): | ||
send = connect(ui, mbox=mbox) | ||||
Matt Mackall
|
r5973 | return send(sender, recipients, msg) | ||
Bryan O'Sullivan
|
r4489 | |||
Augie Fackler
|
r43346 | |||
Bryan O'Sullivan
|
r4489 | def validateconfig(ui): | ||
'''determine if we have enough config data to try sending email.''' | ||||
Augie Fackler
|
r43347 | method = ui.config(b'email', b'method') | ||
if method == b'smtp': | ||||
if not ui.config(b'smtp', b'host'): | ||||
Augie Fackler
|
r43346 | raise error.Abort( | ||
_( | ||||
Augie Fackler
|
r43347 | b'smtp specified as email transport, ' | ||
b'but no smtp host configured' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Bryan O'Sullivan
|
r4489 | else: | ||
Julien Cristau
|
r49927 | command = procutil.shellsplit(method) | ||
command = command[0] if command else b'' | ||||
if not (command and procutil.findexe(command)): | ||||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Julien Cristau
|
r49927 | _(b'%r specified as email transport, but not in PATH') % command | ||
Augie Fackler
|
r43346 | ) | ||
Christian Ebert
|
r7114 | |||
Gábor Stefanik
|
r30089 | def codec2iana(cs): | ||
Denis Laxalde
|
r44025 | # type: (str) -> str | ||
Kyle Lippincott
|
r47856 | ''' ''' | ||
Denis Laxalde
|
r44025 | cs = email.charset.Charset(cs).input_charset.lower() | ||
Gábor Stefanik
|
r30089 | |||
# "latin1" normalizes to "iso8859-1", standard calls for "iso-8859-1" | ||||
Denis Laxalde
|
r44025 | if cs.startswith("iso") and not cs.startswith("iso-"): | ||
return "iso-" + cs[3:] | ||||
Gábor Stefanik
|
r30089 | return cs | ||
Augie Fackler
|
r43346 | |||
Denis Laxalde
|
r44026 | def mimetextpatch(s, subtype='plain', display=False): | ||
# type: (bytes, str, bool) -> email.message.Message | ||||
Augie Fackler
|
r46554 | """Return MIME message suitable for a patch. | ||
Gábor Stefanik
|
r30089 | Charset will be detected by first trying to decode as us-ascii, then utf-8, | ||
and finally the global encodings. If all those fail, fall back to | ||||
ISO-8859-1, an encoding with that allows all byte sequences. | ||||
Augie Fackler
|
r46554 | Transfer encodings will be used if necessary.""" | ||
Rocco Rutte
|
r8332 | |||
Denis Laxalde
|
r44025 | cs = [ | ||
'us-ascii', | ||||
'utf-8', | ||||
pycompat.sysstr(encoding.encoding), | ||||
pycompat.sysstr(encoding.fallbackencoding), | ||||
] | ||||
Gábor Stefanik
|
r30089 | if display: | ||
Denis Laxalde
|
r44025 | cs = ['us-ascii'] | ||
Gábor Stefanik
|
r30089 | for charset in cs: | ||
Rocco Rutte
|
r8332 | try: | ||
Denis Laxalde
|
r44025 | s.decode(charset) | ||
Gábor Stefanik
|
r30089 | return mimetextqp(s, subtype, codec2iana(charset)) | ||
Rocco Rutte
|
r8332 | except UnicodeDecodeError: | ||
Gábor Stefanik
|
r30089 | pass | ||
Rocco Rutte
|
r8332 | |||
Denis Laxalde
|
r44025 | return mimetextqp(s, subtype, "iso-8859-1") | ||
Mads Kiilerich
|
r15562 | |||
Augie Fackler
|
r43346 | |||
Mads Kiilerich
|
r15562 | def mimetextqp(body, subtype, charset): | ||
Denis Laxalde
|
r44026 | # type: (bytes, str, str) -> email.message.Message | ||
Augie Fackler
|
r46554 | """Return MIME message. | ||
Mads Kiilerich
|
r17424 | Quoted-printable transfer encoding will be used if necessary. | ||
Augie Fackler
|
r46554 | """ | ||
Denis Laxalde
|
r44025 | cs = email.charset.Charset(charset) | ||
Igor Ippolitov
|
r34311 | msg = email.message.Message() | ||
Denis Laxalde
|
r44026 | msg.set_type('text/' + subtype) | ||
Igor Ippolitov
|
r34311 | |||
Mads Kiilerich
|
r15562 | for line in body.splitlines(): | ||
if len(line) > 950: | ||||
Igor Ippolitov
|
r34311 | cs.body_encoding = email.charset.QP | ||
Mads Kiilerich
|
r15562 | break | ||
Gregory Szorc
|
r41450 | # On Python 2, this simply assigns a value. Python 3 inspects | ||
# body and does different things depending on whether it has | ||||
# encode() or decode() attributes. We can get the old behavior | ||||
# if we pass a str and charset is None and we call set_charset(). | ||||
# But we may get into trouble later due to Python attempting to | ||||
# encode/decode using the registered charset (or attempting to | ||||
# use ascii in the absence of a charset). | ||||
Igor Ippolitov
|
r34311 | msg.set_payload(body, cs) | ||
Rocco Rutte
|
r8332 | return msg | ||
Christian Ebert
|
r7191 | |||
Augie Fackler
|
r43346 | |||
Christian Ebert
|
r7114 | def _charsets(ui): | ||
Denis Laxalde
|
r44025 | # type: (Any) -> List[str] | ||
Christian Ebert
|
r7114 | '''Obtains charsets to send mail parts not containing patches.''' | ||
Denis Laxalde
|
r44024 | charsets = [ | ||
Denis Laxalde
|
r44025 | pycompat.sysstr(cs.lower()) | ||
for cs in ui.configlist(b'email', b'charsets') | ||||
] | ||||
Augie Fackler
|
r43346 | fallbacks = [ | ||
Denis Laxalde
|
r44025 | pycompat.sysstr(encoding.fallbackencoding.lower()), | ||
pycompat.sysstr(encoding.encoding.lower()), | ||||
'utf-8', | ||||
] | ||||
Augie Fackler
|
r43346 | for cs in fallbacks: # find unique charsets while keeping order | ||
Christian Ebert
|
r7114 | if cs not in charsets: | ||
charsets.append(cs) | ||||
Denis Laxalde
|
r44025 | return [cs for cs in charsets if not cs.endswith('ascii')] | ||
Christian Ebert
|
r7114 | |||
Augie Fackler
|
r43346 | |||
Christian Ebert
|
r7114 | def _encode(ui, s, charsets): | ||
Denis Laxalde
|
r44025 | # type: (Any, bytes, List[str]) -> Tuple[bytes, str] | ||
Augie Fackler
|
r46554 | """Returns (converted) string, charset tuple. | ||
Christian Ebert
|
r7114 | Finds out best charset by cycling through sendcharsets in descending | ||
Matt Mackall
|
r7948 | order. Tries both encoding and fallbackencoding for input. Only as | ||
Christian Ebert
|
r7114 | last resort send as is in fake ascii. | ||
Augie Fackler
|
r46554 | Caveat: Do not use for mail parts containing patches!""" | ||
Augie Fackler
|
r39058 | sendcharsets = charsets or _charsets(ui) | ||
if not isinstance(s, bytes): | ||||
# We have unicode data, which we need to try and encode to | ||||
# some reasonable-ish encoding. Try the encodings the user | ||||
# wants, and fall back to garbage-in-ascii. | ||||
for ocs in sendcharsets: | ||||
try: | ||||
Denis Laxalde
|
r44025 | return s.encode(ocs), ocs | ||
Augie Fackler
|
r39058 | except UnicodeEncodeError: | ||
pass | ||||
except LookupError: | ||||
Denis Laxalde
|
r44025 | ui.warn( | ||
_(b'ignoring invalid sendcharset: %s\n') | ||||
% pycompat.sysbytes(ocs) | ||||
) | ||||
Augie Fackler
|
r39058 | else: | ||
# Everything failed, ascii-armor what we've got and send it. | ||||
Denis Laxalde
|
r44025 | return s.encode('ascii', 'backslashreplace'), 'us-ascii' | ||
Augie Fackler
|
r39058 | # We have a bytes of unknown encoding. We'll try and guess a valid | ||
# encoding, falling back to pretending we had ascii even though we | ||||
# know that's wrong. | ||||
Christian Ebert
|
r7114 | try: | ||
s.decode('ascii') | ||||
except UnicodeDecodeError: | ||||
Matt Mackall
|
r7948 | for ics in (encoding.encoding, encoding.fallbackencoding): | ||
Denis Laxalde
|
r43638 | ics = pycompat.sysstr(ics) | ||
Christian Ebert
|
r7114 | try: | ||
u = s.decode(ics) | ||||
except UnicodeDecodeError: | ||||
continue | ||||
for ocs in sendcharsets: | ||||
try: | ||||
Denis Laxalde
|
r44025 | return u.encode(ocs), ocs | ||
Christian Ebert
|
r7114 | except UnicodeEncodeError: | ||
pass | ||||
except LookupError: | ||||
Denis Laxalde
|
r44025 | ui.warn( | ||
_(b'ignoring invalid sendcharset: %s\n') | ||||
% pycompat.sysbytes(ocs) | ||||
) | ||||
Christian Ebert
|
r7114 | # if ascii, or all conversion attempts fail, send (broken) ascii | ||
Denis Laxalde
|
r44025 | return s, 'us-ascii' | ||
Christian Ebert
|
r7114 | |||
Augie Fackler
|
r43346 | |||
Christian Ebert
|
r7114 | def headencode(ui, s, charsets=None, display=False): | ||
Denis Laxalde
|
r44025 | # type: (Any, Union[bytes, str], List[str], bool) -> str | ||
Christian Ebert
|
r7114 | '''Returns RFC-2047 compliant header from given string.''' | ||
if not display: | ||||
# split into words? | ||||
s, cs = _encode(ui, s, charsets) | ||||
Denis Laxalde
|
r44025 | return email.header.Header(s, cs).encode() | ||
Denis Laxalde
|
r43975 | return encoding.strfromlocal(s) | ||
Christian Ebert
|
r7114 | |||
Augie Fackler
|
r43346 | |||
Marti Raudsepp
|
r9948 | def _addressencode(ui, name, addr, charsets=None): | ||
Denis Laxalde
|
r44027 | # type: (Any, str, str, List[str]) -> str | ||
addr = encoding.strtolocal(addr) | ||||
Denis Laxalde
|
r43975 | name = headencode(ui, name, charsets) | ||
Christian Ebert
|
r7114 | try: | ||
Augie Fackler
|
r43347 | acc, dom = addr.split(b'@') | ||
Yuya Nishihara
|
r39143 | acc.decode('ascii') | ||
Yuya Nishihara
|
r39144 | dom = dom.decode(pycompat.sysstr(encoding.encoding)).encode('idna') | ||
Augie Fackler
|
r43347 | addr = b'%s@%s' % (acc, dom) | ||
Christian Ebert
|
r7114 | except UnicodeDecodeError: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b'invalid email address: %s') % addr) | ||
Christian Ebert
|
r7114 | except ValueError: | ||
try: | ||||
# too strict? | ||||
Yuya Nishihara
|
r39143 | addr.decode('ascii') | ||
Christian Ebert
|
r7114 | except UnicodeDecodeError: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b'invalid local address: %s') % addr) | ||
Denis Laxalde
|
r43976 | return email.utils.formataddr((name, encoding.strfromlocal(addr))) | ||
Augie Fackler
|
r43346 | |||
Christian Ebert
|
r7114 | |||
Marti Raudsepp
|
r9948 | def addressencode(ui, address, charsets=None, display=False): | ||
Denis Laxalde
|
r44025 | # type: (Any, bytes, List[str], bool) -> str | ||
Marti Raudsepp
|
r9948 | '''Turns address into RFC-2047 compliant header.''' | ||
if display or not address: | ||||
Denis Laxalde
|
r43976 | return encoding.strfromlocal(address or b'') | ||
Augie Fackler
|
r39059 | name, addr = email.utils.parseaddr(encoding.strfromlocal(address)) | ||
Denis Laxalde
|
r44027 | return _addressencode(ui, name, addr, charsets) | ||
Marti Raudsepp
|
r9948 | |||
Augie Fackler
|
r43346 | |||
Marti Raudsepp
|
r9948 | def addrlistencode(ui, addrs, charsets=None, display=False): | ||
Denis Laxalde
|
r44025 | # type: (Any, List[bytes], List[str], bool) -> List[str] | ||
Augie Fackler
|
r46554 | """Turns a list of addresses into a list of RFC-2047 compliant headers. | ||
Marti Raudsepp
|
r9948 | A single element of input list may contain multiple addresses, but output | ||
Augie Fackler
|
r46554 | always has one address per item""" | ||
Denis Laxalde
|
r43977 | straddrs = [] | ||
Augie Fackler
|
r39059 | for a in addrs: | ||
Augie Fackler
|
r43906 | assert isinstance(a, bytes), '%r unexpectedly not a bytestr' % a | ||
Denis Laxalde
|
r43977 | straddrs.append(encoding.strfromlocal(a)) | ||
Marti Raudsepp
|
r9948 | if display: | ||
Denis Laxalde
|
r43977 | return [a.strip() for a in straddrs if a.strip()] | ||
Marti Raudsepp
|
r9948 | |||
result = [] | ||||
Denis Laxalde
|
r43977 | for name, addr in email.utils.getaddresses(straddrs): | ||
Marti Raudsepp
|
r9948 | if name or addr: | ||
Denis Laxalde
|
r44027 | r = _addressencode(ui, name, addr, charsets) | ||
Yuya Nishihara
|
r39142 | result.append(r) | ||
Yuya Nishihara
|
r39141 | return result | ||
Marti Raudsepp
|
r9948 | |||
Augie Fackler
|
r43346 | |||
Christian Ebert
|
r7114 | def mimeencode(ui, s, charsets=None, display=False): | ||
Denis Laxalde
|
r44025 | # type: (Any, bytes, List[str], bool) -> email.message.Message | ||
Augie Fackler
|
r46554 | """creates mime text object, encodes it if needed, and sets | ||
charset and transfer-encoding accordingly.""" | ||||
Denis Laxalde
|
r44025 | cs = 'us-ascii' | ||
Christian Ebert
|
r7114 | if not display: | ||
s, cs = _encode(ui, s, charsets) | ||||
Denis Laxalde
|
r44026 | return mimetextqp(s, 'plain', cs) | ||
Julien Cristau
|
r28341 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49738 | Generator = email.generator.BytesGenerator | ||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43670 | |||
Gregory Szorc
|
r49738 | def parse(fp): | ||
# type: (Any) -> email.message.Message | ||||
ep = email.parser.Parser() | ||||
# disable the "universal newlines" mode, which isn't binary safe. | ||||
# I have no idea if ascii/surrogateescape is correct, but that's | ||||
# what the standard Python email parser does. | ||||
fp = io.TextIOWrapper( | ||||
fp, encoding='ascii', errors='surrogateescape', newline=chr(10) | ||||
) | ||||
try: | ||||
return ep.parse(fp) | ||||
finally: | ||||
fp.detach() | ||||
Denis Laxalde
|
r43426 | |||
Yuya Nishihara
|
r38354 | |||
Gregory Szorc
|
r49738 | def parsebytes(data): | ||
# type: (bytes) -> email.message.Message | ||||
ep = email.parser.BytesParser() | ||||
return ep.parsebytes(data) | ||||
Denis Laxalde
|
r43634 | |||
Augie Fackler
|
r43346 | |||
Julien Cristau
|
r28341 | def headdecode(s): | ||
Denis Laxalde
|
r44024 | # type: (Union[email.header.Header, bytes]) -> bytes | ||
Julien Cristau
|
r28341 | '''Decodes RFC-2047 header''' | ||
uparts = [] | ||||
Pulkit Goyal
|
r30072 | for part, charset in email.header.decode_header(s): | ||
Julien Cristau
|
r28341 | if charset is not None: | ||
try: | ||||
uparts.append(part.decode(charset)) | ||||
continue | ||||
Denis Laxalde
|
r43632 | except (UnicodeDecodeError, LookupError): | ||
Julien Cristau
|
r28341 | pass | ||
Yuya Nishihara
|
r37487 | # On Python 3, decode_header() may return either bytes or unicode | ||
# depending on whether the header has =?<charset>? or not | ||||
if isinstance(part, type(u'')): | ||||
uparts.append(part) | ||||
continue | ||||
Julien Cristau
|
r28341 | try: | ||
uparts.append(part.decode('UTF-8')) | ||||
continue | ||||
except UnicodeDecodeError: | ||||
pass | ||||
uparts.append(part.decode('ISO-8859-1')) | ||||
Yuya Nishihara
|
r31447 | return encoding.unitolocal(u' '.join(uparts)) | ||