##// END OF EJS Templates
wireproto: add streams to frame-based protocol...
wireproto: add streams to frame-based protocol Previously, the frame-based protocol was just a series of frames, with each frame associated with a request ID. In order to scale the protocol, we'll want to enable the use of compression. While it is possible to enable compression at the socket/pipe level, this has its disadvantages. The big one is it undermines the point of frames being standalone, atomic units that can be read and written: if you add compression above the framing protocol, you are back to having a stream-based protocol as opposed to something frame-based. So in order to preserve frames, compression needs to occur at the frame payload level. Compressing each frame's payload individually will limit compression ratios because the window size of the compressor will be limited by the max frame size, which is 32-64kb as currently defined. It will also add CPU overhead, as it is more efficient for compressors to operate on fewer, larger blocks of data than more, smaller blocks. So compressing each frame independently is out. This means we need to compress each frame's payload as if it is part of a larger stream. The simplest approach is to have 1 stream per connection. This could certainly work. However, it has disadvantages (documented below). We could also have 1 stream per RPC/command invocation. (This is the model HTTP/2 goes with.) This also has disadvantages. The main disadvantage to one global stream is that it has the very real potential to create CPU bottlenecks doing compression. Networks are only getting faster and the performance of single CPU cores has been relatively flat. Newer compression formats like zstandard offer better CPU cycle efficiency than predecessors like zlib. But it still all too common to saturate your CPU with compression overhead long before you saturate the network pipe. The main disadvantage with streams per request is that you can't reap the benefits of the compression context for multiple requests. For example, if you send 1000 RPC requests (or HTTP/2 requests for that matter), the response to each would have its own compression context. The overall size of the raw responses would be larger because compression contexts wouldn't be able to reference data from another request or response. The approach for streams as implemented in this commit is to support N streams per connection and for streams to potentially span requests and responses. As explained by the added internals docs, this facilitates servers and clients delegating independent streams and compression to independent threads / CPU cores. This helps alleviate the CPU bottleneck of compression. This design also allows compression contexts to be reused across requests/responses. This can result in improved compression ratios and less overhead for compressors and decompressors having to build new contexts. Another feature that was defined was the ability for individual frames within a stream to declare whether that individual frame's payload uses the content encoding (read: compression) defined by the stream. The idea here is that some servers may serve data from a combination of caches and dynamic resolution. Data coming from caches may be pre-compressed. We want to facilitate servers being able to essentially stream bytes from caches to the wire with minimal overhead. Being able to mix and match with frames are compressed within a stream enables these types of advanced server functionality. This commit defines the new streams mechanism. Basic code for supporting streams in frames has been added. But that code is seriously lacking and doesn't fully conform to the defined protocol. For example, we don't close any streams. And support for content encoding within streams is not yet implemented. The change was rather invasive and I didn't think it would be reasonable to implement the entire feature in a single commit. For the record, I would have loved to reuse an existing multiplexing protocol to build the new wire protocol on top of. However, I couldn't find a protocol that offers the performance and scaling characteristics that I desired. Namely, it should support multiple compression contexts to facilitate scaling out to multiple CPU cores and compression contexts should be able to live longer than single RPC requests. HTTP/2 *almost* fits the bill. But the semantics of HTTP message exchange state that streams can only live for a single request-response. We /could/ tunnel on top of HTTP/2 streams and frames with HEADER and DATA frames. But there's no guarantee that HTTP/2 libraries and proxies would allow us to use HTTP/2 streams and frames without the HTTP message exchange semantics defined in RFC 7540 Section 8. Other RPC protocols like gRPC tunnel are built on top of HTTP/2 and thus preserve its semantics of stream per RPC invocation. Even QUIC does this. We could attempt to invent a higher-level stream that spans HTTP/2 streams. But this would be violating HTTP/2 because there is no guarantee that HTTP/2 streams are routed to the same server. The best we can do - which is what this protocol does - is shoehorn all request and response data into a single HTTP message and create streams within. At that point, we've defined a Content-Type in HTTP parlance. It just so happens our media type can also work as a standalone, stream-based protocol, without leaning on HTTP or similar protocol. Differential Revision: https://phab.mercurial-scm.org/D2907

File last commit:

