patchbomb.py
1001 lines
| 31.8 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
|
r13838 | To configure other defaults, add a section like this to your | ||
configuration 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 | 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. | ||||
Pierre-Yves David
|
r23487 | |||
Bryan O'Sullivan
|
r27697 | By default, :hg:`email` will prompt for a ``To`` or ``CC`` header if | ||
you do not supply one via configuration or the command line. You can | ||||
override this to never prompt by configuring an empty value:: | ||||
[email] | ||||
cc = | ||||
Pierre-Yves David
|
r23487 | You can control the default inclusion of an introduction message with the | ||
``patchbomb.intro`` configuration option. The configuration is always | ||||
overwritten by command line flags like --intro and --desc:: | ||||
[patchbomb] | ||||
intro=auto # include introduction message if more than 1 patch (default) | ||||
intro=never # never include an introduction message | ||||
intro=always # always include an introduction message | ||||
Pierre-Yves David
|
r23488 | |||
Yuya Nishihara
|
r31187 | You can specify a template for flags to be added in subject prefixes. Flags | ||
specified by --flag option are exported as ``{flags}`` keyword:: | ||||
[patchbomb] | ||||
flagtemplate = "{separate(' ', | ||||
ifeq(branch, 'default', '', branch|upper), | ||||
flags)}" | ||||
Pierre-Yves David
|
r23488 | You can set patchbomb to always ask for confirmation by setting | ||
``patchbomb.confirm`` to true. | ||||
Martin Geisler
|
r9071 | ''' | ||
timeless
|
r28415 | from __future__ import absolute_import | ||
Vadim Gelfer
|
r1669 | |||
Augie Fackler
|
r39067 | import email.encoders as emailencoders | ||
Pulkit Goyal
|
r36453 | import email.generator as emailgen | ||
Pulkit Goyal
|
r38491 | import email.mime.base as emimebase | ||
Pulkit Goyal
|
r38490 | import email.mime.multipart as emimemultipart | ||
Pulkit Goyal
|
r36466 | import email.utils as eutil | ||
timeless
|
r28415 | import errno | ||
import os | ||||
import socket | ||||
Augie Fackler
|
r19810 | |||
Yuya Nishihara
|
r29205 | from mercurial.i18n import _ | ||
Gregory Szorc
|
r43355 | from mercurial.pycompat import open | ||
timeless
|
r28415 | from mercurial import ( | ||
cmdutil, | ||||
commands, | ||||
Pulkit Goyal
|
r36467 | encoding, | ||
timeless
|
r28415 | error, | ||
Yuya Nishihara
|
r31187 | formatter, | ||
timeless
|
r28415 | hg, | ||
mail, | ||||
node as nodemod, | ||||
patch, | ||||
Pulkit Goyal
|
r35035 | pycompat, | ||
Yuya Nishihara
|
r32337 | registrar, | ||
timeless
|
r28415 | scmutil, | ||
Yuya Nishihara
|
r31187 | templater, | ||
timeless
|
r28415 | util, | ||
) | ||||
Boris Feld
|
r36625 | from mercurial.utils import dateutil | ||
Augie Fackler
|
r43346 | |||
timeless
|
r28861 | stringio = util.stringio | ||
Vadim Gelfer
|
r1669 | |||
Adrian Buehlmann
|
r14309 | cmdtable = {} | ||
Yuya Nishihara
|
r32337 | command = registrar.command(cmdtable) | ||
Boris Feld
|
r34113 | |||
configtable = {} | ||||
configitem = registrar.configitem(configtable) | ||||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'bundletype', default=None, | ||
Boris Feld
|
r34113 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'bcc', default=None, | ||
Boris Feld
|
r34761 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'cc', default=None, | ||
Boris Feld
|
r34762 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'confirm', default=False, | ||
Boris Feld
|
r34114 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'flagtemplate', default=None, | ||
Boris Feld
|
r34115 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'from', default=None, | ||
Boris Feld
|
r34116 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'intro', default=b'auto', | ||
Boris Feld
|
r34117 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'publicurl', default=None, | ||
Boris Feld
|
r34118 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'reply-to', default=None, | ||
Boris Feld
|
r34763 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r43347 | b'patchbomb', b'to', default=None, | ||
Yuya Nishihara
|
r34912 | ) | ||
Boris Feld
|
r34113 | |||
Augie Fackler
|
r39057 | if pycompat.ispy3: | ||
_bytesgenerator = emailgen.BytesGenerator | ||||
else: | ||||
Yuya Nishihara
|
r39140 | _bytesgenerator = emailgen.Generator | ||
Augie Fackler
|
r39057 | |||
Augie Fackler
|
r29841 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
Augie Fackler
|
r25186 | # extensions which SHIP WITH MERCURIAL. Non-mainline extensions should | ||
# be specifying the version(s) of Mercurial they are tested with, or | ||||
# leave the attribute unspecified. | ||||
Augie Fackler
|
r43347 | testedwith = b'ships-with-hg-core' | ||
Adrian Buehlmann
|
r14309 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r26546 | def _addpullheader(seq, ctx): | ||
"""Add a header pointing to a public URL where the changeset is available | ||||
""" | ||||
repo = ctx.repo() | ||||
# experimental config: patchbomb.publicurl | ||||
# waiting for some logic that check that the changeset are available on the | ||||
# destination before patchbombing anything. | ||||
Augie Fackler
|
r43347 | publicurl = repo.ui.config(b'patchbomb', b'publicurl') | ||
Augie Fackler
|
r32825 | if publicurl: | ||
Martin von Zweigbergk
|
r43387 | return b'Available At %s\n# hg pull %s -r %s' % ( | ||
Augie Fackler
|
r43346 | publicurl, | ||
publicurl, | ||||
ctx, | ||||
) | ||||
Pierre-Yves David
|
r26546 | return None | ||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r26546 | def uisetup(ui): | ||
Augie Fackler
|
r43347 | cmdutil.extraexport.append(b'pullurl') | ||
cmdutil.extraexportmap[b'pullurl'] = _addpullheader | ||||
Pierre-Yves David
|
r26546 | |||
Augie Fackler
|
r43346 | |||
Boris Feld
|
r33436 | def reposetup(ui, repo): | ||
if not repo.local(): | ||||
return | ||||
Augie Fackler
|
r43347 | repo._wlockfreeprefix.add(b'last-email.txt') | ||
Pierre-Yves David
|
r26546 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | def prompt(ui, prompt, default=None, rest=b':'): | ||
Dirkjan Ochtman
|
r7354 | if default: | ||
Augie Fackler
|
r43347 | prompt += b' [%s]' % default | ||
Matt Mackall
|
r15166 | return ui.prompt(prompt + rest, default) | ||
Dirkjan Ochtman
|
r7354 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r23487 | def introwanted(ui, opts, number): | ||
Greg Ward
|
r15164 | '''is an introductory message apparently wanted?''' | ||
Augie Fackler
|
r43347 | introconfig = ui.config(b'patchbomb', b'intro') | ||
if opts.get(b'intro') or opts.get(b'desc'): | ||||
Pierre-Yves David
|
r23487 | intro = True | ||
Augie Fackler
|
r43347 | elif introconfig == b'always': | ||
Pierre-Yves David
|
r23487 | intro = True | ||
Augie Fackler
|
r43347 | elif introconfig == b'never': | ||
Pierre-Yves David
|
r23487 | intro = False | ||
Augie Fackler
|
r43347 | elif introconfig == b'auto': | ||
Martin von Zweigbergk
|
r40065 | intro = number > 1 | ||
Pierre-Yves David
|
r23487 | else: | ||
Augie Fackler
|
r43346 | ui.write_err( | ||
Augie Fackler
|
r43347 | _(b'warning: invalid patchbomb.intro value "%s"\n') % introconfig | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | ui.write_err(_(b'(should be one of always, never, auto)\n')) | ||
Martin von Zweigbergk
|
r40065 | intro = number > 1 | ||
Pierre-Yves David
|
r23487 | return intro | ||
Cédric Duval
|
r10734 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r31187 | def _formatflags(ui, repo, rev, flags): | ||
"""build flag string optionally by template""" | ||||
Augie Fackler
|
r43347 | tmpl = ui.config(b'patchbomb', b'flagtemplate') | ||
Yuya Nishihara
|
r31187 | if not tmpl: | ||
Augie Fackler
|
r43347 | return b' '.join(flags) | ||
Yuya Nishihara
|
r31187 | out = util.stringio() | ||
Yuya Nishihara
|
r43369 | spec = formatter.templatespec(b'', templater.unquotestring(tmpl), None) | ||
with formatter.templateformatter(ui, out, b'patchbombflag', {}, spec) as fm: | ||||
Yuya Nishihara
|
r31187 | fm.startitem() | ||
fm.context(ctx=repo[rev]) | ||||
Augie Fackler
|
r43347 | fm.write(b'flags', b'%s', fm.formatlist(flags, name=b'flag')) | ||
Yuya Nishihara
|
r31187 | return out.getvalue() | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r31186 | def _formatprefix(ui, repo, rev, flags, idx, total, numbered): | ||
Yuya Nishihara
|
r31183 | """build prefix to patch subject""" | ||
Yuya Nishihara
|
r31187 | flag = _formatflags(ui, repo, rev, flags) | ||
Yuya Nishihara
|
r31183 | if flag: | ||
Augie Fackler
|
r43347 | flag = b' ' + flag | ||
Yuya Nishihara
|
r31183 | |||
if not numbered: | ||||
Augie Fackler
|
r43347 | return b'[PATCH%s]' % flag | ||
Yuya Nishihara
|
r31183 | else: | ||
Augie Fackler
|
r43347 | tlen = len(b"%d" % total) | ||
return b'[PATCH %0*d of %d%s]' % (tlen, idx, total, flag) | ||||
Yuya Nishihara
|
r31183 | |||
Augie Fackler
|
r43346 | |||
def makepatch( | ||||
ui, | ||||
repo, | ||||
rev, | ||||
patchlines, | ||||
opts, | ||||
_charsets, | ||||
idx, | ||||
total, | ||||
numbered, | ||||
patchname=None, | ||||
): | ||||
Dirkjan Ochtman
|
r7354 | |||
desc = [] | ||||
node = None | ||||
Augie Fackler
|
r43347 | body = b'' | ||
Dirkjan Ochtman
|
r7354 | |||
Martin Geisler
|
r12199 | for line in patchlines: | ||
Augie Fackler
|
r43347 | if line.startswith(b'#'): | ||
if line.startswith(b'# Node ID'): | ||||
Dirkjan Ochtman
|
r7354 | node = line.split()[-1] | ||
continue | ||||
Augie Fackler
|
r43347 | if line.startswith(b'diff -r') or line.startswith(b'diff --git'): | ||
Dirkjan Ochtman
|
r7354 | break | ||
desc.append(line) | ||||
if not patchname and not node: | ||||
raise ValueError | ||||
Augie Fackler
|
r43347 | if opts.get(b'attach') and not opts.get(b'body'): | ||
Augie Fackler
|
r43346 | body = ( | ||
Augie Fackler
|
r43347 | b'\n'.join(desc[1:]).strip() | ||
or b'Patch subject is complete summary.' | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | body += b'\n\n\n' | ||
Dirkjan Ochtman
|
r7354 | |||
Augie Fackler
|
r43347 | if opts.get(b'plain'): | ||
while patchlines and patchlines[0].startswith(b'# '): | ||||
Martin Geisler
|
r12199 | patchlines.pop(0) | ||
if patchlines: | ||||
patchlines.pop(0) | ||||
while patchlines and not patchlines[0].strip(): | ||||
patchlines.pop(0) | ||||
Dirkjan Ochtman
|
r7354 | |||
Henning Schild
|
r30407 | ds = patch.diffstat(patchlines) | ||
Augie Fackler
|
r43347 | if opts.get(b'diffstat'): | ||
body += ds + b'\n\n' | ||||
Dirkjan Ochtman
|
r7354 | |||
Augie Fackler
|
r43347 | addattachment = opts.get(b'attach') or opts.get(b'inline') | ||
if not addattachment or opts.get(b'body'): | ||||
body += b'\n'.join(patchlines) | ||||
Angel Ezquerra
|
r16307 | |||
if addattachment: | ||||
Pulkit Goyal
|
r38490 | msg = emimemultipart.MIMEMultipart() | ||
Dirkjan Ochtman
|
r7354 | if body: | ||
Augie Fackler
|
r43347 | msg.attach(mail.mimeencode(ui, body, _charsets, opts.get(b'test'))) | ||
Augie Fackler
|
r43346 | p = mail.mimetextpatch( | ||
Augie Fackler
|
r43347 | b'\n'.join(patchlines), b'x-patch', opts.get(b'test') | ||
Augie Fackler
|
r43346 | ) | ||
timeless
|
r28415 | binnode = nodemod.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: | ||
Augie Fackler
|
r43346 | patchtags = [ | ||
t | ||||
for t in repo.nodetags(binnode) | ||||
Augie Fackler
|
r43347 | if t.endswith(b'.patch') or t.endswith(b'.diff') | ||
Augie Fackler
|
r43346 | ] | ||
Dirkjan Ochtman
|
r7354 | if patchtags: | ||
patchname = patchtags[0] | ||||
elif total > 1: | ||||
Augie Fackler
|
r43346 | patchname = cmdutil.makefilename( | ||
Augie Fackler
|
r43347 | repo[node], b'%b-%n.patch', seqno=idx, total=total | ||
Augie Fackler
|
r43346 | ) | ||
Dirkjan Ochtman
|
r7354 | else: | ||
Augie Fackler
|
r43347 | patchname = cmdutil.makefilename(repo[node], b'%b.patch') | ||
Augie Fackler
|
r39073 | disposition = r'inline' | ||
Augie Fackler
|
r43347 | if opts.get(b'attach'): | ||
Augie Fackler
|
r39073 | disposition = r'attachment' | ||
p[r'Content-Disposition'] = ( | ||||
Augie Fackler
|
r43346 | disposition + r'; filename=' + encoding.strfromlocal(patchname) | ||
) | ||||
Dirkjan Ochtman
|
r7354 | msg.attach(p) | ||
else: | ||||
Augie Fackler
|
r43347 | msg = mail.mimetextpatch(body, display=opts.get(b'test')) | ||
Dirkjan Ochtman
|
r7354 | |||
Augie Fackler
|
r43346 | prefix = _formatprefix( | ||
Augie Fackler
|
r43347 | ui, repo, rev, opts.get(b'flag'), idx, total, numbered | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | subj = desc[0].strip().rstrip(b'. ') | ||
Greg Ward
|
r15164 | if not numbered: | ||
Augie Fackler
|
r43347 | subj = b' '.join([prefix, opts.get(b'subject') or subj]) | ||
Dirkjan Ochtman
|
r7354 | else: | ||
Augie Fackler
|
r43347 | subj = b' '.join([prefix, subj]) | ||
msg[b'Subject'] = mail.headencode(ui, subj, _charsets, opts.get(b'test')) | ||||
msg[b'X-Mercurial-Node'] = node | ||||
msg[b'X-Mercurial-Series-Index'] = b'%i' % idx | ||||
msg[b'X-Mercurial-Series-Total'] = b'%i' % total | ||||
Christian Ebert
|
r12200 | return msg, subj, ds | ||
Dirkjan Ochtman
|
r7354 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r23210 | def _getpatches(repo, revs, **opts): | ||
"""return a list of patches for a list of revisions | ||||
Each patch in the list is itself a list of lines. | ||||
""" | ||||
ui = repo.ui | ||||
Augie Fackler
|
r43347 | prev = repo[b'.'].rev() | ||
Yuya Nishihara
|
r24568 | for r in revs: | ||
Pierre-Yves David
|
r23210 | if r == prev and (repo[None].files() or repo[None].deleted()): | ||
Martin von Zweigbergk
|
r43387 | ui.warn(_(b'warning: working directory has uncommitted changes\n')) | ||
timeless
|
r28861 | output = stringio() | ||
Augie Fackler
|
r43346 | cmdutil.exportfile( | ||
repo, [r], output, opts=patch.difffeatureopts(ui, opts, git=True) | ||||
) | ||||
Augie Fackler
|
r43347 | yield output.getvalue().split(b'\n') | ||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r23211 | def _getbundle(repo, dest, **opts): | ||
"""return a bundle containing changesets missing in "dest" | ||||
The `opts` keyword-arguments are the same as the one accepted by the | ||||
`bundle` command. | ||||
The bundle is a returned as a single in-memory binary blob. | ||||
""" | ||||
ui = repo.ui | ||||
Augie Fackler
|
r43347 | tmpdir = pycompat.mkdtemp(prefix=b'hg-email-bundle-') | ||
tmpfn = os.path.join(tmpdir, b'bundle') | ||||
btype = ui.config(b'patchbomb', b'bundletype') | ||||
Pierre-Yves David
|
r26563 | if btype: | ||
Pulkit Goyal
|
r35035 | opts[r'type'] = btype | ||
Pierre-Yves David
|
r23211 | try: | ||
commands.bundle(ui, repo, tmpfn, dest, **opts) | ||||
Bryan O'Sullivan
|
r27767 | return util.readfile(tmpfn) | ||
Pierre-Yves David
|
r23211 | finally: | ||
try: | ||||
os.unlink(tmpfn) | ||||
except OSError: | ||||
pass | ||||
os.rmdir(tmpdir) | ||||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r23212 | def _getdescription(repo, defaultbody, sender, **opts): | ||
"""obtain the body of the introduction message and return it | ||||
This is also used for the body of email with an attached bundle. | ||||
The body can be obtained either from the command line option or entered by | ||||
the user through the editor. | ||||
""" | ||||
ui = repo.ui | ||||
Pulkit Goyal
|
r35035 | if opts.get(r'desc'): | ||
body = open(opts.get(r'desc')).read() | ||||
Pierre-Yves David
|
r23212 | else: | ||
Augie Fackler
|
r43346 | ui.write( | ||
Martin von Zweigbergk
|
r43387 | _(b'\nWrite the introductory message for the patch series.\n\n') | ||
Augie Fackler
|
r43346 | ) | ||
body = ui.edit( | ||||
Augie Fackler
|
r43347 | defaultbody, sender, repopath=repo.path, action=b'patchbombbody' | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r23212 | # Save series description in case sendmail fails | ||
Augie Fackler
|
r43347 | msgfile = repo.vfs(b'last-email.txt', b'wb') | ||
Pierre-Yves David
|
r23212 | msgfile.write(body) | ||
msgfile.close() | ||||
return body | ||||
Pierre-Yves David
|
r23211 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r23213 | def _getbundlemsgs(repo, sender, bundle, **opts): | ||
"""Get the full email for sending a given bundle | ||||
This function returns a list of "email" tuples (subject, content, None). | ||||
The list is always one message long in that case. | ||||
""" | ||||
ui = repo.ui | ||||
_charsets = mail._charsets(ui) | ||||
Augie Fackler
|
r43346 | subj = opts.get(r'subject') or prompt( | ||
Augie Fackler
|
r43347 | ui, b'Subject:', b'A bundle for your repository' | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r23213 | |||
Augie Fackler
|
r43347 | body = _getdescription(repo, b'', sender, **opts) | ||
Pulkit Goyal
|
r38490 | msg = emimemultipart.MIMEMultipart() | ||
Pierre-Yves David
|
r23213 | if body: | ||
Pulkit Goyal
|
r35035 | msg.attach(mail.mimeencode(ui, body, _charsets, opts.get(r'test'))) | ||
Augie Fackler
|
r39069 | datapart = emimebase.MIMEBase(r'application', r'x-mercurial-bundle') | ||
Pierre-Yves David
|
r23213 | datapart.set_payload(bundle) | ||
Augie Fackler
|
r43347 | bundlename = b'%s.hg' % opts.get(r'bundlename', b'bundle') | ||
Augie Fackler
|
r43346 | datapart.add_header( | ||
r'Content-Disposition', | ||||
r'attachment', | ||||
filename=encoding.strfromlocal(bundlename), | ||||
) | ||||
Augie Fackler
|
r39067 | emailencoders.encode_base64(datapart) | ||
Pierre-Yves David
|
r23213 | msg.attach(datapart) | ||
Augie Fackler
|
r43347 | msg[b'Subject'] = mail.headencode(ui, subj, _charsets, opts.get(r'test')) | ||
Pierre-Yves David
|
r23213 | return [(msg, subj, None)] | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r31186 | def _makeintro(repo, sender, revs, patches, **opts): | ||
Pierre-Yves David
|
r23214 | """make an introduction email, asking the user for content if needed | ||
email is returned as (subject, body, cumulative-diffstat)""" | ||||
ui = repo.ui | ||||
_charsets = mail._charsets(ui) | ||||
Yuya Nishihara
|
r31186 | # use the last revision which is likely to be a bookmarked head | ||
Augie Fackler
|
r43346 | prefix = _formatprefix( | ||
ui, repo, revs.last(), opts.get(r'flag'), 0, len(patches), numbered=True | ||||
) | ||||
subj = opts.get(r'subject') or prompt( | ||||
Augie Fackler
|
r43347 | ui, b'(optional) Subject: ', rest=prefix, default=b'' | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r23214 | if not subj: | ||
Augie Fackler
|
r43346 | return None # skip intro if the user doesn't bother | ||
Pierre-Yves David
|
r23214 | |||
Augie Fackler
|
r43347 | subj = prefix + b' ' + subj | ||
Pierre-Yves David
|
r23214 | |||
Augie Fackler
|
r43347 | body = b'' | ||
Pulkit Goyal
|
r35035 | if opts.get(r'diffstat'): | ||
Pierre-Yves David
|
r23214 | # generate a cumulative diffstat of the whole patch series | ||
diffstat = patch.diffstat(sum(patches, [])) | ||||
Augie Fackler
|
r43347 | body = b'\n' + diffstat | ||
Pierre-Yves David
|
r23214 | else: | ||
diffstat = None | ||||
body = _getdescription(repo, body, sender, **opts) | ||||
Pulkit Goyal
|
r35035 | msg = mail.mimeencode(ui, body, _charsets, opts.get(r'test')) | ||
Augie Fackler
|
r43347 | msg[b'Subject'] = mail.headencode(ui, subj, _charsets, opts.get(r'test')) | ||
Pierre-Yves David
|
r23214 | return (msg, subj, diffstat) | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r31185 | def _getpatchmsgs(repo, sender, revs, patchnames=None, **opts): | ||
Pierre-Yves David
|
r23215 | """return a list of emails from a list of patches | ||
This involves introduction message creation if necessary. | ||||
This function returns a list of "email" tuples (subject, content, None). | ||||
""" | ||||
Pulkit Goyal
|
r35035 | bytesopts = pycompat.byteskwargs(opts) | ||
Pierre-Yves David
|
r23215 | ui = repo.ui | ||
_charsets = mail._charsets(ui) | ||||
Yuya Nishihara
|
r31185 | patches = list(_getpatches(repo, revs, **opts)) | ||
Pierre-Yves David
|
r23215 | msgs = [] | ||
Augie Fackler
|
r43347 | ui.write(_(b'this patch series consists of %d patches.\n\n') % len(patches)) | ||
Pierre-Yves David
|
r23215 | |||
# build the intro message, or skip it if the user declines | ||||
Pulkit Goyal
|
r35035 | if introwanted(ui, bytesopts, len(patches)): | ||
Yuya Nishihara
|
r31186 | msg = _makeintro(repo, sender, revs, patches, **opts) | ||
Pierre-Yves David
|
r23215 | if msg: | ||
msgs.append(msg) | ||||
# are we going to send more than one message? | ||||
numbered = len(msgs) + len(patches) > 1 | ||||
# now generate the actual patch messages | ||||
name = None | ||||
Yuya Nishihara
|
r31186 | assert len(revs) == len(patches) | ||
for i, (r, p) in enumerate(zip(revs, patches)): | ||||
Pierre-Yves David
|
r23215 | if patchnames: | ||
name = patchnames[i] | ||||
Augie Fackler
|
r43346 | msg = makepatch( | ||
ui, | ||||
repo, | ||||
r, | ||||
p, | ||||
bytesopts, | ||||
_charsets, | ||||
i + 1, | ||||
len(patches), | ||||
numbered, | ||||
name, | ||||
) | ||||
Pierre-Yves David
|
r23215 | msgs.append(msg) | ||
return msgs | ||||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r23486 | def _getoutgoing(repo, dest, revs): | ||
'''Return the revisions present locally but not in dest''' | ||||
ui = repo.ui | ||||
Augie Fackler
|
r43347 | url = ui.expandpath(dest or b'default-push', dest or b'default') | ||
Pierre-Yves David
|
r23486 | url = hg.parseurl(url)[0] | ||
Augie Fackler
|
r43347 | ui.status(_(b'comparing with %s\n') % util.hidepassword(url)) | ||
Pierre-Yves David
|
r23486 | |||
Yuya Nishihara
|
r24568 | revs = [r for r in revs if r >= 0] | ||
Pierre-Yves David
|
r23486 | if not revs: | ||
Boris Feld
|
r35692 | revs = [repo.changelog.tiprev()] | ||
Augie Fackler
|
r43347 | revs = repo.revs(b'outgoing(%s) and ::%ld', dest or b'', revs) | ||
Pierre-Yves David
|
r23486 | if not revs: | ||
Augie Fackler
|
r43347 | ui.status(_(b"no changes found\n")) | ||
Yuya Nishihara
|
r24567 | return revs | ||
Pierre-Yves David
|
r23486 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r39153 | def _msgid(node, timestamp): | ||
Augie Fackler
|
r39154 | hostname = encoding.strtolocal(socket.getfqdn()) | ||
Augie Fackler
|
r43347 | hostname = encoding.environ.get(b'HGHOSTNAME', hostname) | ||
return b'<%s.%d@%s>' % (node, timestamp, hostname) | ||||
Augie Fackler
|
r39153 | |||
Augie Fackler
|
r43346 | |||
Adrian Buehlmann
|
r14309 | emailopts = [ | ||
Augie Fackler
|
r43347 | (b'', b'body', None, _(b'send patches as inline message text (default)')), | ||
(b'a', b'attach', None, _(b'send patches as attachments')), | ||||
(b'i', b'inline', None, _(b'send patches as inline attachments')), | ||||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'', | ||
b'bcc', | ||||
[], | ||||
_(b'email addresses of blind carbon copy recipients'), | ||||
_(b'EMAIL'), | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | (b'c', b'cc', [], _(b'email addresses of copy recipients'), _(b'EMAIL')), | ||
(b'', b'confirm', None, _(b'ask for confirmation before sending')), | ||||
(b'd', b'diffstat', None, _(b'add diffstat output to messages')), | ||||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'', | ||
b'date', | ||||
b'', | ||||
_(b'use the given date as the sending date'), | ||||
_(b'DATE'), | ||||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'', | ||
b'desc', | ||||
b'', | ||||
_(b'use the given file as the series description'), | ||||
_(b'FILE'), | ||||
), | ||||
(b'f', b'from', b'', _(b'email address of sender'), _(b'EMAIL')), | ||||
(b'n', b'test', None, _(b'print messages that would be sent')), | ||||
( | ||||
b'm', | ||||
b'mbox', | ||||
b'', | ||||
_(b'write messages to mbox file instead of sending them'), | ||||
_(b'FILE'), | ||||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'', | ||
b'reply-to', | ||||
[], | ||||
_(b'email addresses replies should be sent to'), | ||||
_(b'EMAIL'), | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | ( | ||
b's', | ||||
b'subject', | ||||
b'', | ||||
_(b'subject of first message (intro or single patch)'), | ||||
_(b'TEXT'), | ||||
), | ||||
( | ||||
b'', | ||||
b'in-reply-to', | ||||
b'', | ||||
_(b'message identifier to reply to'), | ||||
_(b'MSGID'), | ||||
), | ||||
(b'', b'flag', [], _(b'flags to add in subject prefixes'), _(b'FLAG')), | ||||
(b't', b'to', [], _(b'email addresses of recipients'), _(b'EMAIL')), | ||||
Augie Fackler
|
r43346 | ] | ||
Adrian Buehlmann
|
r14309 | |||
Augie Fackler
|
r43346 | @command( | ||
Augie Fackler
|
r43347 | b'email', | ||
Augie Fackler
|
r43346 | [ | ||
Augie Fackler
|
r43347 | (b'g', b'git', None, _(b'use git extended diff format')), | ||
(b'', b'plain', None, _(b'omit hg patch header')), | ||||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'o', | ||
b'outgoing', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b'send changes not found in the target repository'), | ||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'b', | ||
b'bundle', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b'send changes not in target as a binary bundle'), | ||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'B', | ||
b'bookmark', | ||||
b'', | ||||
_(b'send changes only reachable by given bookmark'), | ||||
_(b'BOOKMARK'), | ||||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'', | ||
b'bundlename', | ||||
b'bundle', | ||||
_(b'name of the bundle attachment file'), | ||||
_(b'NAME'), | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | (b'r', b'rev', [], _(b'a revision to send'), _(b'REV')), | ||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'', | ||
b'force', | ||||
Augie Fackler
|
r43346 | None, | ||
_( | ||||
Augie Fackler
|
r43347 | b'run even when remote repository is unrelated ' | ||
b'(with -b/--bundle)' | ||||
Augie Fackler
|
r43346 | ), | ||
), | ||||
( | ||||
Augie Fackler
|
r43347 | b'', | ||
b'base', | ||||
Augie Fackler
|
r43346 | [], | ||
_( | ||||
Augie Fackler
|
r43347 | b'a base changeset to specify instead of a destination ' | ||
b'(with -b/--bundle)' | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | _(b'REV'), | ||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | ( | ||
b'', | ||||
b'intro', | ||||
None, | ||||
_(b'send an introduction email for a single patch'), | ||||
), | ||||
Augie Fackler
|
r43346 | ] | ||
+ emailopts | ||||
+ cmdutil.remoteopts, | ||||
Augie Fackler
|
r43347 | _(b'hg email [OPTION]... [DEST]...'), | ||
Augie Fackler
|
r43346 | helpcategory=command.CATEGORY_IMPORT_EXPORT, | ||
) | ||||
timeless
|
r27150 | def email(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 | ||||
timeless
|
r12749 | description. | ||
With the -d/--diffstat option, if the diffstat program is | ||||
installed, the result of running diffstat on the patch is inserted. | ||||
Finally, the patch itself, as generated by :hg:`export`. | ||||
Julian Cowley
|
r17880 | With the -d/--diffstat or --confirm options, you will be presented | ||
timeless
|
r12749 | with a final summary of all messages and asked for confirmation before | ||
the messages are sent. | ||||
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 | ||||
Angel Ezquerra
|
r16307 | will be created. You can include a patch both as text in the email | ||
body and as a regular or an inline attachment by combining the | ||||
-a/--attach or -i/--inline with the --body option. | ||||
Martin Geisler
|
r8472 | |||
David Demelier
|
r32639 | With -B/--bookmark changesets reachable by the given bookmark are | ||
selected. | ||||
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 | ||||
Pierre-Yves David
|
r26563 | will be sent. Use the ``patchbomb.bundletype`` config option to | ||
control the bundle type as with :hg:`bundle --type`. | ||||
John Goerzen
|
r4280 | |||
timeless
|
r12749 | With -m/--mbox, 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. | ||||
With -n/--test, all steps will run, but mail will not be sent. | ||||
You will be prompted for an email recipient address, a subject and | ||||
an introductory message describing the patches of your patchbomb. | ||||
Yuya Nishihara
|
r31489 | Then when all is done, patchbomb messages are displayed. | ||
timeless
|
r12749 | |||
Nicolas Dumazet
|
r13198 | In case email sending fails, you will find a backup of your series | ||
Pierre-Yves David
|
r23488 | introductory message in ``.hg/last-email.txt``. | ||
The default behavior of this command can be customized through | ||||
configuration. (See :hg:`help patchbomb` for details) | ||||
Nicolas Dumazet
|
r13198 | |||
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 | |||
David Demelier
|
r32639 | hg email -B feature # send all ancestors of feature bookmark | ||
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 | |||
timeless
|
r12749 | hg email -o -m mbox && # generate an mbox file... | ||
mutt -R -f mbox # ... and view it with mutt | ||||
hg email -o -m mbox && # generate an mbox file ... | ||||
Martin Geisler
|
r12839 | formail -s sendmail \\ # ... and use formail to send from the mbox | ||
timeless
|
r12749 | -bm -t < mbox # ... using sendmail | ||
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 | ''' | ||
Pulkit Goyal
|
r35035 | opts = pycompat.byteskwargs(opts) | ||
Brendan Cully
|
r4262 | |||
Christian Ebert
|
r7115 | _charsets = mail._charsets(ui) | ||
Augie Fackler
|
r43347 | bundle = opts.get(b'bundle') | ||
date = opts.get(b'date') | ||||
mbox = opts.get(b'mbox') | ||||
outgoing = opts.get(b'outgoing') | ||||
rev = opts.get(b'rev') | ||||
bookmark = opts.get(b'bookmark') | ||||
Christian Ebert
|
r11413 | |||
Augie Fackler
|
r43347 | if not (opts.get(b'test') or mbox): | ||
Christian Ebert
|
r5472 | # really sending | ||
Bryan O'Sullivan
|
r4489 | mail.validateconfig(ui) | ||
David Demelier
|
r32639 | if not (revs or rev or outgoing or bundle or bookmark): | ||
Augie Fackler
|
r43347 | raise error.Abort( | ||
_(b'specify at least one changeset with -B, -r or -o') | ||||
) | ||||
Bryan O'Sullivan
|
r4493 | |||
Christian Ebert
|
r11413 | if outgoing and bundle: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
_( | ||||
Augie Fackler
|
r43347 | b"--outgoing mode always on with --bundle;" | ||
b" do not re-specify --outgoing" | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
David Demelier
|
r32695 | if rev and bookmark: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b"-r and -B are mutually exclusive")) | ||
John Goerzen
|
r4278 | |||
Christian Ebert
|
r11413 | if outgoing or bundle: | ||
Brendan Cully
|
r4262 | if len(revs) > 1: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b"too many destinations")) | ||
Jordi Gutiérrez Hermoso
|
r24306 | if revs: | ||
dest = revs[0] | ||||
else: | ||||
dest = None | ||||
Brendan Cully
|
r4262 | revs = [] | ||
Christian Ebert
|
r11413 | if rev: | ||
Brendan Cully
|
r4262 | if revs: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b'use only one form to specify the revision')) | ||
Christian Ebert
|
r11413 | revs = rev | ||
David Demelier
|
r32639 | elif bookmark: | ||
if bookmark not in repo._bookmarks: | ||||
Augie Fackler
|
r43347 | raise error.Abort(_(b"bookmark '%s' not found") % bookmark) | ||
David Demelier
|
r38146 | revs = scmutil.bookmarkrevs(repo, bookmark) | ||
Brendan Cully
|
r4262 | |||
Yuya Nishihara
|
r24568 | revs = scmutil.revrange(repo, revs) | ||
Christian Ebert
|
r11413 | if outgoing: | ||
Yuya Nishihara
|
r24568 | revs = _getoutgoing(repo, dest, revs) | ||
Christian Ebert
|
r11413 | if bundle: | ||
Augie Fackler
|
r43347 | opts[b'revs'] = [b"%d" % r for r in revs] | ||
Brendan Cully
|
r4262 | |||
Pierre-Yves David
|
r26626 | # check if revision exist on the public destination | ||
Augie Fackler
|
r43347 | publicurl = repo.ui.config(b'patchbomb', b'publicurl') | ||
Augie Fackler
|
r32825 | if publicurl: | ||
Augie Fackler
|
r43347 | repo.ui.debug(b'checking that revision exist in the public repo\n') | ||
Pierre-Yves David
|
r26626 | try: | ||
publicpeer = hg.peer(repo, {}, publicurl) | ||||
except error.RepoError: | ||||
Augie Fackler
|
r43346 | repo.ui.write_err( | ||
Augie Fackler
|
r43347 | _(b'unable to access public repo: %s\n') % publicurl | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r26626 | raise | ||
Augie Fackler
|
r43347 | if not publicpeer.capable(b'known'): | ||
repo.ui.debug(b'skipping existence checks: public repo too old\n') | ||||
Pierre-Yves David
|
r26626 | else: | ||
out = [repo[r] for r in revs] | ||||
known = publicpeer.known(h.node() for h in out) | ||||
missing = [] | ||||
for idx, h in enumerate(out): | ||||
if not known[idx]: | ||||
missing.append(h) | ||||
if missing: | ||||
Martin von Zweigbergk
|
r40065 | if len(missing) > 1: | ||
Augie Fackler
|
r43347 | msg = _(b'public "%s" is missing %s and %i others') | ||
Pierre-Yves David
|
r26626 | msg %= (publicurl, missing[0], len(missing) - 1) | ||
else: | ||||
Augie Fackler
|
r43347 | msg = _(b'public url %s is missing %s') | ||
Pierre-Yves David
|
r26626 | msg %= (publicurl, missing[0]) | ||
Gregory Szorc
|
r36428 | missingrevs = [ctx.rev() for ctx in missing] | ||
Augie Fackler
|
r43347 | revhint = b' '.join( | ||
b'-r %s' % h for h in repo.set(b'heads(%ld)', missingrevs) | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | hint = _(b"use 'hg push %s %s'") % (publicurl, revhint) | ||
Pierre-Yves David
|
r26626 | raise error.Abort(msg, hint=hint) | ||
Brendan Cully
|
r4262 | # start | ||
Christian Ebert
|
r11413 | if date: | ||
Boris Feld
|
r36625 | start_time = dateutil.parsedate(date) | ||
Bryan O'Sullivan
|
r4566 | else: | ||
Boris Feld
|
r36625 | start_time = dateutil.makedate() | ||
Vadim Gelfer
|
r1669 | |||
def genmsgid(id): | ||||
Augie Fackler
|
r39153 | return _msgid(id[:20], int(start_time[0])) | ||
Vadim Gelfer
|
r1669 | |||
Matt Mackall
|
r25825 | # deprecated config: patchbomb.from | ||
Augie Fackler
|
r43346 | sender = ( | ||
Augie Fackler
|
r43347 | opts.get(b'from') | ||
or ui.config(b'email', b'from') | ||||
or ui.config(b'patchbomb', b'from') | ||||
or prompt(ui, b'From', ui.username()) | ||||
Augie Fackler
|
r43346 | ) | ||
Vadim Gelfer
|
r1669 | |||
Yuya Nishihara
|
r31184 | if bundle: | ||
Pulkit Goyal
|
r35035 | stropts = pycompat.strkwargs(opts) | ||
bundledata = _getbundle(repo, dest, **stropts) | ||||
bundleopts = stropts.copy() | ||||
bundleopts.pop(r'bundle', None) # already processed | ||||
Pierre-Yves David
|
r23213 | msgs = _getbundlemsgs(repo, sender, bundledata, **bundleopts) | ||
John Goerzen
|
r4278 | else: | ||
Pulkit Goyal
|
r35035 | msgs = _getpatchmsgs(repo, sender, revs, **pycompat.strkwargs(opts)) | ||
Vadim Gelfer
|
r1669 | |||
Christian Ebert
|
r12200 | showaddrs = [] | ||
Greg Ward
|
r15162 | def getaddrs(header, ask=False, default=None): | ||
configkey = header.lower() | ||||
Augie Fackler
|
r43347 | opt = header.replace(b'-', b'_').lower() | ||
Greg Ward
|
r15162 | addrs = opts.get(opt) | ||
Cédric Duval
|
r11150 | if addrs: | ||
Augie Fackler
|
r43347 | showaddrs.append(b'%s: %s' % (header, b', '.join(addrs))) | ||
return mail.addrlistencode(ui, addrs, _charsets, opts.get(b'test')) | ||||
Marti Raudsepp
|
r9947 | |||
Greg Ward
|
r15162 | # not on the command line: fallback to config and then maybe ask | ||
Augie Fackler
|
r43347 | addr = ui.config(b'email', configkey) or ui.config( | ||
b'patchbomb', configkey | ||||
Augie Fackler
|
r43346 | ) | ||
Bryan O'Sullivan
|
r27697 | if not addr: | ||
Augie Fackler
|
r43347 | specified = ui.hasconfig(b'email', configkey) or ui.hasconfig( | ||
b'patchbomb', configkey | ||||
Augie Fackler
|
r43346 | ) | ||
Bryan O'Sullivan
|
r27697 | if not specified and ask: | ||
addr = prompt(ui, header, default=default) | ||||
Greg Ward
|
r15162 | if addr: | ||
Augie Fackler
|
r43347 | showaddrs.append(b'%s: %s' % (header, addr)) | ||
return mail.addrlistencode(ui, [addr], _charsets, opts.get(b'test')) | ||||
Augie Fackler
|
r32826 | elif default: | ||
return mail.addrlistencode( | ||||
Augie Fackler
|
r43347 | ui, [default], _charsets, opts.get(b'test') | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r32826 | return [] | ||
Marti Raudsepp
|
r9947 | |||
Augie Fackler
|
r43347 | to = getaddrs(b'To', ask=True) | ||
Greg Ward
|
r15164 | if not to: | ||
# we can get here in non-interactive mode | ||||
Augie Fackler
|
r43347 | raise error.Abort(_(b'no recipient addresses provided')) | ||
cc = getaddrs(b'Cc', ask=True, default=b'') | ||||
bcc = getaddrs(b'Bcc') | ||||
replyto = getaddrs(b'Reply-To') | ||||
Christian Ebert
|
r2679 | |||
Augie Fackler
|
r43347 | confirm = ui.configbool(b'patchbomb', b'confirm') | ||
confirm |= bool(opts.get(b'diffstat') or opts.get(b'confirm')) | ||||
Pierre-Yves David
|
r23488 | |||
if confirm: | ||||
Augie Fackler
|
r43347 | ui.write(_(b'\nFinal summary:\n\n'), label=b'patchbomb.finalsummary') | ||
ui.write((b'From: %s\n' % sender), label=b'patchbomb.from') | ||||
Christian Ebert
|
r12200 | for addr in showaddrs: | ||
Augie Fackler
|
r43347 | ui.write(b'%s\n' % addr, label=b'patchbomb.to') | ||
Christian Ebert
|
r12200 | for m, subj, ds in msgs: | ||
Augie Fackler
|
r43347 | ui.write((b'Subject: %s\n' % subj), label=b'patchbomb.subject') | ||
Christian Ebert
|
r12200 | if ds: | ||
Augie Fackler
|
r43347 | ui.write(ds, label=b'patchbomb.diffstats') | ||
ui.write(b'\n') | ||||
Augie Fackler
|
r43346 | if ui.promptchoice( | ||
Martin von Zweigbergk
|
r43387 | _(b'are you sure you want to send (yn)?$$ &Yes $$ &No') | ||
Augie Fackler
|
r43346 | ): | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b'patchbomb canceled')) | ||
Christian Ebert
|
r12200 | |||
Augie Fackler
|
r43347 | ui.write(b'\n') | ||
Vadim Gelfer
|
r1669 | |||
Augie Fackler
|
r43347 | parent = opts.get(b'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: | ||||
Augie Fackler
|
r43347 | if not parent.startswith(b'<'): | ||
parent = b'<' + parent | ||||
if not parent.endswith(b'>'): | ||||
parent += b'>' | ||||
Cédric Duval
|
r8826 | |||
Pulkit Goyal
|
r36468 | sender_addr = eutil.parseaddr(encoding.strfromlocal(sender))[1] | ||
Augie Fackler
|
r43347 | sender = mail.addressencode(ui, sender, _charsets, opts.get(b'test')) | ||
Matt Mackall
|
r5973 | sendmail = None | ||
Augie Fackler
|
r21726 | firstpatch = None | ||
Augie Fackler
|
r43347 | progress = ui.makeprogress( | ||
_(b'sending'), unit=_(b'emails'), total=len(msgs) | ||||
) | ||||
Yuya Nishihara
|
r12265 | for i, (m, subj, ds) in enumerate(msgs): | ||
Vadim Gelfer
|
r1669 | try: | ||
Augie Fackler
|
r43347 | m[b'Message-Id'] = genmsgid(m[b'X-Mercurial-Node']) | ||
Augie Fackler
|
r21726 | if not firstpatch: | ||
Augie Fackler
|
r43347 | firstpatch = m[b'Message-Id'] | ||
m[b'X-Mercurial-Series-Id'] = firstpatch | ||||
Vadim Gelfer
|
r1669 | except TypeError: | ||
Augie Fackler
|
r43347 | m[b'Message-Id'] = genmsgid(b'patchbomb') | ||
Vadim Gelfer
|
r1669 | if parent: | ||
Augie Fackler
|
r43347 | m[b'In-Reply-To'] = parent | ||
m[b'References'] = parent | ||||
if not parent or b'X-Mercurial-Node' not in m: | ||||
parent = m[b'Message-Id'] | ||||
Cédric Duval
|
r8514 | |||
Augie Fackler
|
r43347 | m[b'User-Agent'] = b'Mercurial-patchbomb/%s' % util.version() | ||
m[b'Date'] = eutil.formatdate(start_time[0], localtime=True) | ||||
Volker Kleinfeld
|
r2443 | |||
Christian Ebert
|
r4027 | start_time = (start_time[0] + 1, start_time[1]) | ||
Augie Fackler
|
r43347 | m[b'From'] = sender | ||
m[b'To'] = b', '.join(to) | ||||
Christian Ebert
|
r5785 | if cc: | ||
Augie Fackler
|
r43347 | m[b'Cc'] = b', '.join(cc) | ||
Christian Ebert
|
r5785 | if bcc: | ||
Augie Fackler
|
r43347 | m[b'Bcc'] = b', '.join(bcc) | ||
Cédric Duval
|
r11150 | if replyto: | ||
Augie Fackler
|
r43347 | m[b'Reply-To'] = b', '.join(replyto) | ||
Augie Fackler
|
r38799 | # Fix up all headers to be native strings. | ||
# TODO(durin42): this should probably be cleaned up above in the future. | ||||
if pycompat.ispy3: | ||||
for hdr, val in list(m.items()): | ||||
Augie Fackler
|
r39070 | change = False | ||
Augie Fackler
|
r38799 | if isinstance(hdr, bytes): | ||
del m[hdr] | ||||
hdr = pycompat.strurl(hdr) | ||||
Augie Fackler
|
r39070 | change = True | ||
Augie Fackler
|
r38799 | if isinstance(val, bytes): | ||
val = pycompat.strurl(val) | ||||
Augie Fackler
|
r39070 | if not change: | ||
# prevent duplicate headers | ||||
del m[hdr] | ||||
change = True | ||||
if change: | ||||
m[hdr] = val | ||||
Augie Fackler
|
r43347 | if opts.get(b'test'): | ||
ui.status(_(b'displaying '), subj, b' ...\n') | ||||
ui.pager(b'email') | ||||
Yuya Nishihara
|
r39140 | generator = _bytesgenerator(ui, mangle_from_=False) | ||
Vadim Gelfer
|
r1871 | try: | ||
Benoit Boissinot
|
r6447 | generator.flatten(m, 0) | ||
Augie Fackler
|
r43347 | ui.write(b'\n') | ||
Gregory Szorc
|
r25660 | except IOError as inst: | ||
Vadim Gelfer
|
r1871 | if inst.errno != errno.EPIPE: | ||
raise | ||||
Mads Kiilerich
|
r15560 | else: | ||
if not sendmail: | ||||
Gregory Szorc
|
r29285 | sendmail = mail.connect(ui, mbox=mbox) | ||
Augie Fackler
|
r43347 | ui.status(_(b'sending '), subj, b' ...\n') | ||
Martin von Zweigbergk
|
r38422 | progress.update(i, item=subj) | ||
Mads Kiilerich
|
r15560 | if not mbox: | ||
# Exim does not remove the Bcc field | ||||
Augie Fackler
|
r43347 | del m[b'Bcc'] | ||
timeless
|
r28861 | fp = stringio() | ||
Yuya Nishihara
|
r39140 | generator = _bytesgenerator(fp, mangle_from_=False) | ||
Benoit Boissinot
|
r6447 | generator.flatten(m, 0) | ||
Augie Fackler
|
r39062 | alldests = to + bcc + cc | ||
alldests = [encoding.strfromlocal(d) for d in alldests] | ||||
sendmail(sender_addr, alldests, fp.getvalue()) | ||||
Vadim Gelfer
|
r1669 | |||
Martin von Zweigbergk
|
r38422 | progress.complete() | ||