notify.py
656 lines
| 20.0 KiB
| text/x-python
|
PythonLexer
/ hgext / notify.py
Vadim Gelfer
|
r2203 | # notify.py - email notifications for mercurial | ||
# | ||||
# Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com> | ||||
# | ||||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Dirkjan Ochtman
|
r7127 | |||
Patrick Mezard
|
r14940 | '''hooks for sending email push notifications | ||
Dirkjan Ochtman
|
r7127 | |||
David Champion
|
r16950 | This extension implements hooks to send email notifications when | ||
changesets are sent from or received by the local repository. | ||||
Dirkjan Ochtman
|
r7127 | |||
Patrick Mezard
|
r14940 | First, enable the extension as explained in :hg:`help extensions`, and | ||
FUJIWARA Katsunori
|
r16500 | register the hook you want to run. ``incoming`` and ``changegroup`` hooks | ||
David Champion
|
r16950 | are run when changesets are received, while ``outgoing`` hooks are for | ||
changesets sent to another repository:: | ||||
Dirkjan Ochtman
|
r7127 | |||
Martin Geisler
|
r9105 | [hooks] | ||
# one email for each incoming changeset | ||||
incoming.notify = python:hgext.notify.hook | ||||
Patrick Mezard
|
r14940 | # one email for all incoming changesets | ||
Martin Geisler
|
r9105 | changegroup.notify = python:hgext.notify.hook | ||
Patrick Mezard
|
r14940 | |||
# one email for all outgoing changesets | ||||
Ingo Bressler
|
r14617 | outgoing.notify = python:hgext.notify.hook | ||
Dirkjan Ochtman
|
r7127 | |||
David Champion
|
r16950 | This registers the hooks. To enable notification, subscribers must | ||
be assigned to repositories. The ``[usersubs]`` section maps multiple | ||||
repositories to a given recipient. The ``[reposubs]`` section maps | ||||
multiple recipients to a single repository:: | ||||
Martin Geisler
|
r9157 | |||
Martin Geisler
|
r9105 | [usersubs] | ||
Michal Sznajder
|
r17754 | # key is subscriber email, value is a comma-separated list of repo patterns | ||
Martin Geisler
|
r9105 | user@host = pattern | ||
Dirkjan Ochtman
|
r7127 | |||
Martin Geisler
|
r9105 | [reposubs] | ||
Michal Sznajder
|
r17754 | # key is repo pattern, value is a comma-separated list of subscriber emails | ||
Martin Geisler
|
r9105 | pattern = user@host | ||
Dirkjan Ochtman
|
r7127 | |||
Michal Sznajder
|
r17754 | A ``pattern`` is a ``glob`` matching the absolute path to a repository, | ||
optionally combined with a revset expression. A revset expression, if | ||||
present, is separated from the glob by a hash. Example:: | ||||
[reposubs] | ||||
*/widgets#branch(release) = qa-team@example.com | ||||
This sends to ``qa-team@example.com`` whenever a changeset on the ``release`` | ||||
branch triggers a notification in any repository ending in ``widgets``. | ||||
David Champion
|
r16950 | |||
In order to place them under direct user management, ``[usersubs]`` and | ||||
``[reposubs]`` sections may be placed in a separate ``hgrc`` file and | ||||
incorporated by reference:: | ||||
Patrick Mezard
|
r14940 | |||
[notify] | ||||
config = /path/to/subscriptionsfile | ||||
David Champion
|
r16950 | Notifications will not be sent until the ``notify.test`` value is set | ||
to ``False``; see below. | ||||
Patrick Mezard
|
r14940 | |||
Notifications content can be tweaked with the following configuration entries: | ||||
notify.test | ||||
If ``True``, print messages to stdout instead of sending them. Default: True. | ||||
notify.sources | ||||
David Champion
|
r16950 | Space-separated list of change sources. Notifications are activated only | ||
when a changeset's source is in this list. Sources may be: | ||||
:``serve``: changesets received via http or ssh | ||||
:``pull``: changesets received via ``hg pull`` | ||||
:``unbundle``: changesets received via ``hg unbundle`` | ||||
:``push``: changesets sent or received via ``hg push`` | ||||
:``bundle``: changesets sent via ``hg unbundle`` | ||||
Default: serve. | ||||
Patrick Mezard
|
r14940 | |||
notify.strip | ||||
Number of leading slashes to strip from url paths. By default, notifications | ||||
David Champion
|
r16950 | reference repositories with their absolute path. ``notify.strip`` lets you | ||
Patrick Mezard
|
r14940 | turn them into relative paths. For example, ``notify.strip=3`` will change | ||
``/long/path/repository`` into ``repository``. Default: 0. | ||||
notify.domain | ||||
David Champion
|
r16950 | Default email domain for sender or recipients with no explicit domain. | ||
Joerg Sonnenberger
|
r43174 | It is also used for the domain part of the ``Message-Id`` when using | ||
``notify.messageidseed``. | ||||
notify.messageidseed | ||||
Create deterministic ``Message-Id`` headers for the mails based on the seed | ||||
and the revision identifier of the first commit in the changeset. | ||||
Dirkjan Ochtman
|
r7127 | |||
Patrick Mezard
|
r14940 | notify.style | ||
Style file to use when formatting emails. | ||||
notify.template | ||||
Template to use when formatting emails. | ||||
notify.incoming | ||||
David Champion
|
r16950 | Template to use when run as an incoming hook, overriding ``notify.template``. | ||
Patrick Mezard
|
r14940 | |||
notify.outgoing | ||||
David Champion
|
r16950 | Template to use when run as an outgoing hook, overriding ``notify.template``. | ||
Patrick Mezard
|
r14940 | |||
notify.changegroup | ||||
David Champion
|
r16950 | Template to use when running as a changegroup hook, overriding | ||
Patrick Mezard
|
r14940 | ``notify.template``. | ||
notify.maxdiff | ||||
Maximum number of diff lines to include in notification email. Set to 0 | ||||
David Champion
|
r16950 | to disable the diff, or -1 to include all of it. Default: 300. | ||
Patrick Mezard
|
r14940 | |||
Joerg Sonnenberger
|
r37795 | notify.maxdiffstat | ||
Maximum number of diffstat lines to include in notification email. Set to -1 | ||||
to include all of it. Default: -1. | ||||
Patrick Mezard
|
r14940 | notify.maxsubject | ||
David Champion
|
r16950 | Maximum number of characters in email's subject line. Default: 67. | ||
Patrick Mezard
|
r14940 | |||
notify.diffstat | ||||
Set to True to include a diffstat before diff content. Default: True. | ||||
Joerg Sonnenberger
|
r38048 | notify.showfunc | ||
If set, override ``diff.showfunc`` for the diff content. Default: None. | ||||
Patrick Mezard
|
r14940 | notify.merge | ||
If True, send notifications for merge changesets. Default: True. | ||||
Mads Kiilerich
|
r15561 | notify.mbox | ||
If set, append mails to this mbox file instead of sending. Default: None. | ||||
Nikolaus Schueler
|
r15654 | notify.fromauthor | ||
David Champion
|
r16950 | If set, use the committer of the first changeset in a changegroup for | ||
the "From" field of the notification mail. If not set, take the user | ||||
from the pushing repo. Default: False. | ||||
Nikolaus Schueler
|
r15654 | |||
Joerg Sonnenberger
|
r45117 | notify.reply-to-predecessor (EXPERIMENTAL) | ||
If set and the changeset has a predecessor in the repository, try to thread | ||||
the notification mail with the predecessor. This adds the "In-Reply-To" header | ||||
to the notification mail with a reference to the predecessor with the smallest | ||||
revision number. Mail threads can still be torn, especially when changesets | ||||
are folded. | ||||
This option must be used in combination with ``notify.messageidseed``. | ||||
David Champion
|
r16950 | If set, the following entries will also be used to customize the | ||
notifications: | ||||
Patrick Mezard
|
r14940 | |||
email.from | ||||
David Champion
|
r16950 | Email ``From`` address to use if none can be found in the generated | ||
email content. | ||||
Patrick Mezard
|
r14940 | |||
web.baseurl | ||||
David Champion
|
r16950 | Root repository URL to combine with repository paths when making | ||
Patrick Mezard
|
r14940 | references. See also ``notify.strip``. | ||
Martin Geisler
|
r9068 | ''' | ||
timeless
|
r28416 | from __future__ import absolute_import | ||
Vadim Gelfer
|
r2203 | |||
Yuya Nishihara
|
r40326 | import email.errors as emailerrors | ||
Denis Laxalde
|
r43635 | import email.utils as emailutils | ||
timeless
|
r28416 | import fnmatch | ||
Joerg Sonnenberger
|
r43174 | import hashlib | ||
timeless
|
r28416 | import socket | ||
import time | ||||
Yuya Nishihara
|
r29205 | from mercurial.i18n import _ | ||
timeless
|
r28416 | from mercurial import ( | ||
Augie Fackler
|
r40340 | encoding, | ||
timeless
|
r28416 | error, | ||
Yuya Nishihara
|
r35906 | logcmdutil, | ||
timeless
|
r28416 | mail, | ||
Joerg Sonnenberger
|
r45117 | obsutil, | ||
timeless
|
r28416 | patch, | ||
Gregory Szorc
|
r43434 | pycompat, | ||
Boris Feld
|
r33738 | registrar, | ||
timeless
|
r28416 | util, | ||
) | ||||
Yuya Nishihara
|
r37102 | from mercurial.utils import ( | ||
dateutil, | ||||
stringutil, | ||||
) | ||||
Vadim Gelfer
|
r2203 | |||
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' | ||
Augie Fackler
|
r16743 | |||
Boris Feld
|
r33738 | configtable = {} | ||
configitem = registrar.configitem(configtable) | ||||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'changegroup', | ||||
default=None, | ||||
Boris Feld
|
r34755 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'config', | ||||
default=None, | ||||
Boris Feld
|
r33738 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'diffstat', | ||||
default=True, | ||||
Boris Feld
|
r33739 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'domain', | ||||
default=None, | ||||
Boris Feld
|
r33740 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'messageidseed', | ||||
default=None, | ||||
Joerg Sonnenberger
|
r43174 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'fromauthor', | ||||
default=None, | ||||
Boris Feld
|
r33741 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'incoming', | ||||
default=None, | ||||
Boris Feld
|
r34753 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'maxdiff', | ||||
default=300, | ||||
Boris Feld
|
r33742 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'maxdiffstat', | ||||
default=-1, | ||||
Joerg Sonnenberger
|
r37795 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'maxsubject', | ||||
default=67, | ||||
Boris Feld
|
r33743 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'mbox', | ||||
default=None, | ||||
Boris Feld
|
r33744 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'merge', | ||||
default=True, | ||||
Boris Feld
|
r33745 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'outgoing', | ||||
default=None, | ||||
Boris Feld
|
r34754 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'reply-to-predecessor', | ||||
default=False, | ||||
Joerg Sonnenberger
|
r45117 | ) | ||
configitem( | ||||
Augie Fackler
|
r46554 | b'notify', | ||
b'sources', | ||||
default=b'serve', | ||||
Boris Feld
|
r33746 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'showfunc', | ||||
default=None, | ||||
Joerg Sonnenberger
|
r38048 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'strip', | ||||
default=0, | ||||
Boris Feld
|
r33747 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'style', | ||||
default=None, | ||||
Boris Feld
|
r33748 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'template', | ||||
default=None, | ||||
Boris Feld
|
r33749 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'notify', | ||
b'test', | ||||
default=True, | ||||
Boris Feld
|
r33750 | ) | ||
Boris Feld
|
r33738 | |||
Vadim Gelfer
|
r2203 | # template for single changeset can include email headers. | ||
Augie Fackler
|
r40319 | single_template = b''' | ||
Vadim Gelfer
|
r2203 | Subject: changeset in {webroot}: {desc|firstline|strip} | ||
From: {author} | ||||
changeset {node|short} in {root} | ||||
details: {baseurl}{webroot}?cmd=changeset;node={node|short} | ||||
description: | ||||
\t{desc|tabindent|strip} | ||||
'''.lstrip() | ||||
# template for multiple changesets should not contain email headers, | ||||
# because only first set of headers will be used and result will look | ||||
# strange. | ||||
Augie Fackler
|
r40319 | multiple_template = b''' | ||
Vadim Gelfer
|
r2203 | changeset {node|short} in {root} | ||
details: {baseurl}{webroot}?cmd=changeset;node={node|short} | ||||
summary: {desc|firstline} | ||||
''' | ||||
deftemplates = { | ||||
Augie Fackler
|
r43347 | b'changegroup': multiple_template, | ||
Thomas Arendsen Hein
|
r4498 | } | ||
Bryan O'Sullivan
|
r2201 | |||
Augie Fackler
|
r43346 | |||
Bryan O'Sullivan
|
r2201 | class notifier(object): | ||
Vadim Gelfer
|
r2203 | '''email notification class.''' | ||
def __init__(self, ui, repo, hooktype): | ||||
Bryan O'Sullivan
|
r2201 | self.ui = ui | ||
Augie Fackler
|
r43347 | cfg = self.ui.config(b'notify', b'config') | ||
Vadim Gelfer
|
r2329 | if cfg: | ||
Augie Fackler
|
r43347 | self.ui.readconfig(cfg, sections=[b'usersubs', b'reposubs']) | ||
Bryan O'Sullivan
|
r2201 | self.repo = repo | ||
Augie Fackler
|
r43347 | self.stripcount = int(self.ui.config(b'notify', b'strip')) | ||
Bryan O'Sullivan
|
r2201 | self.root = self.strip(self.repo.root) | ||
Augie Fackler
|
r43347 | self.domain = self.ui.config(b'notify', b'domain') | ||
self.mbox = self.ui.config(b'notify', b'mbox') | ||||
self.test = self.ui.configbool(b'notify', b'test') | ||||
Christian Ebert
|
r7116 | self.charsets = mail._charsets(self.ui) | ||
Vadim Gelfer
|
r2203 | self.subs = self.subscribers() | ||
Augie Fackler
|
r43347 | self.merge = self.ui.configbool(b'notify', b'merge') | ||
self.showfunc = self.ui.configbool(b'notify', b'showfunc') | ||||
self.messageidseed = self.ui.config(b'notify', b'messageidseed') | ||||
Joerg Sonnenberger
|
r45117 | self.reply = self.ui.configbool(b'notify', b'reply-to-predecessor') | ||
if self.reply and not self.messageidseed: | ||||
raise error.Abort( | ||||
_( | ||||
b'notify.reply-to-predecessor used without ' | ||||
b'notify.messageidseed' | ||||
) | ||||
) | ||||
Joerg Sonnenberger
|
r38048 | if self.showfunc is None: | ||
Augie Fackler
|
r43347 | self.showfunc = self.ui.configbool(b'diff', b'showfunc') | ||
Vadim Gelfer
|
r2203 | |||
Yuya Nishihara
|
r28951 | mapfile = None | ||
Augie Fackler
|
r43347 | template = self.ui.config(b'notify', hooktype) or self.ui.config( | ||
b'notify', b'template' | ||||
Augie Fackler
|
r43346 | ) | ||
Yuya Nishihara
|
r28951 | if not template: | ||
Augie Fackler
|
r43347 | mapfile = self.ui.config(b'notify', b'style') | ||
Vadim Gelfer
|
r2203 | if not mapfile and not template: | ||
template = deftemplates.get(hooktype) or single_template | ||||
Yuya Nishihara
|
r35906 | spec = logcmdutil.templatespec(template, mapfile) | ||
Yuya Nishihara
|
r35972 | self.t = logcmdutil.changesettemplater(self.ui, self.repo, spec) | ||
Bryan O'Sullivan
|
r2201 | |||
def strip(self, path): | ||||
Vadim Gelfer
|
r2203 | '''strip leading slashes from local path, turn into web-safe path.''' | ||
Bryan O'Sullivan
|
r2201 | path = util.pconvert(path) | ||
count = self.stripcount | ||||
Vadim Gelfer
|
r2326 | while count > 0: | ||
Augie Fackler
|
r43347 | c = path.find(b'/') | ||
Bryan O'Sullivan
|
r2201 | if c == -1: | ||
break | ||||
Augie Fackler
|
r43346 | path = path[c + 1 :] | ||
Bryan O'Sullivan
|
r2201 | count -= 1 | ||
return path | ||||
Vadim Gelfer
|
r2203 | def fixmail(self, addr): | ||
'''try to clean up email addresses.''' | ||||
Yuya Nishihara
|
r37102 | addr = stringutil.email(addr.strip()) | ||
Alexis S. L. Carvalho
|
r4094 | if self.domain: | ||
Augie Fackler
|
r43347 | a = addr.find(b'@localhost') | ||
Alexis S. L. Carvalho
|
r4094 | if a != -1: | ||
addr = addr[:a] | ||||
Augie Fackler
|
r43347 | if b'@' not in addr: | ||
return addr + b'@' + self.domain | ||||
Vadim Gelfer
|
r2203 | return addr | ||
Bryan O'Sullivan
|
r2201 | def subscribers(self): | ||
Vadim Gelfer
|
r2203 | '''return list of email addresses of subscribers to this repo.''' | ||
Martin Geisler
|
r8154 | subs = set() | ||
Augie Fackler
|
r43347 | for user, pats in self.ui.configitems(b'usersubs'): | ||
for pat in pats.split(b','): | ||||
if b'#' in pat: | ||||
pat, revs = pat.split(b'#', 1) | ||||
Michal Sznajder
|
r17754 | else: | ||
revs = None | ||||
Vadim Gelfer
|
r2203 | if fnmatch.fnmatch(self.repo.root, pat.strip()): | ||
Michal Sznajder
|
r17754 | subs.add((self.fixmail(user), revs)) | ||
Augie Fackler
|
r43347 | for pat, users in self.ui.configitems(b'reposubs'): | ||
if b'#' in pat: | ||||
pat, revs = pat.split(b'#', 1) | ||||
Michal Sznajder
|
r17754 | else: | ||
revs = None | ||||
Vadim Gelfer
|
r2203 | if fnmatch.fnmatch(self.repo.root, pat): | ||
Augie Fackler
|
r43347 | for user in users.split(b','): | ||
Michal Sznajder
|
r17754 | subs.add((self.fixmail(user), revs)) | ||
Augie Fackler
|
r43346 | return [ | ||
(mail.addressencode(self.ui, s, self.charsets, self.test), r) | ||||
for s, r in sorted(subs) | ||||
] | ||||
Bryan O'Sullivan
|
r2201 | |||
Bryan O'Sullivan
|
r9486 | def node(self, ctx, **props): | ||
David Champion
|
r9516 | '''format one changeset, unless it is a suppressed merge.''' | ||
if not self.merge and len(ctx.parents()) > 1: | ||||
return False | ||||
Augie Fackler
|
r43346 | self.t.show( | ||
ctx, | ||||
changes=ctx.changeset(), | ||||
Augie Fackler
|
r43347 | baseurl=self.ui.config(b'web', b'baseurl'), | ||
Augie Fackler
|
r43346 | root=self.repo.root, | ||
webroot=self.root, | ||||
**props | ||||
) | ||||
David Champion
|
r9516 | return True | ||
Vadim Gelfer
|
r2203 | |||
Vadim Gelfer
|
r2230 | def skipsource(self, source): | ||
'''true if incoming changes from this source should be skipped.''' | ||||
Augie Fackler
|
r43347 | ok_sources = self.ui.config(b'notify', b'sources').split() | ||
Vadim Gelfer
|
r2230 | return source not in ok_sources | ||
Dirkjan Ochtman
|
r7726 | def send(self, ctx, count, data): | ||
Vadim Gelfer
|
r2203 | '''send message.''' | ||
Michal Sznajder
|
r17754 | # Select subscribers by revset | ||
subs = set() | ||||
for sub, spec in self.subs: | ||||
if spec is None: | ||||
subs.add(sub) | ||||
continue | ||||
Augie Fackler
|
r43347 | revs = self.repo.revs(b'%r and %d:', spec, ctx.rev()) | ||
Michal Sznajder
|
r17754 | if len(revs): | ||
subs.add(sub) | ||||
continue | ||||
if len(subs) == 0: | ||||
Augie Fackler
|
r43346 | self.ui.debug( | ||
Martin von Zweigbergk
|
r43387 | b'notify: no subscribers to selected repo and revset\n' | ||
Augie Fackler
|
r43346 | ) | ||
Michal Sznajder
|
r17754 | return | ||
Christian Ebert
|
r9313 | try: | ||
Denis Laxalde
|
r43634 | msg = mail.parsebytes(data) | ||
Augie Fackler
|
r40320 | except emailerrors.MessageParseError as inst: | ||
Pierre-Yves David
|
r26587 | raise error.Abort(inst) | ||
Vadim Gelfer
|
r2203 | |||
Christian Ebert
|
r7116 | # store sender and subject | ||
Augie Fackler
|
r43906 | sender = msg['From'] | ||
subject = msg['Subject'] | ||||
Gregory Szorc
|
r41449 | if sender is not None: | ||
Denis Laxalde
|
r43636 | sender = mail.headdecode(sender) | ||
Gregory Szorc
|
r41449 | if subject is not None: | ||
Denis Laxalde
|
r43636 | subject = mail.headdecode(subject) | ||
Augie Fackler
|
r43906 | del msg['From'], msg['Subject'] | ||
Christian Ebert
|
r9313 | |||
if not msg.is_multipart(): | ||||
# create fresh mime message from scratch | ||||
# (multipart templates must take care of this themselves) | ||||
headers = msg.items() | ||||
Denis Laxalde
|
r43637 | payload = msg.get_payload(decode=pycompat.ispy3) | ||
Christian Ebert
|
r9313 | # for notification prefer readability over data precision | ||
msg = mail.mimeencode(self.ui, payload, self.charsets, self.test) | ||||
# reinstate custom headers | ||||
for k, v in headers: | ||||
msg[k] = v | ||||
Christian Ebert
|
r7116 | |||
Augie Fackler
|
r43906 | msg['Date'] = encoding.strfromlocal( | ||
Augie Fackler
|
r43347 | dateutil.datestr(format=b"%a, %d %b %Y %H:%M:%S %1%2") | ||
Augie Fackler
|
r43346 | ) | ||
Vadim Gelfer
|
r2203 | |||
Christian Ebert
|
r7705 | # try to make subject line exist and be useful | ||
if not subject: | ||||
if count > 1: | ||||
Augie Fackler
|
r43347 | subject = _(b'%s: %d new changesets') % (self.root, count) | ||
Christian Ebert
|
r7705 | else: | ||
Augie Fackler
|
r43347 | s = ctx.description().lstrip().split(b'\n', 1)[0].rstrip() | ||
subject = b'%s: %s' % (self.root, s) | ||||
maxsubject = int(self.ui.config(b'notify', b'maxsubject')) | ||||
Yuya Nishihara
|
r13202 | if maxsubject: | ||
Yuya Nishihara
|
r37102 | subject = stringutil.ellipsis(subject, maxsubject) | ||
Denis Laxalde
|
r43975 | msg['Subject'] = mail.headencode( | ||
self.ui, subject, self.charsets, self.test | ||||
Augie Fackler
|
r43346 | ) | ||
Vadim Gelfer
|
r2203 | |||
Christian Ebert
|
r7705 | # try to make message have proper sender | ||
if not sender: | ||||
Augie Fackler
|
r43347 | sender = self.ui.config(b'email', b'from') or self.ui.username() | ||
if b'@' not in sender or b'@localhost' in sender: | ||||
Christian Ebert
|
r7705 | sender = self.fixmail(sender) | ||
Denis Laxalde
|
r43976 | msg['From'] = mail.addressencode( | ||
self.ui, sender, self.charsets, self.test | ||||
Augie Fackler
|
r43346 | ) | ||
Vadim Gelfer
|
r2203 | |||
Augie Fackler
|
r43906 | msg['X-Hg-Notification'] = 'changeset %s' % ctx | ||
if not msg['Message-Id']: | ||||
msg['Message-Id'] = messageid(ctx, self.domain, self.messageidseed) | ||||
Joerg Sonnenberger
|
r45117 | if self.reply: | ||
unfi = self.repo.unfiltered() | ||||
has_node = unfi.changelog.index.has_node | ||||
predecessors = [ | ||||
unfi[ctx2] | ||||
for ctx2 in obsutil.allpredecessors(unfi.obsstore, [ctx.node()]) | ||||
if ctx2 != ctx.node() and has_node(ctx2) | ||||
] | ||||
if predecessors: | ||||
# There is at least one predecessor, so which to pick? | ||||
# Ideally, there is a unique root because changesets have | ||||
# been evolved/rebased one step at a time. In this case, | ||||
# just picking the oldest known changeset provides a stable | ||||
# base. It doesn't help when changesets are folded. Any | ||||
# better solution would require storing more information | ||||
# in the repository. | ||||
pred = min(predecessors, key=lambda ctx: ctx.rev()) | ||||
msg['In-Reply-To'] = messageid( | ||||
pred, self.domain, self.messageidseed | ||||
) | ||||
Denis Laxalde
|
r43976 | msg['To'] = ', '.join(sorted(subs)) | ||
Vadim Gelfer
|
r2203 | |||
Denis Laxalde
|
r43630 | msgtext = msg.as_bytes() if pycompat.ispy3 else msg.as_string() | ||
Christian Ebert
|
r7498 | if self.test: | ||
Vadim Gelfer
|
r2203 | self.ui.write(msgtext) | ||
Augie Fackler
|
r43347 | if not msgtext.endswith(b'\n'): | ||
self.ui.write(b'\n') | ||||
Bryan O'Sullivan
|
r2201 | else: | ||
Augie Fackler
|
r43346 | self.ui.status( | ||
Augie Fackler
|
r43347 | _(b'notify: sending %d subscribers %d changes\n') | ||
Augie Fackler
|
r43346 | % (len(subs), count) | ||
) | ||||
mail.sendmail( | ||||
self.ui, | ||||
Augie Fackler
|
r43906 | emailutils.parseaddr(msg['From'])[1], | ||
Augie Fackler
|
r43346 | subs, | ||
msgtext, | ||||
mbox=self.mbox, | ||||
) | ||||
Bryan O'Sullivan
|
r2201 | |||
Dirkjan Ochtman
|
r7726 | def diff(self, ctx, ref=None): | ||
Augie Fackler
|
r43347 | maxdiff = int(self.ui.config(b'notify', b'maxdiff')) | ||
Matt Mackall
|
r13878 | prev = ctx.p1().node() | ||
Jordi Gutiérrez Hermoso
|
r24306 | if ref: | ||
ref = ref.node() | ||||
else: | ||||
ref = ctx.node() | ||||
Joerg Sonnenberger
|
r38048 | diffopts = patch.diffallopts(self.ui) | ||
diffopts.showfunc = self.showfunc | ||||
chunks = patch.diff(self.repo, prev, ref, opts=diffopts) | ||||
Augie Fackler
|
r43347 | difflines = b''.join(chunks).splitlines() | ||
divy@chelsio.com
|
r6979 | |||
Augie Fackler
|
r43347 | if self.ui.configbool(b'notify', b'diffstat'): | ||
maxdiffstat = int(self.ui.config(b'notify', b'maxdiffstat')) | ||||
Matt Doar
|
r3096 | s = patch.diffstat(difflines) | ||
Sean Dague
|
r4077 | # s may be nil, don't include the header if it is | ||
if s: | ||||
Augie Fackler
|
r43347 | if maxdiffstat >= 0 and s.count(b"\n") > maxdiffstat + 1: | ||
s = s.split(b"\n") | ||||
msg = _(b'\ndiffstat (truncated from %d to %d lines):\n\n') | ||||
Joerg Sonnenberger
|
r37795 | self.ui.write(msg % (len(s) - 2, maxdiffstat)) | ||
Augie Fackler
|
r43347 | self.ui.write(b"\n".join(s[:maxdiffstat] + s[-2:])) | ||
Joerg Sonnenberger
|
r37795 | else: | ||
Augie Fackler
|
r43347 | self.ui.write(_(b'\ndiffstat:\n\n%s') % s) | ||
Dirkjan Ochtman
|
r7726 | |||
Benoît Allard
|
r6305 | if maxdiff == 0: | ||
return | ||||
Dirkjan Ochtman
|
r7726 | elif maxdiff > 0 and len(difflines) > maxdiff: | ||
Augie Fackler
|
r43347 | msg = _(b'\ndiffs (truncated from %d to %d lines):\n\n') | ||
Dirkjan Ochtman
|
r7726 | self.ui.write(msg % (len(difflines), maxdiff)) | ||
Vadim Gelfer
|
r2203 | difflines = difflines[:maxdiff] | ||
elif difflines: | ||||
Augie Fackler
|
r43347 | self.ui.write(_(b'\ndiffs (%d lines):\n\n') % len(difflines)) | ||
Dirkjan Ochtman
|
r7726 | |||
Augie Fackler
|
r43347 | self.ui.write(b"\n".join(difflines)) | ||
Bryan O'Sullivan
|
r2201 | |||
Augie Fackler
|
r43346 | |||
Vadim Gelfer
|
r2230 | def hook(ui, repo, hooktype, node=None, source=None, **kwargs): | ||
Augie Fackler
|
r46554 | """send email notifications to interested subscribers. | ||
Vadim Gelfer
|
r2203 | |||
if used as changegroup hook, send one email for all changesets in | ||||
Augie Fackler
|
r46554 | changegroup. else send one email per changeset.""" | ||
Dirkjan Ochtman
|
r7726 | |||
Vadim Gelfer
|
r2203 | n = notifier(ui, repo, hooktype) | ||
Boris Feld
|
r37812 | ctx = repo.unfiltered()[node] | ||
Dirkjan Ochtman
|
r7726 | |||
Vadim Gelfer
|
r2329 | if not n.subs: | ||
Augie Fackler
|
r43347 | ui.debug(b'notify: no subscribers to repository %s\n' % n.root) | ||
Vadim Gelfer
|
r2329 | return | ||
if n.skipsource(source): | ||||
Augie Fackler
|
r43347 | ui.debug(b'notify: changes have source "%s" - skipping\n' % source) | ||
Vadim Gelfer
|
r2230 | return | ||
Dirkjan Ochtman
|
r7726 | |||
Matt Mackall
|
r3739 | ui.pushbuffer() | ||
Augie Fackler
|
r43347 | data = b'' | ||
David Champion
|
r9516 | count = 0 | ||
Augie Fackler
|
r43347 | author = b'' | ||
if hooktype == b'changegroup' or hooktype == b'outgoing': | ||||
Boris Feld
|
r37811 | for rev in repo.changelog.revs(start=ctx.rev()): | ||
David Champion
|
r9516 | if n.node(repo[rev]): | ||
count += 1 | ||||
Nikolaus Schueler
|
r15654 | if not author: | ||
author = repo[rev].user() | ||||
David Champion
|
r9516 | else: | ||
data += ui.popbuffer() | ||||
Augie Fackler
|
r43346 | ui.note( | ||
Augie Fackler
|
r43347 | _(b'notify: suppressing notification for merge %d:%s\n') | ||
Augie Fackler
|
r43346 | % (rev, repo[rev].hex()[:12]) | ||
) | ||||
David Champion
|
r9516 | ui.pushbuffer() | ||
if count: | ||||
Augie Fackler
|
r43347 | n.diff(ctx, repo[b'tip']) | ||
Boris Feld
|
r37813 | elif ctx.rev() in repo: | ||
David Champion
|
r9516 | if not n.node(ctx): | ||
ui.popbuffer() | ||||
Augie Fackler
|
r43346 | ui.note( | ||
Augie Fackler
|
r43347 | _(b'notify: suppressing notification for merge %d:%s\n') | ||
Augie Fackler
|
r43346 | % (ctx.rev(), ctx.hex()[:12]) | ||
) | ||||
David Champion
|
r9516 | return | ||
count += 1 | ||||
Dirkjan Ochtman
|
r7726 | n.diff(ctx) | ||
Bruce Cran
|
r26503 | if not author: | ||
author = ctx.user() | ||||
Dirkjan Ochtman
|
r7726 | |||
David Champion
|
r9516 | data += ui.popbuffer() | ||
Augie Fackler
|
r43347 | fromauthor = ui.config(b'notify', b'fromauthor') | ||
Nikolaus Schueler
|
r15654 | if author and fromauthor: | ||
Augie Fackler
|
r43347 | data = b'\n'.join([b'From: %s' % author, data]) | ||
Nikolaus Schueler
|
r15654 | |||
David Champion
|
r9516 | if count: | ||
n.send(ctx, count, data) | ||||
Joerg Sonnenberger
|
r43174 | |||
Augie Fackler
|
r43346 | |||
Joerg Sonnenberger
|
r43174 | def messageid(ctx, domain, messageidseed): | ||
if domain and messageidseed: | ||||
host = domain | ||||
else: | ||||
host = encoding.strtolocal(socket.getfqdn()) | ||||
if messageidseed: | ||||
messagehash = hashlib.sha512(ctx.hex() + messageidseed) | ||||
Gregory Szorc
|
r43434 | messageid = b'<hg.%s@%s>' % ( | ||
pycompat.sysbytes(messagehash.hexdigest()[:64]), | ||||
host, | ||||
) | ||||
Joerg Sonnenberger
|
r43174 | else: | ||
Augie Fackler
|
r43347 | messageid = b'<hg.%s.%d.%d@%s>' % ( | ||
Augie Fackler
|
r43346 | ctx, | ||
int(time.time()), | ||||
hash(ctx.repo().root), | ||||
host, | ||||
) | ||||
Joerg Sonnenberger
|
r43174 | return encoding.strfromlocal(messageid) | ||