##// END OF EJS Templates
revset: inline isvalidsymbol() and getsymbol() into _parsealiasdecl()...
revset: inline isvalidsymbol() and getsymbol() into _parsealiasdecl() Since I'm going to extract a common alias parser, I want to eliminate dependencies to the revset parsing rules. These functions are trivial, so we can go without them.

File last commit:

r28341:8286f551 default
r28706:b33ca687 default
Show More
mail.py
352 lines | 12.5 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
Gregory Szorc
mail: use print function...
r27619 from __future__ import absolute_import, print_function
Gregory Szorc
mail: use absolute_import
r25957
Augie Fackler
mail: correct import of email module
r19790 import email
Gregory Szorc
mail: use absolute_import
r25957 import os
import quopri
import smtplib
import socket
import sys
import time
from .i18n import _
from . import (
encoding,
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 error,
Gregory Szorc
mail: use absolute_import
r25957 sslutil,
util,
)
Matt Mackall
Move ui.sendmail to mail.connect/sendmail
r2889
Nicolas Dumazet
mail: ensure that Python2.4 to 2.7 use the same header format...
r11542 _oldheaderinit = email.Header.Header.__init__
def _unifiedheaderinit(self, *args, **kw):
"""
Mads Kiilerich
avoid using abbreviations that look like spelling errors
r17428 Python 2.7 introduces a backwards incompatible change
Nicolas Dumazet
mail: ensure that Python2.4 to 2.7 use the same header format...
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
spelling: behaviour -> behavior
r26098 behavior is different in <2.7 and 2.7
Nicolas Dumazet
mail: ensure that Python2.4 to 2.7 use the same header format...
r11542
timeless@mozdev.org
spelling: behaviour -> behavior
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
mail: ensure that Python2.4 to 2.7 use the same header format...
r11542 """
# override continuation_ws
kw['continuation_ws'] = ' '
_oldheaderinit(self, *args, **kw)
email.Header.Header.__dict__['__init__'] = _unifiedheaderinit
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.
'''
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
ssl: rename ssl_wrap_socket() to conform to our naming convention...
r25429 self.sock = sslutil.wrapsocket(self.sock, keyfile, certfile,
**self._sslkwargs)
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.
'''
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
mail: use print function...
r27619 print('connect:', (host, port), file=sys.stderr)
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,
**self._sslkwargs)
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')
Patrick Mezard
mail: fix regression when parsing unset smtp.tls option
r13244 tls = ui.config('smtp', 'tls', 'none')
Zhigang Wang
smtp: fix for server doesn't support starttls extension...
r13201 # backward compatible: when tls = true, we use starttls.
starttls = tls == 'starttls' or util.parsebool(tls)
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 verifycert = ui.config('smtp', 'verifycert', 'strict')
if verifycert not in ['strict', 'loose']:
if util.parsebool(verifycert) is not False:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('invalid smtp.verifycert configuration: %s')
FUJIWARA Katsunori
smtp: verify the certificate of the SMTP server for STARTTLS/SMTPS...
r18888 % (verifycert))
Pierre-Yves David
mail: actually use the verifycert config value...
r23223 verifycert = False
FUJIWARA Katsunori
smtp: verify the certificate of the SMTP server for STARTTLS/SMTPS...
r18888 if (starttls or smtps) and verifycert:
sslkwargs = sslutil.sslkwargs(ui, mailhost)
else:
Yuya Nishihara
mail: pass ui to sslutil.wrapsocket() even if verifycert is off (issue4713)...
r25463 # 'ui' is required by sslutil.wrapsocket() and set by sslkwargs()
sslkwargs = {'ui': ui}
FUJIWARA Katsunori
smtp: verify the certificate of the SMTP server for STARTTLS/SMTPS...
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
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()
FUJIWARA Katsunori
smtp: verify the certificate of the SMTP server for STARTTLS/SMTPS...
r18888 if (starttls or smtps) and verifycert:
ui.note(_('(verifying remote certificate)\n'))
sslutil.validator(ui, mailhost)(s.sock, verifycert == 'strict')
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.'''
Matt Mackall
email: fix config default value inconsistency
r25842 program = ui.config('email', 'method', 'smtp')
Matt Mackall
templater: move email function to util
r5975 cmdline = '%s -f %s %s' % (program, util.email(sender),
' '.join(map(util.email, recipients)))
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 ui.note(_('sending mail: %s\n') % cmdline)
Dirkjan Ochtman
replace usage of os.popen() with util.popen()...
r6548 fp = util.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]),
Adrian Buehlmann
rename explain_exit to explainexit
r14234 util.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.
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
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)
Matt Mackall
patchbomb: undo backout and fix bugs in the earlier patch
r5973 if ui.config('email', 'method', 'smtp') == '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.'''
method = ui.config('email', 'method', 'smtp')
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:
Adrian Buehlmann
rename util.find_exe to findexe
r14271 if not util.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
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.
Charset will be detected as utf-8 or (possibly fake) us-ascii.
Transfer encodings will be used if necessary.'''
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332
cs = 'us-ascii'
Christian Ebert
mail: mime-encode patches that are utf-8...
r7191 if not display:
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332 try:
s.decode('us-ascii')
except UnicodeDecodeError:
Christian Ebert
mail: mime-encode patches that are utf-8...
r7191 try:
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332 s.decode('utf-8')
cs = 'utf-8'
Christian Ebert
mail: mime-encode patches that are utf-8...
r7191 except UnicodeDecodeError:
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332 # We'll go with us-ascii as a fallback.
Christian Ebert
mail: mime-encode patches that are utf-8...
r7191 pass
Rocco Rutte
patchbomb: quoted-printable encode overly long lines...
r8332
Mads Kiilerich
mail: use quoted-printable for mime encoding to avoid too long lines (issue3075)...
r15562 return mimetextqp(s, subtype, cs)
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 '''
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
patchbomb: quoted-printable encode overly long lines...
r8332 if enc:
del msg['Content-Transfer-Encoding']
msg['Content-Transfer-Encoding'] = enc
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)
return str(email.Header.Header(s, cs))
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)
Christian Ebert
mail: add methods to handle non-ascii chars...
r7114 return email.Utils.formataddr((name, addr))
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 ''
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
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 = []
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'))