patchbomb.py
539 lines
| 19.7 KiB
| text/x-python
|
PythonLexer
/ hgext / patchbomb.py
Martin Geisler
|
r8252 | # patchbomb.py - sending Mercurial changesets as patch emails | ||
# | ||||
# Copyright 2005-2009 Matt Mackall <mpm@selenic.com> and others | ||||
# | ||||
# 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. | ||
Martin Geisler
|
r8252 | |||
Dirkjan Ochtman
|
r8935 | '''command to send changesets as (a series of) patch emails | ||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r7997 | The series is started off with a "[PATCH 0 of N]" introduction, which | ||
describes the series as a whole. | ||||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9269 | Each patch email has a Subject line of "[PATCH M of N] ...", using the | ||
first line of the changeset description as the subject text. The | ||||
message contains two or three body parts: | ||||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9300 | - The changeset description. | ||
- [Optional] The result of running diffstat on the patch. | ||||
Martin Geisler
|
r10973 | - The patch itself, as generated by :hg:`export`. | ||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9269 | Each message refers to the first in the series using the In-Reply-To | ||
and References headers, so they will show up as a sequence in threaded | ||||
mail and news readers, and in mail archives. | ||||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9269 | With the -d/--diffstat option, you will be prompted for each changeset | ||
with a diffstat summary and the changeset summary, so you can be sure | ||||
you are sending the right changes. | ||||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9269 | To configure other defaults, add a section like this to your hgrc | ||
file:: | ||||
Dirkjan Ochtman
|
r6666 | |||
[email] | ||||
from = My Name <my@email> | ||||
to = recipient1, recipient2, ... | ||||
cc = cc1, cc2, ... | ||||
bcc = bcc1, bcc2, ... | ||||
Cédric Duval
|
r11150 | reply-to = address1, address2, ... | ||
Dirkjan Ochtman
|
r6666 | |||
Christian Ebert
|
r10284 | Use ``[patchbomb]`` as configuration section name if you need to | ||
override global ``[email]`` address settings. | ||||
Martin Geisler
|
r10973 | Then you can use the :hg:`email` command to mail a series of | ||
changesets as a patchbomb. | ||||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9269 | To avoid sending patches prematurely, it is a good idea to first run | ||
Martin Geisler
|
r10973 | the :hg:`email` command with the "-n" option (test only). You will be | ||
Martin Geisler
|
r9269 | prompted for an email recipient address, a subject and an introductory | ||
message describing the patches of your patchbomb. Then when all is | ||||
done, patchbomb messages are displayed. If the PAGER environment | ||||
variable is set, your pager will be fired up once for each patchbomb | ||||
message, so you can verify everything is alright. | ||||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9269 | The -m/--mbox option is also very useful. Instead of previewing each | ||
patchbomb message in a pager or sending the messages directly, it will | ||||
create a UNIX mailbox file with the patch emails. This mailbox file | ||||
can be previewed with any mail user agent which supports UNIX mbox | ||||
files, e.g. with mutt:: | ||||
Dirkjan Ochtman
|
r6666 | |||
% mutt -R -f mbox | ||||
Cédric Duval
|
r9286 | When you are previewing the patchbomb messages, you can use ``formail`` | ||
Martin Geisler
|
r9269 | (a utility that is commonly installed as part of the procmail | ||
package), to send each message out:: | ||||
Dirkjan Ochtman
|
r6666 | |||
% formail -s sendmail -bm -t < mbox | ||||
Bill Barry
|
r7694 | That should be all. Now your patchbomb is on its way out. | ||
Martin Geisler
|
r9269 | You can also either configure the method option in the email section | ||
to be a sendmail compatible mailer or fill out the [smtp] section so | ||||
that the patchbomb extension can automatically send patchbombs | ||||
directly from the commandline. See the [email] and [smtp] sections in | ||||
hgrc(5) for details. | ||||
Martin Geisler
|
r9071 | ''' | ||
Vadim Gelfer
|
r1669 | |||
Cédric Duval
|
r9046 | import os, errno, socket, tempfile, cStringIO, time | ||
Christian Ebert
|
r7192 | import email.MIMEMultipart, email.MIMEBase | ||
Benoit Boissinot
|
r6447 | import email.Utils, email.Encoders, email.Generator | ||
Dirkjan Ochtman
|
r11301 | from mercurial import cmdutil, commands, hg, mail, patch, util, discovery | ||
Matt Mackall
|
r3891 | from mercurial.i18n import _ | ||
Joel Rosdahl
|
r6211 | from mercurial.node import bin | ||
Vadim Gelfer
|
r1669 | |||
Alexander Solovyov
|
r9648 | def prompt(ui, prompt, default=None, rest=':'): | ||
Matt Mackall
|
r8208 | if not ui.interactive(): | ||
Alexander Solovyov
|
r9648 | if default is not None: | ||
Alexander Solovyov
|
r9612 | return default | ||
Matt Mackall
|
r10282 | raise util.Abort(_("%s Please enter a valid value" % (prompt + rest))) | ||
Dirkjan Ochtman
|
r7354 | if default: | ||
prompt += ' [%s]' % default | ||||
prompt += rest | ||||
while True: | ||||
r = ui.prompt(prompt, default=default) | ||||
if r: | ||||
return r | ||||
if default is not None: | ||||
return default | ||||
ui.warn(_('Please enter a valid value.\n')) | ||||
def cdiffstat(ui, summary, patchlines): | ||||
s = patch.diffstat(patchlines) | ||||
Alexander Solovyov
|
r7547 | if summary: | ||
ui.write(summary, '\n') | ||||
ui.write(s, '\n') | ||||
Martin Geisler
|
r9647 | ans = prompt(ui, _('does the diffstat above look okay?'), 'y') | ||
Alexander Solovyov
|
r7547 | if not ans.lower().startswith('y'): | ||
raise util.Abort(_('diffstat rejected')) | ||||
Dirkjan Ochtman
|
r7354 | return s | ||
Cédric Duval
|
r10734 | def introneeded(opts, number): | ||
'''is an introductory message required?''' | ||||
return number > 1 or opts.get('intro') or opts.get('desc') | ||||
Dirkjan Ochtman
|
r7354 | def makepatch(ui, repo, patch, opts, _charsets, idx, total, patchname=None): | ||
desc = [] | ||||
node = None | ||||
body = '' | ||||
for line in patch: | ||||
if line.startswith('#'): | ||||
if line.startswith('# Node ID'): | ||||
node = line.split()[-1] | ||||
continue | ||||
if line.startswith('diff -r') or line.startswith('diff --git'): | ||||
break | ||||
desc.append(line) | ||||
if not patchname and not node: | ||||
raise ValueError | ||||
if opts.get('attach'): | ||||
body = ('\n'.join(desc[1:]).strip() or | ||||
'Patch subject is complete summary.') | ||||
body += '\n\n\n' | ||||
if opts.get('plain'): | ||||
while patch and patch[0].startswith('# '): | ||||
patch.pop(0) | ||||
if patch: | ||||
patch.pop(0) | ||||
while patch and not patch[0].strip(): | ||||
patch.pop(0) | ||||
if opts.get('diffstat'): | ||||
body += cdiffstat(ui, '\n'.join(desc), patch) + '\n\n' | ||||
if opts.get('attach') or opts.get('inline'): | ||||
msg = email.MIMEMultipart.MIMEMultipart() | ||||
if body: | ||||
msg.attach(mail.mimeencode(ui, body, _charsets, opts.get('test'))) | ||||
p = mail.mimetextpatch('\n'.join(patch), 'x-patch', opts.get('test')) | ||||
binnode = bin(node) | ||||
timeless
|
r8761 | # if node is mq patch, it will have the patch file's name as a tag | ||
Dirkjan Ochtman
|
r7354 | if not patchname: | ||
patchtags = [t for t in repo.nodetags(binnode) | ||||
if t.endswith('.patch') or t.endswith('.diff')] | ||||
if patchtags: | ||||
patchname = patchtags[0] | ||||
elif total > 1: | ||||
patchname = cmdutil.make_filename(repo, '%b-%n.patch', | ||||
Peter Arrenbrecht
|
r7359 | binnode, seqno=idx, total=total) | ||
Dirkjan Ochtman
|
r7354 | else: | ||
patchname = cmdutil.make_filename(repo, '%b.patch', binnode) | ||||
disposition = 'inline' | ||||
if opts.get('attach'): | ||||
disposition = 'attachment' | ||||
p['Content-Disposition'] = disposition + '; filename=' + patchname | ||||
msg.attach(p) | ||||
else: | ||||
body += '\n'.join(patch) | ||||
msg = mail.mimetextpatch(body, display=opts.get('test')) | ||||
Nicolas Dumazet
|
r9346 | flag = ' '.join(opts.get('flag')) | ||
if flag: | ||||
flag = ' ' + flag | ||||
Dirkjan Ochtman
|
r7354 | subj = desc[0].strip().rstrip('. ') | ||
Cédric Duval
|
r10734 | if not introneeded(opts, total): | ||
Nicolas Dumazet
|
r9346 | subj = '[PATCH%s] %s' % (flag, opts.get('subject') or subj) | ||
Dirkjan Ochtman
|
r7354 | else: | ||
tlen = len(str(total)) | ||||
Nicolas Dumazet
|
r9346 | subj = '[PATCH %0*d of %d%s] %s' % (tlen, idx, total, flag, subj) | ||
Dirkjan Ochtman
|
r7354 | msg['Subject'] = mail.headencode(ui, subj, _charsets, opts.get('test')) | ||
msg['X-Mercurial-Node'] = node | ||||
return msg, subj | ||||
Vadim Gelfer
|
r1669 | def patchbomb(ui, repo, *revs, **opts): | ||
John Goerzen
|
r4283 | '''send changesets by email | ||
Vadim Gelfer
|
r1669 | |||
Martin Geisler
|
r11193 | By default, diffs are sent in the format generated by | ||
:hg:`export`, one per message. The series starts with a "[PATCH 0 | ||||
of N]" introduction, which describes the series as a whole. | ||||
Vadim Gelfer
|
r1672 | |||
Martin Geisler
|
r9269 | Each patch email has a Subject line of "[PATCH M of N] ...", using | ||
the first line of the changeset description as the subject text. | ||||
The message contains two or three parts. First, the changeset | ||||
description. Next, (optionally) if the diffstat program is | ||||
installed and -d/--diffstat is used, the result of running | ||||
diffstat on the patch. Finally, the patch itself, as generated by | ||||
Martin Geisler
|
r10973 | :hg:`export`. | ||
Brendan Cully
|
r4262 | |||
Martin Geisler
|
r9269 | By default the patch is included as text in the email body for | ||
easy reviewing. Using the -a/--attach option will instead create | ||||
an attachment for the patch. With -i/--inline an inline attachment | ||||
will be created. | ||||
Martin Geisler
|
r8472 | |||
Martin Geisler
|
r9269 | With -o/--outgoing, emails will be generated for patches not found | ||
in the destination repository (or only those which are ancestors | ||||
of the specified revisions if any are provided) | ||||
John Goerzen
|
r4280 | |||
Martin Geisler
|
r9269 | With -b/--bundle, changesets are selected as for --outgoing, but a | ||
single email containing a binary Mercurial bundle as an attachment | ||||
will be sent. | ||||
John Goerzen
|
r4280 | |||
Christian Ebert
|
r9289 | Examples:: | ||
John Goerzen
|
r4280 | |||
Christian Ebert
|
r9289 | hg email -r 3000 # send patch 3000 only | ||
hg email -r 3000 -r 3001 # send patches 3000 and 3001 | ||||
hg email -r 3000:3005 # send patches 3000 through 3005 | ||||
hg email 3000 # send patch 3000 (deprecated) | ||||
John Goerzen
|
r4280 | |||
Christian Ebert
|
r9289 | hg email -o # send all patches not in default | ||
hg email -o DEST # send all patches not in DEST | ||||
hg email -o -r 3000 # send all ancestors of 3000 not in default | ||||
hg email -o -r 3000 DEST # send all ancestors of 3000 not in DEST | ||||
John Goerzen
|
r4280 | |||
Christian Ebert
|
r9289 | hg email -b # send bundle of all patches not in default | ||
hg email -b DEST # send bundle of all patches not in DEST | ||||
hg email -b -r 3000 # bundle of all ancestors of 3000 not in default | ||||
hg email -b -r 3000 DEST # bundle of all ancestors of 3000 not in DEST | ||||
John Goerzen
|
r4280 | |||
Martin Geisler
|
r9269 | Before using this command, you will need to enable email in your | ||
hgrc. See the [email] section in hgrc(5) for details. | ||||
Brendan Cully
|
r4262 | ''' | ||
Christian Ebert
|
r7115 | _charsets = mail._charsets(ui) | ||
Christian Ebert
|
r11413 | bundle = opts.get('bundle') | ||
date = opts.get('date') | ||||
mbox = opts.get('mbox') | ||||
outgoing = opts.get('outgoing') | ||||
rev = opts.get('rev') | ||||
# internal option used by pbranches | ||||
patches = opts.get('patches') | ||||
def getoutgoing(dest, revs): | ||||
Brendan Cully
|
r4262 | '''Return the revisions present locally but not in dest''' | ||
dest = ui.expandpath(dest or 'default-push', dest or 'default') | ||||
Sune Foldager
|
r10365 | dest, branches = hg.parseurl(dest) | ||
revs, checkout = hg.addbranchrevs(repo, repo, branches, revs) | ||||
Sune Foldager
|
r10022 | if revs: | ||
revs = [repo.lookup(rev) for rev in revs] | ||||
Matt Mackall
|
r11273 | other = hg.repository(hg.remoteui(repo, opts), dest) | ||
Brendan Cully
|
r4262 | ui.status(_('comparing with %s\n') % dest) | ||
Dirkjan Ochtman
|
r11301 | o = discovery.findoutgoing(repo, other) | ||
Brendan Cully
|
r4262 | if not o: | ||
ui.status(_("no changes found\n")) | ||||
return [] | ||||
Sune Foldager
|
r10022 | o = repo.changelog.nodesbetween(o, revs)[0] | ||
Brendan Cully
|
r4262 | return [str(repo.changelog.rev(r)) for r in o] | ||
Benoit Boissinot
|
r7615 | def getpatches(revs): | ||
for r in cmdutil.revrange(repo, revs): | ||||
output = cStringIO.StringIO() | ||||
Benoit Boissinot
|
r10611 | cmdutil.export(repo, [r], fp=output, | ||
Peter Arrenbrecht
|
r7874 | opts=patch.diffopts(ui, opts)) | ||
Benoit Boissinot
|
r7615 | yield output.getvalue().split('\n') | ||
John Goerzen
|
r4279 | def getbundle(dest): | ||
John Goerzen
|
r4278 | tmpdir = tempfile.mkdtemp(prefix='hg-email-bundle-') | ||
tmpfn = os.path.join(tmpdir, 'bundle') | ||||
try: | ||||
John Goerzen
|
r4279 | commands.bundle(ui, repo, tmpfn, dest, **opts) | ||
Patrick Mezard
|
r5752 | return open(tmpfn, 'rb').read() | ||
John Goerzen
|
r4278 | finally: | ||
try: | ||||
os.unlink(tmpfn) | ||||
except: | ||||
pass | ||||
os.rmdir(tmpdir) | ||||
Christian Ebert
|
r11413 | if not (opts.get('test') or mbox): | ||
Christian Ebert
|
r5472 | # really sending | ||
Bryan O'Sullivan
|
r4489 | mail.validateconfig(ui) | ||
Christian Ebert
|
r11413 | if not (revs or rev or outgoing or bundle or patches): | ||
Bryan O'Sullivan
|
r4493 | raise util.Abort(_('specify at least one changeset with -r or -o')) | ||
Christian Ebert
|
r11413 | if outgoing and bundle: | ||
Christian Ebert
|
r5746 | raise util.Abort(_("--outgoing mode always on with --bundle;" | ||
" do not re-specify --outgoing")) | ||||
John Goerzen
|
r4278 | |||
Christian Ebert
|
r11413 | if outgoing or bundle: | ||
Brendan Cully
|
r4262 | if len(revs) > 1: | ||
raise util.Abort(_("too many destinations")) | ||||
dest = revs and revs[0] or None | ||||
revs = [] | ||||
Christian Ebert
|
r11413 | if rev: | ||
Brendan Cully
|
r4262 | if revs: | ||
raise util.Abort(_('use only one form to specify the revision')) | ||||
Christian Ebert
|
r11413 | revs = rev | ||
Brendan Cully
|
r4262 | |||
Christian Ebert
|
r11413 | if outgoing: | ||
revs = getoutgoing(dest, rev) | ||||
if bundle: | ||||
John Goerzen
|
r4279 | opts['revs'] = revs | ||
Brendan Cully
|
r4262 | |||
# start | ||||
Christian Ebert
|
r11413 | if date: | ||
start_time = util.parsedate(date) | ||||
Bryan O'Sullivan
|
r4566 | else: | ||
start_time = util.makedate() | ||||
Vadim Gelfer
|
r1669 | |||
def genmsgid(id): | ||||
Christian Ebert
|
r4027 | return '<%s.%s@%s>' % (id[:20], int(start_time[0]), socket.getfqdn()) | ||
Vadim Gelfer
|
r1669 | |||
Patrick Mezard
|
r5753 | def getdescription(body, sender): | ||
Christian Ebert
|
r5818 | if opts.get('desc'): | ||
body = open(opts.get('desc')).read() | ||||
Patrick Mezard
|
r5753 | else: | ||
ui.write(_('\nWrite the introductory message for the ' | ||||
'patch series.\n\n')) | ||||
body = ui.edit(body, sender) | ||||
return body | ||||
Peter Arrenbrecht
|
r7353 | def getpatchmsgs(patches, patchnames=None): | ||
John Goerzen
|
r4278 | jumbo = [] | ||
msgs = [] | ||||
Vadim Gelfer
|
r1669 | |||
Christian Ebert
|
r5746 | ui.write(_('This patch series consists of %d patches.\n\n') | ||
% len(patches)) | ||||
John Goerzen
|
r4278 | |||
Peter Arrenbrecht
|
r7353 | name = None | ||
Benoit Boissinot
|
r7616 | for i, p in enumerate(patches): | ||
John Goerzen
|
r4278 | jumbo.extend(p) | ||
Peter Arrenbrecht
|
r7353 | if patchnames: | ||
name = patchnames[i] | ||||
Dirkjan Ochtman
|
r7354 | msg = makepatch(ui, repo, p, opts, _charsets, i + 1, | ||
len(patches), name) | ||||
msgs.append(msg) | ||||
John Goerzen
|
r4278 | |||
Cédric Duval
|
r10734 | if introneeded(opts, len(patches)): | ||
John Goerzen
|
r4278 | tlen = len(str(len(patches))) | ||
Vadim Gelfer
|
r1669 | |||
Nicolas Dumazet
|
r9346 | flag = ' '.join(opts.get('flag')) | ||
if flag: | ||||
Martin Geisler
|
r9647 | subj = '[PATCH %0*d of %d %s]' % (tlen, 0, len(patches), flag) | ||
Nicolas Dumazet
|
r9346 | else: | ||
Martin Geisler
|
r9647 | subj = '[PATCH %0*d of %d]' % (tlen, 0, len(patches)) | ||
subj += ' ' + (opts.get('subject') or | ||||
prompt(ui, 'Subject: ', rest=subj)) | ||||
Vadim Gelfer
|
r1669 | |||
John Goerzen
|
r4278 | body = '' | ||
Christian Ebert
|
r5818 | if opts.get('diffstat'): | ||
Dirkjan Ochtman
|
r7354 | d = cdiffstat(ui, _('Final summary:\n'), jumbo) | ||
Christian Ebert
|
r5785 | if d: | ||
body = '\n' + d | ||||
John Goerzen
|
r4278 | |||
Patrick Mezard
|
r5753 | body = getdescription(body, sender) | ||
Christian Ebert
|
r7115 | msg = mail.mimeencode(ui, body, _charsets, opts.get('test')) | ||
msg['Subject'] = mail.headencode(ui, subj, _charsets, | ||||
opts.get('test')) | ||||
Vadim Gelfer
|
r1669 | |||
Christian Ebert
|
r7115 | msgs.insert(0, (msg, subj)) | ||
John Goerzen
|
r4278 | return msgs | ||
Christian Ebert
|
r2704 | |||
John Goerzen
|
r4278 | def getbundlemsgs(bundle): | ||
Christian Ebert
|
r5818 | subj = (opts.get('subject') | ||
Dirkjan Ochtman
|
r7354 | or prompt(ui, 'Subject:', 'A bundle for your repository')) | ||
Vadim Gelfer
|
r1669 | |||
Patrick Mezard
|
r5753 | body = getdescription('', sender) | ||
John Goerzen
|
r4278 | msg = email.MIMEMultipart.MIMEMultipart() | ||
if body: | ||||
Christian Ebert
|
r7115 | msg.attach(mail.mimeencode(ui, body, _charsets, opts.get('test'))) | ||
John Goerzen
|
r4278 | datapart = email.MIMEBase.MIMEBase('application', 'x-mercurial-bundle') | ||
datapart.set_payload(bundle) | ||||
John Mulligan
|
r7889 | bundlename = '%s.hg' % opts.get('bundlename', 'bundle') | ||
John Goerzen
|
r4284 | datapart.add_header('Content-Disposition', 'attachment', | ||
John Mulligan
|
r7889 | filename=bundlename) | ||
John Goerzen
|
r4278 | email.Encoders.encode_base64(datapart) | ||
msg.attach(datapart) | ||||
Christian Ebert
|
r7115 | msg['Subject'] = mail.headencode(ui, subj, _charsets, opts.get('test')) | ||
return [(msg, subj)] | ||||
Vadim Gelfer
|
r1669 | |||
Christian Ebert
|
r5818 | sender = (opts.get('from') or ui.config('email', 'from') or | ||
Vadim Gelfer
|
r2198 | ui.config('patchbomb', 'from') or | ||
Dirkjan Ochtman
|
r7354 | prompt(ui, 'From', ui.username())) | ||
Vadim Gelfer
|
r1669 | |||
Peter Arrenbrecht
|
r7353 | if patches: | ||
msgs = getpatchmsgs(patches, opts.get('patchnames')) | ||||
Christian Ebert
|
r11413 | elif bundle: | ||
John Goerzen
|
r4279 | msgs = getbundlemsgs(getbundle(dest)) | ||
John Goerzen
|
r4278 | else: | ||
Benoit Boissinot
|
r7615 | msgs = getpatchmsgs(list(getpatches(revs))) | ||
Vadim Gelfer
|
r1669 | |||
Marti Raudsepp
|
r9947 | def getaddrs(opt, prpt=None, default=None): | ||
Cédric Duval
|
r11150 | addrs = opts.get(opt.replace('-', '_')) | ||
if addrs: | ||||
return mail.addrlistencode(ui, addrs, _charsets, | ||||
Marti Raudsepp
|
r9947 | opts.get('test')) | ||
addrs = (ui.config('email', opt) or | ||||
ui.config('patchbomb', opt) or '') | ||||
if not addrs and prpt: | ||||
addrs = prompt(ui, prpt, default) | ||||
return mail.addrlistencode(ui, [addrs], _charsets, opts.get('test')) | ||||
Bryan O'Sullivan
|
r4485 | |||
Vadim Gelfer
|
r1669 | to = getaddrs('to', 'To') | ||
cc = getaddrs('cc', 'Cc', '') | ||||
Marti Raudsepp
|
r9947 | bcc = getaddrs('bcc') | ||
Cédric Duval
|
r11150 | replyto = getaddrs('reply-to') | ||
Christian Ebert
|
r2679 | |||
Vadim Gelfer
|
r1669 | ui.write('\n') | ||
Henrik Stuart <henrik.stuart at edlund.dk>
|
r8025 | parent = opts.get('in_reply_to') or None | ||
Cédric Duval
|
r8826 | # angle brackets may be omitted, they're not semantically part of the msg-id | ||
if parent is not None: | ||||
if not parent.startswith('<'): | ||||
parent = '<' + parent | ||||
if not parent.endswith('>'): | ||||
parent += '>' | ||||
Cédric Duval
|
r8514 | first = True | ||
Volker Kleinfeld
|
r2443 | |||
Vadim Gelfer
|
r1827 | sender_addr = email.Utils.parseaddr(sender)[1] | ||
Christian Ebert
|
r7115 | sender = mail.addressencode(ui, sender, _charsets, opts.get('test')) | ||
Matt Mackall
|
r5973 | sendmail = None | ||
Christian Ebert
|
r7115 | for m, subj in msgs: | ||
Vadim Gelfer
|
r1669 | try: | ||
m['Message-Id'] = genmsgid(m['X-Mercurial-Node']) | ||||
except TypeError: | ||||
m['Message-Id'] = genmsgid('patchbomb') | ||||
if parent: | ||||
m['In-Reply-To'] = parent | ||||
Benoit Allard
|
r7413 | m['References'] = parent | ||
Cédric Duval
|
r8514 | if first: | ||
Vadim Gelfer
|
r1669 | parent = m['Message-Id'] | ||
Cédric Duval
|
r8514 | first = False | ||
Henrik Stuart
|
r8160 | m['User-Agent'] = 'Mercurial-patchbomb/%s' % util.version() | ||
Cédric Duval
|
r9047 | m['Date'] = email.Utils.formatdate(start_time[0], localtime=True) | ||
Volker Kleinfeld
|
r2443 | |||
Christian Ebert
|
r4027 | start_time = (start_time[0] + 1, start_time[1]) | ||
Vadim Gelfer
|
r1669 | m['From'] = sender | ||
m['To'] = ', '.join(to) | ||||
Christian Ebert
|
r5785 | if cc: | ||
m['Cc'] = ', '.join(cc) | ||||
if bcc: | ||||
m['Bcc'] = ', '.join(bcc) | ||||
Cédric Duval
|
r11150 | if replyto: | ||
m['Reply-To'] = ', '.join(replyto) | ||||
Christian Ebert
|
r5818 | if opts.get('test'): | ||
Christian Ebert
|
r7115 | ui.status(_('Displaying '), subj, ' ...\n') | ||
Patrick Mezard
|
r4596 | ui.flush() | ||
Yuya Nishihara
|
r11183 | if 'PAGER' in os.environ and not ui.plain(): | ||
Dirkjan Ochtman
|
r6548 | fp = util.popen(os.environ['PAGER'], 'w') | ||
Patrick Mezard
|
r4599 | else: | ||
fp = ui | ||||
Benoit Boissinot
|
r6447 | generator = email.Generator.Generator(fp, mangle_from_=False) | ||
Vadim Gelfer
|
r1871 | try: | ||
Benoit Boissinot
|
r6447 | generator.flatten(m, 0) | ||
Vadim Gelfer
|
r1871 | fp.write('\n') | ||
except IOError, inst: | ||||
if inst.errno != errno.EPIPE: | ||||
raise | ||||
Patrick Mezard
|
r4599 | if fp is not ui: | ||
fp.close() | ||||
Christian Ebert
|
r11413 | elif mbox: | ||
Christian Ebert
|
r7115 | ui.status(_('Writing '), subj, ' ...\n') | ||
Christian Ebert
|
r11413 | fp = open(mbox, 'In-Reply-To' in m and 'ab+' or 'wb+') | ||
Benoit Boissinot
|
r6447 | generator = email.Generator.Generator(fp, mangle_from_=True) | ||
Patrick Mezard
|
r9818 | # 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(start_time[0])) | ||||
Johannes Stezenbach
|
r1702 | fp.write('From %s %s\n' % (sender_addr, date)) | ||
Benoit Boissinot
|
r6447 | generator.flatten(m, 0) | ||
Johannes Stezenbach
|
r1702 | fp.write('\n\n') | ||
fp.close() | ||||
Vadim Gelfer
|
r1669 | else: | ||
Matt Mackall
|
r5973 | if not sendmail: | ||
sendmail = mail.connect(ui) | ||||
Christian Ebert
|
r7115 | ui.status(_('Sending '), subj, ' ...\n') | ||
Benoit Boissinot
|
r2790 | # Exim does not remove the Bcc field | ||
del m['Bcc'] | ||||
Benoit Boissinot
|
r6447 | fp = cStringIO.StringIO() | ||
generator = email.Generator.Generator(fp, mangle_from_=False) | ||||
generator.flatten(m, 0) | ||||
sendmail(sender, to + bcc + cc, fp.getvalue()) | ||||
Vadim Gelfer
|
r1669 | |||
Peter Arrenbrecht
|
r7352 | emailopts = [ | ||
('a', 'attach', None, _('send patches as attachments')), | ||||
Dennis Schoen
|
r5819 | ('i', 'inline', None, _('send patches as inline attachments')), | ||
timeless
|
r7807 | ('', 'bcc', [], _('email addresses of blind carbon copy recipients')), | ||
Thomas Arendsen Hein
|
r4730 | ('c', 'cc', [], _('email addresses of copy recipients')), | ||
('d', 'diffstat', None, _('add diffstat output to messages')), | ||||
('', 'date', '', _('use the given date as the sending date')), | ||||
Bryan O'Sullivan
|
r4887 | ('', 'desc', '', _('use the given file as the series description')), | ||
Thomas Arendsen Hein
|
r4730 | ('f', 'from', '', _('email address of sender')), | ||
('n', 'test', None, _('print messages that would be sent')), | ||||
('m', 'mbox', '', | ||||
_('write messages to mbox file instead of sending them')), | ||||
Cédric Duval
|
r11150 | ('', 'reply-to', [], _('email addresses replies should be sent to')), | ||
Peter Arrenbrecht
|
r7352 | ('s', 'subject', '', | ||
_('subject of first message (intro or single patch)')), | ||||
Henrik Stuart <henrik.stuart at edlund.dk>
|
r8025 | ('', 'in-reply-to', '', | ||
Martin Geisler
|
r8331 | _('message identifier to reply to')), | ||
Nicolas Dumazet
|
r9346 | ('', 'flag', [], _('flags to add in subject prefixes')), | ||
Peter Arrenbrecht
|
r7352 | ('t', 'to', [], _('email addresses of recipients')), | ||
] | ||||
cmdtable = { | ||||
"email": | ||||
(patchbomb, | ||||
[('g', 'git', None, _('use git extended diff format')), | ||||
('', 'plain', None, _('omit hg patch header')), | ||||
Thomas Arendsen Hein
|
r4730 | ('o', 'outgoing', None, | ||
_('send changes not found in the target repository')), | ||||
('b', 'bundle', None, | ||||
_('send changes not in target as a binary bundle')), | ||||
John Mulligan
|
r7889 | ('', 'bundlename', 'bundle', | ||
FUJIWARA Katsunori
|
r11321 | _('name of the bundle attachment file'), _('NAME')), | ||
('r', 'rev', [], | ||||
_('a revision to send'), _('REV')), | ||||
Thomas Arendsen Hein
|
r4730 | ('', 'force', None, | ||
Martin Geisler
|
r8076 | _('run even when remote repository is unrelated ' | ||
'(with -b/--bundle)')), | ||||
Thomas Arendsen Hein
|
r4730 | ('', 'base', [], | ||
Martin Geisler
|
r8076 | _('a base changeset to specify instead of a destination ' | ||
FUJIWARA Katsunori
|
r11321 | '(with -b/--bundle)'), | ||
_('REV')), | ||||
Chris Winter
|
r7360 | ('', 'intro', None, | ||
_('send an introduction email for a single patch')), | ||||
Peter Arrenbrecht
|
r7352 | ] + emailopts + commands.remoteopts, | ||
Thomas Arendsen Hein
|
r4730 | _('hg email [OPTION]... [DEST]...')) | ||
} | ||||