patchbomb.py
445 lines
| 16.3 KiB
| text/x-python
|
PythonLexer
/ hgext / patchbomb.py
Vadim Gelfer
|
r1669 | # Command for sending a collection of Mercurial changesets as a series | ||
# of patch emails. | ||||
# | ||||
# The series is started off with a "[PATCH 0 of N]" introduction, | ||||
# which describes the series as a whole. | ||||
# | ||||
# 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: | ||||
# | ||||
# The remainder of the changeset description. | ||||
# | ||||
# [Optional] If the diffstat program is installed, the result of | ||||
# running diffstat on the patch. | ||||
# | ||||
# The patch itself, as generated by "hg export". | ||||
# | ||||
# Each message refers to all of its predecessors 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. | ||||
# | ||||
# For each changeset, you will be prompted with a diffstat summary and | ||||
# the changeset summary, so you can be sure you are sending the right | ||||
# changes. | ||||
# | ||||
Giorgos Keramidas
|
r2926 | # To enable this extension: | ||
Vadim Gelfer
|
r1669 | # | ||
Giorgos Keramidas
|
r2926 | # [extensions] | ||
# hgext.patchbomb = | ||||
Johannes Stezenbach
|
r1702 | # | ||
Vadim Gelfer
|
r1669 | # To configure other defaults, add a section like this to your hgrc | ||
# file: | ||||
# | ||||
Giorgos Keramidas
|
r2926 | # [email] | ||
# from = My Name <my@email> | ||||
# to = recipient1, recipient2, ... | ||||
# cc = cc1, cc2, ... | ||||
# bcc = bcc1, bcc2, ... | ||||
# | ||||
# Then you can use the "hg email" command to mail a series of changesets | ||||
# as a patchbomb. | ||||
# | ||||
# To avoid sending patches prematurely, it is a good idea to first run | ||||
# the "email" command with the "-n" option (test only). You will be | ||||
# prompted for an email recipient address, a subject an an introductory | ||||
# message describing the patches of your patchbomb. Then when all is | ||||
Patrick Mezard
|
r4599 | # done, patchbomb messages are displayed. If PAGER environment variable | ||
# is set, your pager will be fired up once for each patchbomb message, so | ||||
Giorgos Keramidas
|
r2926 | # you can verify everything is alright. | ||
# | ||||
# 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, i.e. with mutt: | ||||
# | ||||
# % mutt -R -f mbox | ||||
# | ||||
# When you are previewing the patchbomb messages, you can use `formail' | ||||
# (a utility that is commonly installed as part of the procmail package), | ||||
# to send each message out: | ||||
# | ||||
# % formail -s sendmail -bm -t < mbox | ||||
# | ||||
# That should be all. Now your patchbomb is on its way out. | ||||
Vadim Gelfer
|
r1669 | |||
John Goerzen
|
r4278 | import os, errno, socket, tempfile | ||
import email.MIMEMultipart, email.MIMEText, email.MIMEBase | ||||
import email.Utils, email.Encoders | ||||
Benoit Boissinot
|
r4029 | from mercurial import cmdutil, commands, hg, mail, ui, patch, util | ||
Matt Mackall
|
r3891 | from mercurial.i18n import _ | ||
Vadim Gelfer
|
r2708 | from mercurial.node import * | ||
Vadim Gelfer
|
r1669 | |||
def patchbomb(ui, repo, *revs, **opts): | ||||
John Goerzen
|
r4283 | '''send changesets by email | ||
Vadim Gelfer
|
r1669 | |||
John Goerzen
|
r4283 | 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 | |||
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. First, the rest of | ||||
the changeset description. Next, (optionally) if the diffstat | ||||
program is installed, the result of running diffstat on the patch. | ||||
Brendan Cully
|
r4262 | Finally, the patch itself, as generated by "hg export". | ||
With --outgoing, emails will be generated for patches not | ||||
John Goerzen
|
r4280 | found in the destination repository (or only those which are | ||
Brendan Cully
|
r4262 | ancestors of the specified revisions if any are provided) | ||
John Goerzen
|
r4280 | |||
With --bundle, changesets are selected as for --outgoing, | ||||
but a single email containing a binary Mercurial bundle as an | ||||
attachment will be sent. | ||||
Examples: | ||||
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) | ||||
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 | ||||
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 | ||||
Before using this command, you will need to enable email in your hgrc. | ||||
John Goerzen
|
r4283 | See the [email] section in hgrc(5) for details. | ||
Brendan Cully
|
r4262 | ''' | ||
Vadim Gelfer
|
r1669 | def prompt(prompt, default = None, rest = ': ', empty_ok = False): | ||
Bryan O'Sullivan
|
r4486 | try: | ||
# readline gives raw_input editing capabilities, but is not | ||||
# present on windows | ||||
import readline | ||||
except ImportError: pass | ||||
Vadim Gelfer
|
r1669 | if default: prompt += ' [%s]' % default | ||
prompt += rest | ||||
while True: | ||||
r = raw_input(prompt) | ||||
if r: return r | ||||
if default is not None: return default | ||||
if empty_ok: return r | ||||
Vadim Gelfer
|
r1670 | ui.warn(_('Please enter a valid value.\n')) | ||
Vadim Gelfer
|
r1669 | |||
def confirm(s): | ||||
if not prompt(s, default = 'y', rest = '? ').lower().startswith('y'): | ||||
raise ValueError | ||||
Matt Doar
|
r3096 | def cdiffstat(summary, patchlines): | ||
s = patch.diffstat(patchlines) | ||||
Vadim Gelfer
|
r1669 | if s: | ||
if summary: | ||||
ui.write(summary, '\n') | ||||
ui.write(s, '\n') | ||||
Vadim Gelfer
|
r1670 | confirm(_('Does the diffstat above look okay')) | ||
Vadim Gelfer
|
r1669 | return s | ||
def makepatch(patch, idx, total): | ||||
desc = [] | ||||
node = None | ||||
body = '' | ||||
for line in patch: | ||||
if line.startswith('#'): | ||||
if line.startswith('# Node ID'): node = line.split()[-1] | ||||
continue | ||||
Brendan Cully
|
r3054 | if (line.startswith('diff -r') | ||
or line.startswith('diff --git')): | ||||
break | ||||
Vadim Gelfer
|
r1669 | desc.append(line) | ||
if not node: raise ValueError | ||||
#body = ('\n'.join(desc[1:]).strip() or | ||||
# 'Patch subject is complete summary.') | ||||
#body += '\n\n\n' | ||||
if opts['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['diffstat']: | ||||
body += cdiffstat('\n'.join(desc), patch) + '\n\n' | ||||
Christian Ebert
|
r2707 | if opts['attach']: | ||
msg = email.MIMEMultipart.MIMEMultipart() | ||||
if body: msg.attach(email.MIMEText.MIMEText(body, 'plain')) | ||||
Vadim Gelfer
|
r2708 | p = email.MIMEText.MIMEText('\n'.join(patch), 'x-patch') | ||
Christian Ebert
|
r2722 | binnode = bin(node) | ||
Vadim Gelfer
|
r2708 | # if node is mq patch, it will have patch file name as tag | ||
Christian Ebert
|
r2722 | patchname = [t for t in repo.nodetags(binnode) | ||
Vadim Gelfer
|
r2708 | if t.endswith('.patch') or t.endswith('.diff')] | ||
if patchname: | ||||
patchname = patchname[0] | ||||
elif total > 1: | ||||
Brendan Cully
|
r3253 | patchname = cmdutil.make_filename(repo, '%b-%n.patch', | ||
Christian Ebert
|
r2722 | binnode, idx, total) | ||
Vadim Gelfer
|
r2708 | else: | ||
Brendan Cully
|
r3253 | patchname = cmdutil.make_filename(repo, '%b.patch', binnode) | ||
Vadim Gelfer
|
r2708 | p['Content-Disposition'] = 'inline; filename=' + patchname | ||
msg.attach(p) | ||||
Christian Ebert
|
r2707 | else: | ||
body += '\n'.join(patch) | ||||
msg = email.MIMEText.MIMEText(body) | ||||
Thomas Arendsen Hein
|
r4141 | |||
Thomas Arendsen Hein
|
r4142 | subj = desc[0].strip().rstrip('. ') | ||
Vadim Gelfer
|
r1846 | if total == 1: | ||
Thomas Arendsen Hein
|
r4141 | subj = '[PATCH] ' + (opts['subject'] or subj) | ||
Vadim Gelfer
|
r1846 | else: | ||
Josef "Jeff" Sipek
|
r3291 | tlen = len(str(total)) | ||
Thomas Arendsen Hein
|
r4141 | subj = '[PATCH %0*d of %d] %s' % (tlen, idx, total, subj) | ||
Vadim Gelfer
|
r1669 | msg['Subject'] = subj | ||
msg['X-Mercurial-Node'] = node | ||||
return msg | ||||
Brendan Cully
|
r4262 | def outgoing(dest, revs): | ||
'''Return the revisions present locally but not in dest''' | ||||
dest = ui.expandpath(dest or 'default-push', dest or 'default') | ||||
revs = [repo.lookup(rev) for rev in revs] | ||||
other = hg.repository(ui, dest) | ||||
ui.status(_('comparing with %s\n') % dest) | ||||
o = repo.findoutgoing(other) | ||||
if not o: | ||||
ui.status(_("no changes found\n")) | ||||
return [] | ||||
o = repo.changelog.nodesbetween(o, revs or None)[0] | ||||
return [str(repo.changelog.rev(r)) for r in o] | ||||
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) | ||
John Goerzen
|
r4278 | return open(tmpfn).read() | ||
finally: | ||||
try: | ||||
os.unlink(tmpfn) | ||||
except: | ||||
pass | ||||
os.rmdir(tmpdir) | ||||
Bryan O'Sullivan
|
r4565 | really_sending = not (opts['test'] or opts['mbox']) | ||
if really_sending: | ||||
Bryan O'Sullivan
|
r4489 | mail.validateconfig(ui) | ||
Bryan O'Sullivan
|
r4493 | if not (revs or opts.get('rev') or opts.get('outgoing')): | ||
raise util.Abort(_('specify at least one changeset with -r or -o')) | ||||
Bryan O'Sullivan
|
r4564 | cmdutil.setremoteconfig(ui, opts) | ||
Bryan O'Sullivan
|
r4492 | if opts.get('outgoing') and opts.get('bundle'): | ||
John Goerzen
|
r4278 | raise util.Abort(_("--outgoing mode always on with --bundle; do not re-specify --outgoing")) | ||
if opts.get('outgoing') or opts.get('bundle'): | ||||
Brendan Cully
|
r4262 | if len(revs) > 1: | ||
raise util.Abort(_("too many destinations")) | ||||
dest = revs and revs[0] or None | ||||
revs = [] | ||||
if opts.get('rev'): | ||||
if revs: | ||||
raise util.Abort(_('use only one form to specify the revision')) | ||||
revs = opts.get('rev') | ||||
if opts.get('outgoing'): | ||||
revs = outgoing(dest, opts.get('rev')) | ||||
John Goerzen
|
r4279 | if opts.get('bundle'): | ||
opts['revs'] = revs | ||||
Brendan Cully
|
r4262 | |||
# start | ||||
Bryan O'Sullivan
|
r4566 | if opts.get('date'): | ||
start_time = util.parsedate(opts['date']) | ||||
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 | |||
John Goerzen
|
r4278 | def getexportmsgs(): | ||
patches = [] | ||||
class exportee: | ||||
def __init__(self, container): | ||||
self.lines = [] | ||||
self.container = container | ||||
self.name = 'email' | ||||
def write(self, data): | ||||
self.lines.append(data) | ||||
def close(self): | ||||
self.container.append(''.join(self.lines).split('\n')) | ||||
self.lines = [] | ||||
Vadim Gelfer
|
r1669 | |||
John Goerzen
|
r4278 | commands.export(ui, repo, *revs, **{'output': exportee(patches), | ||
'switch_parent': False, | ||||
'text': None, | ||||
'git': opts.get('git')}) | ||||
jumbo = [] | ||||
msgs = [] | ||||
Vadim Gelfer
|
r1669 | |||
John Goerzen
|
r4278 | ui.write(_('This patch series consists of %d patches.\n\n') % len(patches)) | ||
for p, i in zip(patches, xrange(len(patches))): | ||||
jumbo.extend(p) | ||||
msgs.append(makepatch(p, i + 1, len(patches))) | ||||
if len(patches) > 1: | ||||
tlen = len(str(len(patches))) | ||||
Vadim Gelfer
|
r1669 | |||
John Goerzen
|
r4278 | subj = '[PATCH %0*d of %d] %s' % ( | ||
tlen, 0, | ||||
len(patches), | ||||
opts['subject'] or | ||||
prompt('Subject:', rest = ' [PATCH %0*d of %d] ' % (tlen, 0, | ||||
len(patches)))) | ||||
Vadim Gelfer
|
r1669 | |||
John Goerzen
|
r4278 | body = '' | ||
if opts['diffstat']: | ||||
d = cdiffstat(_('Final summary:\n'), jumbo) | ||||
if d: body = '\n' + d | ||||
Bryan O'Sullivan
|
r4887 | if opts['desc']: | ||
body = open(opts['desc']).read() | ||||
else: | ||||
ui.write(_('\nWrite the introductory message for the ' | ||||
'patch series.\n\n')) | ||||
body = ui.edit(body, sender) | ||||
John Goerzen
|
r4278 | |||
msg = email.MIMEText.MIMEText(body) | ||||
msg['Subject'] = subj | ||||
Vadim Gelfer
|
r1669 | |||
John Goerzen
|
r4278 | msgs.insert(0, msg) | ||
return msgs | ||||
Christian Ebert
|
r2704 | |||
John Goerzen
|
r4278 | def getbundlemsgs(bundle): | ||
Thomas Arendsen Hein
|
r4633 | subj = (opts['subject'] | ||
or prompt('Subject:', default='A bundle for your repository')) | ||||
John Goerzen
|
r4278 | ui.write(_('\nWrite the introductory message for the bundle.\n\n')) | ||
body = ui.edit('', sender) | ||||
Vadim Gelfer
|
r1669 | |||
John Goerzen
|
r4278 | msg = email.MIMEMultipart.MIMEMultipart() | ||
if body: | ||||
msg.attach(email.MIMEText.MIMEText(body, 'plain')) | ||||
datapart = email.MIMEBase.MIMEBase('application', 'x-mercurial-bundle') | ||||
datapart.set_payload(bundle) | ||||
John Goerzen
|
r4284 | datapart.add_header('Content-Disposition', 'attachment', | ||
filename='bundle.hg') | ||||
John Goerzen
|
r4278 | email.Encoders.encode_base64(datapart) | ||
msg.attach(datapart) | ||||
Christian Ebert
|
r2704 | msg['Subject'] = subj | ||
John Goerzen
|
r4278 | return [msg] | ||
Vadim Gelfer
|
r1669 | |||
Vadim Gelfer
|
r2198 | sender = (opts['from'] or ui.config('email', 'from') or | ||
ui.config('patchbomb', 'from') or | ||||
Vadim Gelfer
|
r1669 | prompt('From', ui.username())) | ||
John Goerzen
|
r4278 | if opts.get('bundle'): | ||
John Goerzen
|
r4279 | msgs = getbundlemsgs(getbundle(dest)) | ||
John Goerzen
|
r4278 | else: | ||
msgs = getexportmsgs() | ||||
Vadim Gelfer
|
r1669 | |||
def getaddrs(opt, prpt, default = None): | ||||
Vadim Gelfer
|
r2198 | addrs = opts[opt] or (ui.config('email', opt) or | ||
ui.config('patchbomb', opt) or | ||||
Vadim Gelfer
|
r1669 | prompt(prpt, default = default)).split(',') | ||
return [a.strip() for a in addrs if a.strip()] | ||||
Bryan O'Sullivan
|
r4485 | |||
Vadim Gelfer
|
r1669 | to = getaddrs('to', 'To') | ||
cc = getaddrs('cc', 'Cc', '') | ||||
Christian Ebert
|
r2679 | bcc = opts['bcc'] or (ui.config('email', 'bcc') or | ||
ui.config('patchbomb', 'bcc') or '').split(',') | ||||
bcc = [a.strip() for a in bcc if a.strip()] | ||||
Vadim Gelfer
|
r1669 | ui.write('\n') | ||
Bryan O'Sullivan
|
r4565 | if really_sending: | ||
Matt Mackall
|
r2889 | mailer = mail.connect(ui) | ||
Vadim Gelfer
|
r1669 | parent = None | ||
Volker Kleinfeld
|
r2443 | |||
Vadim Gelfer
|
r1827 | sender_addr = email.Utils.parseaddr(sender)[1] | ||
Vadim Gelfer
|
r1669 | for m in msgs: | ||
try: | ||||
m['Message-Id'] = genmsgid(m['X-Mercurial-Node']) | ||||
except TypeError: | ||||
m['Message-Id'] = genmsgid('patchbomb') | ||||
if parent: | ||||
m['In-Reply-To'] = parent | ||||
else: | ||||
parent = m['Message-Id'] | ||||
Christian Ebert
|
r4027 | m['Date'] = util.datestr(date=start_time, | ||
format="%a, %d %b %Y %H:%M:%S", timezone=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
|
r2679 | if cc: m['Cc'] = ', '.join(cc) | ||
if bcc: m['Bcc'] = ', '.join(bcc) | ||||
Vadim Gelfer
|
r1669 | if opts['test']: | ||
Johannes Stezenbach
|
r1702 | ui.status('Displaying ', m['Subject'], ' ...\n') | ||
Patrick Mezard
|
r4596 | ui.flush() | ||
Patrick Mezard
|
r4599 | if 'PAGER' in os.environ: | ||
Brendan Cully
|
r4600 | fp = os.popen(os.environ['PAGER'], 'w') | ||
Patrick Mezard
|
r4599 | else: | ||
fp = ui | ||||
Vadim Gelfer
|
r1871 | try: | ||
fp.write(m.as_string(0)) | ||||
fp.write('\n') | ||||
except IOError, inst: | ||||
if inst.errno != errno.EPIPE: | ||||
raise | ||||
Patrick Mezard
|
r4599 | if fp is not ui: | ||
fp.close() | ||||
Johannes Stezenbach
|
r1702 | elif opts['mbox']: | ||
ui.status('Writing ', m['Subject'], ' ...\n') | ||||
fp = open(opts['mbox'], m.has_key('In-Reply-To') and 'ab+' or 'wb+') | ||||
Christian Ebert
|
r4027 | date = util.datestr(date=start_time, | ||
format='%a %b %d %H:%M:%S %Y', timezone=False) | ||||
Johannes Stezenbach
|
r1702 | fp.write('From %s %s\n' % (sender_addr, date)) | ||
fp.write(m.as_string(0)) | ||||
fp.write('\n\n') | ||||
fp.close() | ||||
Vadim Gelfer
|
r1669 | else: | ||
Johannes Stezenbach
|
r1702 | ui.status('Sending ', m['Subject'], ' ...\n') | ||
Benoit Boissinot
|
r2790 | # Exim does not remove the Bcc field | ||
del m['Bcc'] | ||||
Matt Mackall
|
r2889 | mailer.sendmail(sender, to + bcc + cc, m.as_string(0)) | ||
Vadim Gelfer
|
r1669 | |||
cmdtable = { | ||||
Thomas Arendsen Hein
|
r4730 | "email": | ||
(patchbomb, | ||||
[('a', 'attach', None, _('send patches as inline attachments')), | ||||
('', 'bcc', [], _('email addresses of blind copy recipients')), | ||||
('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 | ('g', 'git', None, _('use git extended diff format')), | ||
('f', 'from', '', _('email address of sender')), | ||||
('', 'plain', None, _('omit hg patch header')), | ||||
('n', 'test', None, _('print messages that would be sent')), | ||||
('m', 'mbox', '', | ||||
_('write messages to mbox file instead of sending them')), | ||||
('o', 'outgoing', None, | ||||
_('send changes not found in the target repository')), | ||||
('b', 'bundle', None, | ||||
_('send changes not in target as a binary bundle')), | ||||
('r', 'rev', [], _('a revision to send')), | ||||
('s', 'subject', '', | ||||
_('subject of first message (intro or single patch)')), | ||||
('t', 'to', [], _('email addresses of recipients')), | ||||
('', 'force', None, | ||||
_('run even when remote repository is unrelated (with -b)')), | ||||
('', 'base', [], | ||||
_('a base changeset to specify instead of a destination (with -b)')), | ||||
] + commands.remoteopts, | ||||
_('hg email [OPTION]... [DEST]...')) | ||||
} | ||||