mail.py
352 lines
| 12.5 KiB
| text/x-python
|
PythonLexer
/ mercurial / mail.py
Matt Mackall
|
r2889 | # mail.py - mail sending bits for mercurial | ||
# | ||||
# Copyright 2006 Matt Mackall <mpm@selenic.com> | ||||
# | ||||
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
|
r27619 | from __future__ import absolute_import, print_function | ||
Gregory Szorc
|
r25957 | |||
Augie Fackler
|
r19790 | import email | ||
Gregory Szorc
|
r25957 | import os | ||
import quopri | ||||
import smtplib | ||||
import socket | ||||
import sys | ||||
import time | ||||
from .i18n import _ | ||||
from . import ( | ||||
encoding, | ||||
Pierre-Yves David
|
r26587 | error, | ||
Gregory Szorc
|
r25957 | sslutil, | ||
util, | ||||
) | ||||
Matt Mackall
|
r2889 | |||
Nicolas Dumazet
|
r11542 | _oldheaderinit = email.Header.Header.__init__ | ||
def _unifiedheaderinit(self, *args, **kw): | ||||
""" | ||||
Mads Kiilerich
|
r17428 | Python 2.7 introduces a backwards incompatible change | ||
Nicolas Dumazet
|
r11542 | (Python issue1974, r70772) in email.Generator.Generator code: | ||
pre-2.7 code passed "continuation_ws='\t'" to the Header | ||||
constructor, and 2.7 removed this parameter. | ||||
Default argument is continuation_ws=' ', which means that the | ||||
timeless@mozdev.org
|
r26098 | behavior is different in <2.7 and 2.7 | ||
Nicolas Dumazet
|
r11542 | |||
timeless@mozdev.org
|
r26098 | We consider the 2.7 behavior to be preferable, but need | ||
to have an unified behavior for versions 2.4 to 2.7 | ||||
Nicolas Dumazet
|
r11542 | """ | ||
# override continuation_ws | ||||
kw['continuation_ws'] = ' ' | ||||
_oldheaderinit(self, *args, **kw) | ||||
email.Header.Header.__dict__['__init__'] = _unifiedheaderinit | ||||
FUJIWARA Katsunori
|
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. | ||||
''' | ||||
def __init__(self, sslkwargs, **kwargs): | ||||
smtplib.SMTP.__init__(self, **kwargs) | ||||
self._sslkwargs = sslkwargs | ||||
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
|
r25429 | self.sock = sslutil.wrapsocket(self.sock, keyfile, certfile, | ||
**self._sslkwargs) | ||||
FUJIWARA Katsunori
|
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
|
r26673 | class SMTPS(smtplib.SMTP): | ||
'''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. | ||
''' | ||||
def __init__(self, sslkwargs, keyfile=None, certfile=None, **kwargs): | ||||
self.keyfile = keyfile | ||||
self.certfile = certfile | ||||
smtplib.SMTP.__init__(self, **kwargs) | ||||
self.default_port = smtplib.SMTP_SSL_PORT | ||||
self._sslkwargs = sslkwargs | ||||
def _get_socket(self, host, port, timeout): | ||||
if self.debuglevel > 0: | ||||
Gregory Szorc
|
r27619 | print('connect:', (host, port), file=sys.stderr) | ||
timeless@mozdev.org
|
r26673 | new_socket = socket.create_connection((host, port), timeout) | ||
new_socket = sslutil.wrapsocket(new_socket, | ||||
self.keyfile, self.certfile, | ||||
**self._sslkwargs) | ||||
self.file = smtplib.SSLFakeFile(new_socket) | ||||
return new_socket | ||||
FUJIWARA Katsunori
|
r18886 | |||
Matt Mackall
|
r2889 | def _smtp(ui): | ||
Matt Mackall
|
r5973 | '''build an smtp connection and return a function to send mail''' | ||
Matt Mackall
|
r2889 | local_hostname = ui.config('smtp', 'local_hostname') | ||
Patrick Mezard
|
r13244 | tls = ui.config('smtp', 'tls', 'none') | ||
Zhigang Wang
|
r13201 | # backward compatible: when tls = true, we use starttls. | ||
starttls = tls == 'starttls' or util.parsebool(tls) | ||||
smtps = tls == 'smtps' | ||||
Augie Fackler
|
r14965 | if (starttls or smtps) and not util.safehasattr(socket, 'ssl'): | ||
Pierre-Yves David
|
r26587 | raise error.Abort(_("can't use TLS: Python SSL support not installed")) | ||
Matt Mackall
|
r2889 | mailhost = ui.config('smtp', 'host') | ||
if not mailhost: | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_('smtp.host not configured - cannot send mail')) | ||
FUJIWARA Katsunori
|
r18888 | verifycert = ui.config('smtp', 'verifycert', 'strict') | ||
if verifycert not in ['strict', 'loose']: | ||||
if util.parsebool(verifycert) is not False: | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_('invalid smtp.verifycert configuration: %s') | ||
FUJIWARA Katsunori
|
r18888 | % (verifycert)) | ||
Pierre-Yves David
|
r23223 | verifycert = False | ||
FUJIWARA Katsunori
|
r18888 | if (starttls or smtps) and verifycert: | ||
sslkwargs = sslutil.sslkwargs(ui, mailhost) | ||||
else: | ||||
Yuya Nishihara
|
r25463 | # 'ui' is required by sslutil.wrapsocket() and set by sslkwargs() | ||
sslkwargs = {'ui': ui} | ||||
FUJIWARA Katsunori
|
r18888 | if smtps: | ||
ui.note(_('(using smtps)\n')) | ||||
s = SMTPS(sslkwargs, local_hostname=local_hostname) | ||||
elif starttls: | ||||
s = STARTTLS(sslkwargs, local_hostname=local_hostname) | ||||
else: | ||||
s = smtplib.SMTP(local_hostname=local_hostname) | ||||
FUJIWARA Katsunori
|
r19050 | if smtps: | ||
defaultport = 465 | ||||
else: | ||||
defaultport = 25 | ||||
mailport = util.getport(ui.config('smtp', 'port', defaultport)) | ||||
timeless@mozdev.org
|
r26778 | ui.note(_('sending mail: smtp host %s, port %d\n') % | ||
Alexis S. L. Carvalho
|
r2964 | (mailhost, mailport)) | ||
Matt Mackall
|
r2889 | s.connect(host=mailhost, port=mailport) | ||
Zhigang Wang
|
r13201 | if starttls: | ||
ui.note(_('(using starttls)\n')) | ||||
Matt Mackall
|
r2889 | s.ehlo() | ||
s.starttls() | ||||
s.ehlo() | ||||
FUJIWARA Katsunori
|
r18888 | if (starttls or smtps) and verifycert: | ||
ui.note(_('(verifying remote certificate)\n')) | ||||
sslutil.validator(ui, mailhost)(s.sock, verifycert == 'strict') | ||||
Matt Mackall
|
r2889 | username = ui.config('smtp', 'username') | ||
password = ui.config('smtp', 'password') | ||||
Arun Thomas
|
r5749 | if username and not password: | ||
password = ui.getpass() | ||||
Matt Mackall
|
r2889 | if username and password: | ||
ui.note(_('(authenticating to mail server as %s)\n') % | ||||
(username)) | ||||
David Soria Parra
|
r9246 | try: | ||
s.login(username, password) | ||||
Gregory Szorc
|
r25660 | except smtplib.SMTPException as inst: | ||
Pierre-Yves David
|
r26587 | raise error.Abort(inst) | ||
Matt Mackall
|
r2889 | |||
Matt Mackall
|
r5973 | def send(sender, recipients, msg): | ||
try: | ||||
return s.sendmail(sender, recipients, msg) | ||||
Gregory Szorc
|
r25660 | except smtplib.SMTPRecipientsRefused as inst: | ||
Matt Mackall
|
r5973 | recipients = [r[1] for r in inst.recipients.values()] | ||
Pierre-Yves David
|
r26587 | raise error.Abort('\n' + '\n'.join(recipients)) | ||
Gregory Szorc
|
r25660 | except smtplib.SMTPException as inst: | ||
Pierre-Yves David
|
r26587 | raise error.Abort(inst) | ||
Bryan O'Sullivan
|
r5947 | |||
Matt Mackall
|
r5973 | return send | ||
Bryan O'Sullivan
|
r5947 | |||
Matt Mackall
|
r5973 | def _sendmail(ui, sender, recipients, msg): | ||
'''send mail using sendmail.''' | ||||
Matt Mackall
|
r25842 | program = ui.config('email', 'method', 'smtp') | ||
Matt Mackall
|
r5975 | cmdline = '%s -f %s %s' % (program, util.email(sender), | ||
' '.join(map(util.email, recipients))) | ||||
Matt Mackall
|
r5973 | ui.note(_('sending mail: %s\n') % cmdline) | ||
Dirkjan Ochtman
|
r6548 | fp = util.popen(cmdline, 'w') | ||
Matt Mackall
|
r5973 | fp.write(msg) | ||
ret = fp.close() | ||||
if ret: | ||||
Pierre-Yves David
|
r26587 | raise error.Abort('%s %s' % ( | ||
Matt Mackall
|
r5973 | os.path.basename(program.split(None, 1)[0]), | ||
Adrian Buehlmann
|
r14234 | util.explainexit(ret)[0])) | ||
Matt Mackall
|
r2889 | |||
Mads Kiilerich
|
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. | ||||
date = time.strftime('%a %b %d %H:%M:%S %Y', time.localtime()) | ||||
fp.write('From %s %s\n' % (sender, date)) | ||||
fp.write(msg) | ||||
fp.write('\n\n') | ||||
fp.close() | ||||
def connect(ui, mbox=None): | ||||
Matt Mackall
|
r5973 | '''make a mail connection. return a function to send mail. | ||
Matt Mackall
|
r2889 | call as sendmail(sender, list-of-recipients, msg).''' | ||
Mads Kiilerich
|
r15560 | if mbox: | ||
open(mbox, 'wb').close() | ||||
return lambda s, r, m: _mbox(mbox, s, r, m) | ||||
Matt Mackall
|
r5973 | if ui.config('email', 'method', 'smtp') == 'smtp': | ||
Bryan O'Sullivan
|
r5947 | return _smtp(ui) | ||
Matt Mackall
|
r5973 | return lambda s, r, m: _sendmail(ui, s, r, m) | ||
Matt Mackall
|
r2889 | |||
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 | |||
def validateconfig(ui): | ||||
'''determine if we have enough config data to try sending email.''' | ||||
method = ui.config('email', 'method', 'smtp') | ||||
if method == 'smtp': | ||||
if not ui.config('smtp', 'host'): | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_('smtp specified as email transport, ' | ||
Bryan O'Sullivan
|
r4489 | 'but no smtp host configured')) | ||
else: | ||||
Adrian Buehlmann
|
r14271 | if not util.findexe(method): | ||
Pierre-Yves David
|
r26587 | raise error.Abort(_('%r specified as email transport, ' | ||
Bryan O'Sullivan
|
r4489 | 'but not in PATH') % method) | ||
Christian Ebert
|
r7114 | |||
Christian Ebert
|
r7191 | def mimetextpatch(s, subtype='plain', display=False): | ||
Mads Kiilerich
|
r15562 | '''Return MIME message suitable for a patch. | ||
Charset will be detected as utf-8 or (possibly fake) us-ascii. | ||||
Transfer encodings will be used if necessary.''' | ||||
Rocco Rutte
|
r8332 | |||
cs = 'us-ascii' | ||||
Christian Ebert
|
r7191 | if not display: | ||
Rocco Rutte
|
r8332 | try: | ||
s.decode('us-ascii') | ||||
except UnicodeDecodeError: | ||||
Christian Ebert
|
r7191 | try: | ||
Rocco Rutte
|
r8332 | s.decode('utf-8') | ||
cs = 'utf-8' | ||||
Christian Ebert
|
r7191 | except UnicodeDecodeError: | ||
Rocco Rutte
|
r8332 | # We'll go with us-ascii as a fallback. | ||
Christian Ebert
|
r7191 | pass | ||
Rocco Rutte
|
r8332 | |||
Mads Kiilerich
|
r15562 | return mimetextqp(s, subtype, cs) | ||
def mimetextqp(body, subtype, charset): | ||||
'''Return MIME message. | ||||
Mads Kiilerich
|
r17424 | Quoted-printable transfer encoding will be used if necessary. | ||
Mads Kiilerich
|
r15562 | ''' | ||
enc = None | ||||
for line in body.splitlines(): | ||||
if len(line) > 950: | ||||
body = quopri.encodestring(body) | ||||
enc = "quoted-printable" | ||||
break | ||||
msg = email.MIMEText.MIMEText(body, subtype, charset) | ||||
Rocco Rutte
|
r8332 | if enc: | ||
del msg['Content-Transfer-Encoding'] | ||||
msg['Content-Transfer-Encoding'] = enc | ||||
return msg | ||||
Christian Ebert
|
r7191 | |||
Christian Ebert
|
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
|
r7948 | fallbacks = [encoding.fallbackencoding.lower(), | ||
encoding.encoding.lower(), 'utf-8'] | ||||
Martin Geisler
|
r8343 | for cs in fallbacks: # find unique charsets while keeping order | ||
Christian Ebert
|
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
|
r7948 | order. Tries both encoding and fallbackencoding for input. Only as | ||
Christian Ebert
|
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
|
r7948 | for ics in (encoding.encoding, encoding.fallbackencoding): | ||
Christian Ebert
|
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
|
r7195 | ui.warn(_('ignoring invalid sendcharset: %s\n') % ocs) | ||
Christian Ebert
|
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) | ||||
return str(email.Header.Header(s, cs)) | ||||
return s | ||||
Marti Raudsepp
|
r9948 | def _addressencode(ui, name, addr, charsets=None): | ||
Christian Ebert
|
r7114 | name = headencode(ui, name, charsets) | ||
try: | ||||
acc, dom = addr.split('@') | ||||
acc = acc.encode('ascii') | ||||
Marti Raudsepp
|
r9715 | dom = dom.decode(encoding.encoding).encode('idna') | ||
Christian Ebert
|
r7114 | addr = '%s@%s' % (acc, dom) | ||
except UnicodeDecodeError: | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_('invalid email address: %s') % addr) | ||
Christian Ebert
|
r7114 | except ValueError: | ||
try: | ||||
# too strict? | ||||
addr = addr.encode('ascii') | ||||
except UnicodeDecodeError: | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_('invalid local address: %s') % addr) | ||
Christian Ebert
|
r7114 | return email.Utils.formataddr((name, addr)) | ||
Marti Raudsepp
|
r9948 | def addressencode(ui, address, charsets=None, display=False): | ||
'''Turns address into RFC-2047 compliant header.''' | ||||
if display or not address: | ||||
return address or '' | ||||
name, addr = email.Utils.parseaddr(address) | ||||
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 = [] | ||||
for name, addr in email.Utils.getaddresses(addrs): | ||||
if name or addr: | ||||
result.append(_addressencode(ui, name, addr, charsets)) | ||||
return result | ||||
Christian Ebert
|
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
|
r15562 | return mimetextqp(s, 'plain', cs) | ||
Julien Cristau
|
r28341 | |||
def headdecode(s): | ||||
'''Decodes RFC-2047 header''' | ||||
uparts = [] | ||||
for part, charset in email.Header.decode_header(s): | ||||
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')) | ||||
return encoding.tolocal(u' '.join(uparts).encode('UTF-8')) | ||||