r37138:a8a902d7 default
r37304:9bfcbe4f default
Show More
mail.py
341 lines | 11.9 KiB | text/x-python | PythonLexer
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889 # mail.py - mail sending bits for mercurial
#
# Copyright 2006 Matt Mackall <mpm@selenic.com>
#
Martin Geisler
updated license to be explicit about GPL version 2
r8225 # This software may be used and distributed according to the terms of the
Matt Mackall
Update license to GPLv2+
r10263 # GNU General Public License version 2 or any later version.
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889
Yuya Nishihara
mail: do not print(), use ui.debug() instead...
r30325 from __future__ import absolute_import
Gregory Szorc
mail: use absolute_import
r25957
Augie Fackler
mail: correct import of email module
r19790 import email
Gábor Stefanik
mail: take --encoding and HGENCODING into account...
r30089 import email.charset
Pulkit Goyal
mail: handle renamed email.Header...
r30072 import email.header
Igor Ippolitov
mail: encode long unicode lines in emails properly (issue5687)...
r34311 import email.message
Gregory Szorc
mail: use absolute_import
r25957 import os
import smtplib
import socket
import time
from .i18n import _
from . import (
encoding,
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 error,
Gregory Szorc
py3: pass system string to email.message.Message.set_type()...
r36064 pycompat,
Gregory Szorc
mail: use absolute_import
r25957 sslutil,
util,
)
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 from .utils import (
Yuya Nishihara
procutil: bulk-replace function calls to point to new module
r37138 procutil,
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 stringutil,
)
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889
FUJIWARA Katsunori
smtp: add the class to verify the certificate of the SMTP server for STARTTLS...
r18885 class STARTTLS(smtplib.SMTP):
'''Derived class to verify the peer certificate for STARTTLS.
This class allows to pass any keyword arguments to SSL socket creation.
'''
Gregory Szorc
mail: remove use of sslkwargs
r29251 def __init__(self, ui, host=None, **kwargs):
FUJIWARA Katsunori
smtp: add the class to verify the certificate of the SMTP server for STARTTLS...
r18885 smtplib.SMTP.__init__(self, **kwargs)
Gregory Szorc
sslutil: remove ui from sslkwargs (API)...
r29248 self._ui = ui
timeless
mail: retain hostname for sslutil.wrapsocket (issue5203)...
r28935 self._host = host
FUJIWARA Katsunori
smtp: add the class to verify the certificate of the SMTP server for STARTTLS...
r18885
def starttls(self, keyfile=None, certfile=None):
if not self.has_extn("starttls"):
msg = "STARTTLS extension not supported by server"
raise smtplib.SMTPException(msg)
(resp, reply) = self.docmd("STARTTLS")
if resp == 220:
Yuya Nishihara
ssl: rename ssl_wrap_socket() to conform to our naming convention...
r25429 self.sock = sslutil.wrapsocket(self.sock, keyfile, certfile,
Gregory Szorc
sslutil: remove ui from sslkwargs (API)...
r29248 ui=self._ui,
Gregory Szorc
mail: remove use of sslkwargs
r29251 serverhostname=self._host)
FUJIWARA Katsunori
smtp: add the class to verify the certificate of the SMTP server for STARTTLS...
r18885 self.file = smtplib.SSLFakeFile(self.sock)
self.helo_resp = None
self.ehlo_resp = None
self.esmtp_features = {}
self.does_esmtp = 0
return (resp, reply)
timeless@mozdev.org
mail: drop python 2.5 support
r26673 class SMTPS(smtplib.SMTP):
'''Derived class to verify the peer certificate for SMTPS.
FUJIWARA Katsunori
smtp: add the class to verify the certificate of the SMTP server for SMTPS...
r18886
timeless@mozdev.org
mail: drop python 2.5 support
r26673 This class allows to pass any keyword arguments to SSL socket creation.
'''
Gregory Szorc
mail: remove use of sslkwargs
r29251 def __init__(self, ui, keyfile=None, certfile=None, host=None,
timeless
mail: retain hostname for sslutil.wrapsocket (issue5203)...
r28935 **kwargs):
timeless@mozdev.org
mail: drop python 2.5 support
r26673 self.keyfile = keyfile
self.certfile = certfile
smtplib.SMTP.__init__(self, **kwargs)
timeless
mail: retain hostname for sslutil.wrapsocket (issue5203)...
r28935 self._host = host
timeless@mozdev.org
mail: drop python 2.5 support
r26673 self.default_port = smtplib.SMTP_SSL_PORT
Gregory Szorc
sslutil: remove ui from sslkwargs (API)...
r29248 self._ui = ui
timeless@mozdev.org
mail: drop python 2.5 support
r26673
def _get_socket(self, host, port, timeout):
if self.debuglevel > 0:
Yuya Nishihara
mail: do not print(), use ui.debug() instead...
r30325 self._ui.debug('connect: %r\n' % (host, port))
timeless@mozdev.org
mail: drop python 2.5 support
r26673 new_socket = socket.create_connection((host, port), timeout)
new_socket = sslutil.wrapsocket(new_socket,
self.keyfile, self.certfile,
Gregory Szorc
sslutil: remove ui from sslkwargs (API)...
r29248 ui=self._ui,
Gregory Szorc
mail: remove use of sslkwargs
r29251 serverhostname=self._host)
timeless@mozdev.org
mail: drop python 2.5 support
r26673 self.file = smtplib.SSLFakeFile(new_socket)
return new_socket
FUJIWARA Katsunori
smtp: add the class to verify the certificate of the SMTP server for SMTPS...
r18886
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889 def _smtp(ui):
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 '''build an smtp connection and return a function to send mail'''
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889 local_hostname = ui.config('smtp', 'local_hostname')
Jun Wu
codemod: register core configitems using a script...
r33499 tls = ui.config('smtp', 'tls')
Zhigang Wang
smtp: fix for server doesn't support starttls extension...
r13201 # backward compatible: when tls = true, we use starttls.
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 starttls = tls == 'starttls' or stringutil.parsebool(tls)
Zhigang Wang
smtp: fix for server doesn't support starttls extension...
r13201 smtps = tls == 'smtps'
Augie Fackler
mail: use safehasattr instead of hasattr
r14965 if (starttls or smtps) and not util.safehasattr(socket, 'ssl'):
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("can't use TLS: Python SSL support not installed"))
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889 mailhost = ui.config('smtp', 'host')
if not mailhost:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('smtp.host not configured - cannot send mail'))
FUJIWARA Katsunori
smtp: verify the certificate of the SMTP server for STARTTLS/SMTPS...
r18888 if smtps:
ui.note(_('(using smtps)\n'))
Gregory Szorc
mail: remove use of sslkwargs
r29251 s = SMTPS(ui, local_hostname=local_hostname, host=mailhost)
FUJIWARA Katsunori
smtp: verify the certificate of the SMTP server for STARTTLS/SMTPS...
r18888 elif starttls:
Gregory Szorc
mail: remove use of sslkwargs
r29251 s = STARTTLS(ui, local_hostname=local_hostname, host=mailhost)
FUJIWARA Katsunori
smtp: verify the certificate of the SMTP server for STARTTLS/SMTPS...
r18888 else:
s = smtplib.SMTP(local_hostname=local_hostname)
FUJIWARA Katsunori
smtp: use 465 as default port for SMTPS...
r19050 if smtps:
defaultport = 465
else:
defaultport = 25
mailport = util.getport(ui.config('smtp', 'port', defaultport))
timeless@mozdev.org
l10n: use %d instead of %s for numbers
r26778 ui.note(_('sending mail: smtp host %s, port %d\n') %
Alexis S. L. Carvalho
fix typo in mail.py
r2964 (mailhost, mailport))
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889 s.connect(host=mailhost, port=mailport)
Zhigang Wang
smtp: fix for server doesn't support starttls extension...
r13201 if starttls:
ui.note(_('(using starttls)\n'))
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889 s.ehlo()
s.starttls()
s.ehlo()
Gregory Szorc
mail: unsupport smtp.verifycert (BC)...
r29285 if starttls or smtps:
FUJIWARA Katsunori
smtp: verify the certificate of the SMTP server for STARTTLS/SMTPS...
r18888 ui.note(_('(verifying remote certificate)\n'))
Gregory Szorc
mail: unsupport smtp.verifycert (BC)...
r29285 sslutil.validatesocket(s.sock)
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889 username = ui.config('smtp', 'username')
password = ui.config('smtp', 'password')
Arun Thomas
Patchbomb: Prompt password when using SMTP/TLS and no password in .hgrc....
r5749 if username and not password:
password = ui.getpass()
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889 if username and password:
ui.note(_('(authenticating to mail server as %s)\n') %
(username))
David Soria Parra
email: Catch exceptions during send....
r9246 try:
s.login(username, password)
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except smtplib.SMTPException as inst:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(inst)
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 def send(sender, recipients, msg):
try:
return s.sendmail(sender, recipients, msg)
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except smtplib.SMTPRecipientsRefused as inst:
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 recipients = [r[1] for r in inst.recipients.values()]
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort('\n' + '\n'.join(recipients))
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except smtplib.SMTPException as inst:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(inst)
Bryan O'Sullivan
Backed out changeset dc6ed2736c81
r5947
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 return send
Bryan O'Sullivan
Backed out changeset dc6ed2736c81
r5947
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 def _sendmail(ui, sender, recipients, msg):
'''send mail using sendmail.'''
Jun Wu
codemod: register core configitems using a script...
r33499 program = ui.config('email', 'method')
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 cmdline = '%s -f %s %s' % (program, stringutil.email(sender),
' '.join(map(stringutil.email, recipients)))
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 ui.note(_('sending mail: %s\n') % cmdline)
Yuya Nishihara
procutil: bulk-replace function calls to point to new module
r37138 fp = procutil.popen(cmdline, 'w')
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 fp.write(msg)
ret = fp.close()
if ret:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort('%s %s' % (
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 os.path.basename(program.split(None, 1)[0]),
Yuya Nishihara
procutil: bulk-replace function calls to point to new module
r37138 procutil.explainexit(ret)[0]))
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889
Mads Kiilerich
mail: mbox handling as a part of mail handling, refactored from patchbomb
r15560 def _mbox(mbox, sender, recipients, msg):
'''write mails to mbox'''
fp = open(mbox, 'ab+')
# Should be time.asctime(), but Windows prints 2-characters day
# of month instead of one. Make them print the same thing.
Pulkit Goyal
py3: make sure the first argument of time.strftime() is str...
r35152 date = time.strftime(r'%a %b %d %H:%M:%S %Y', time.localtime())
Mads Kiilerich
mail: mbox handling as a part of mail handling, refactored from patchbomb
r15560 fp.write('From %s %s\n' % (sender, date))
fp.write(msg)
fp.write('\n\n')
fp.close()
def connect(ui, mbox=None):
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 '''make a mail connection. return a function to send mail.
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889 call as sendmail(sender, list-of-recipients, msg).'''
Mads Kiilerich
mail: mbox handling as a part of mail handling, refactored from patchbomb
r15560 if mbox:
open(mbox, 'wb').close()
return lambda s, r, m: _mbox(mbox, s, r, m)
Jun Wu
codemod: register core configitems using a script...
r33499 if ui.config('email', 'method') == 'smtp':
Bryan O'Sullivan
Backed out changeset dc6ed2736c81
r5947 return _smtp(ui)
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 return lambda s, r, m: _sendmail(ui, s, r, m)
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889
Mads Kiilerich
notify: add option for writing to mbox...
r15561 def sendmail(ui, sender, recipients, msg, mbox=None):
send = connect(ui, mbox=mbox)
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 return send(sender, recipients, msg)
Bryan O'Sullivan
patchbomb: Validate email config before we start prompting for info.
r4489
def validateconfig(ui):
'''determine if we have enough config data to try sending email.'''
Jun Wu
codemod: register core configitems using a script...
r33499 method = ui.config('email', 'method')
Bryan O'Sullivan
patchbomb: Validate email config before we start prompting for info.
r4489 if method == 'smtp':
if not ui.config('smtp', 'host'):
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('smtp specified as email transport, '
Bryan O'Sullivan
patchbomb: Validate email config before we start prompting for info.
r4489 'but no smtp host configured'))
else:
Yuya Nishihara
procutil: bulk-replace function calls to point to new module
r37138 if not procutil.findexe(method):
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('%r specified as email transport, '
Bryan O'Sullivan
patchbomb: Validate email config before we start prompting for info.
r4489 'but not in PATH') % method)
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114
Gábor Stefanik
mail: take --encoding and HGENCODING into account...
r30089 def codec2iana(cs):
''''''
Gregory Szorc
py3: cast character set to bytes...
r36136 cs = pycompat.sysbytes(email.charset.Charset(cs).input_charset.lower())
Gábor Stefanik
mail: take --encoding and HGENCODING into account...
r30089
# "latin1" normalizes to "iso8859-1", standard calls for "iso-8859-1"
if cs.startswith("iso") and not cs.startswith("iso-"):
return "iso-" + cs[3:]
return cs
Christian Ebert
mail: mime-encode patches that are utf-8...
r7191 def mimetextpatch(s, subtype='plain', display=False):
Mads Kiilerich
mail: use quoted-printable for mime encoding to avoid too long lines (issue3075)...
r15562 '''Return MIME message suitable for a patch.
Gábor Stefanik
mail: take --encoding and HGENCODING into account...
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.
Mads Kiilerich
mail: use quoted-printable for mime encoding to avoid too long lines (issue3075)...
r15562 Transfer encodings will be used if necessary.'''
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332
Gábor Stefanik
mail: take --encoding and HGENCODING into account...
r30089 cs = ['us-ascii', 'utf-8', encoding.encoding, encoding.fallbackencoding]
if display:
return mimetextqp(s, subtype, 'us-ascii')
for charset in cs:
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332 try:
Gregory Szorc
py3: cast decode() argument to system string...
r36135 s.decode(pycompat.sysstr(charset))
Gábor Stefanik
mail: take --encoding and HGENCODING into account...
r30089 return mimetextqp(s, subtype, codec2iana(charset))
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332 except UnicodeDecodeError:
Gábor Stefanik
mail: take --encoding and HGENCODING into account...
r30089 pass
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332
Gábor Stefanik
mail: take --encoding and HGENCODING into account...
r30089 return mimetextqp(s, subtype, "iso-8859-1")
Mads Kiilerich
mail: use quoted-printable for mime encoding to avoid too long lines (issue3075)...
r15562
def mimetextqp(body, subtype, charset):
'''Return MIME message.
Mads Kiilerich
fix trivial spelling errors
r17424 Quoted-printable transfer encoding will be used if necessary.
Mads Kiilerich
mail: use quoted-printable for mime encoding to avoid too long lines (issue3075)...
r15562 '''
Igor Ippolitov
mail: encode long unicode lines in emails properly (issue5687)...
r34311 cs = email.charset.Charset(charset)
msg = email.message.Message()
Gregory Szorc
py3: pass system string to email.message.Message.set_type()...
r36064 msg.set_type(pycompat.sysstr('text/' + subtype))
Igor Ippolitov
mail: encode long unicode lines in emails properly (issue5687)...
r34311
Mads Kiilerich
mail: use quoted-printable for mime encoding to avoid too long lines (issue3075)...
r15562 for line in body.splitlines():
if len(line) > 950:
Igor Ippolitov
mail: encode long unicode lines in emails properly (issue5687)...
r34311 cs.body_encoding = email.charset.QP
Mads Kiilerich
mail: use quoted-printable for mime encoding to avoid too long lines (issue3075)...
r15562 break
Igor Ippolitov
mail: encode long unicode lines in emails properly (issue5687)...
r34311 msg.set_payload(body, cs)
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332 return msg
Christian Ebert
mail: mime-encode patches that are utf-8...
r7191
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 def _charsets(ui):
'''Obtains charsets to send mail parts not containing patches.'''
charsets = [cs.lower() for cs in ui.configlist('email', 'charsets')]
Matt Mackall
move encoding bits from util to encoding...
r7948 fallbacks = [encoding.fallbackencoding.lower(),
encoding.encoding.lower(), 'utf-8']
Martin Geisler
mail: updated comment
r8343 for cs in fallbacks: # find unique charsets while keeping order
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 if cs not in charsets:
charsets.append(cs)
return [cs for cs in charsets if not cs.endswith('ascii')]
def _encode(ui, s, charsets):
'''Returns (converted) string, charset tuple.
Finds out best charset by cycling through sendcharsets in descending
Matt Mackall
move encoding bits from util to encoding...
r7948 order. Tries both encoding and fallbackencoding for input. Only as
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 last resort send as is in fake ascii.
Caveat: Do not use for mail parts containing patches!'''
try:
s.decode('ascii')
except UnicodeDecodeError:
sendcharsets = charsets or _charsets(ui)
Matt Mackall
move encoding bits from util to encoding...
r7948 for ics in (encoding.encoding, encoding.fallbackencoding):
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 try:
u = s.decode(ics)
except UnicodeDecodeError:
continue
for ocs in sendcharsets:
try:
return u.encode(ocs), ocs
except UnicodeEncodeError:
pass
except LookupError:
Christian Ebert
mail: correct typo in variable name
r7195 ui.warn(_('ignoring invalid sendcharset: %s\n') % ocs)
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 # if ascii, or all conversion attempts fail, send (broken) ascii
return s, 'us-ascii'
def headencode(ui, s, charsets=None, display=False):
'''Returns RFC-2047 compliant header from given string.'''
if not display:
# split into words?
s, cs = _encode(ui, s, charsets)
Pulkit Goyal
mail: handle renamed email.Header...
r30072 return str(email.header.Header(s, cs))
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 return s
Marti Raudsepp
mail: add parseaddrlist() function for parsing many addresses at once...
r9948 def _addressencode(ui, name, addr, charsets=None):
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 name = headencode(ui, name, charsets)
try:
acc, dom = addr.split('@')
acc = acc.encode('ascii')
Marti Raudsepp
patchbomb: fix handling of email addresses with Unicode domains (IDNA)...
r9715 dom = dom.decode(encoding.encoding).encode('idna')
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 addr = '%s@%s' % (acc, dom)
except UnicodeDecodeError:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('invalid email address: %s') % addr)
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 except ValueError:
try:
# too strict?
addr = addr.encode('ascii')
except UnicodeDecodeError:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('invalid local address: %s') % addr)
Gregory Szorc
mail: import email.utils not email.Utils...
r36137 return email.utils.formataddr((name, addr))
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114
Marti Raudsepp
mail: add parseaddrlist() function for parsing many addresses at once...
r9948 def addressencode(ui, address, charsets=None, display=False):
'''Turns address into RFC-2047 compliant header.'''
if display or not address:
return address or ''
Gregory Szorc
mail: import email.utils not email.Utils...
r36137 name, addr = email.utils.parseaddr(address)
Marti Raudsepp
mail: add parseaddrlist() function for parsing many addresses at once...
r9948 return _addressencode(ui, name, addr, charsets)
def addrlistencode(ui, addrs, charsets=None, display=False):
'''Turns a list of addresses into a list of RFC-2047 compliant headers.
A single element of input list may contain multiple addresses, but output
always has one address per item'''
if display:
return [a.strip() for a in addrs if a.strip()]
result = []
Gregory Szorc
mail: import email.utils not email.Utils...
r36137 for name, addr in email.utils.getaddresses(addrs):
Marti Raudsepp
mail: add parseaddrlist() function for parsing many addresses at once...
r9948 if name or addr:
result.append(_addressencode(ui, name, addr, charsets))
return result
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 def mimeencode(ui, s, charsets=None, display=False):
'''creates mime text object, encodes it if needed, and sets
charset and transfer-encoding accordingly.'''
cs = 'us-ascii'
if not display:
s, cs = _encode(ui, s, charsets)
Mads Kiilerich
mail: use quoted-printable for mime encoding to avoid too long lines (issue3075)...
r15562 return mimetextqp(s, 'plain', cs)
Julien Cristau
patch: when importing from email, RFC2047-decode From/Subject headers...
r28341
def headdecode(s):
'''Decodes RFC-2047 header'''
uparts = []
Pulkit Goyal
mail: handle renamed email.Header...
r30072 for part, charset in email.header.decode_header(s):
Julien Cristau
patch: when importing from email, RFC2047-decode From/Subject headers...
r28341 if charset is not None:
try:
uparts.append(part.decode(charset))
continue
except UnicodeDecodeError:
pass
try:
uparts.append(part.decode('UTF-8'))
continue
except UnicodeDecodeError:
pass
uparts.append(part.decode('ISO-8859-1'))
Yuya Nishihara
encoding: factor out unicode variants of from/tolocal()...
r31447 return encoding.unitolocal(u' '.join(uparts))