webutil.py
932 lines
| 26.7 KiB
| text/x-python
|
PythonLexer
Dirkjan Ochtman
|
r6392 | # hgweb/webutil.py - utility library for the web interface. | ||
# | ||||
# Copyright 21 May 2005 - (c) 2005 Jake Edge <jake@edge2.net> | ||||
# Copyright 2005-2007 Matt Mackall <mpm@selenic.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
|
r6392 | |||
Yuya Nishihara
|
r27046 | from __future__ import absolute_import | ||
import copy | ||||
import difflib | ||||
import os | ||||
Gregory Szorc
|
r26162 | import re | ||
Yuya Nishihara
|
r27046 | |||
from ..i18n import _ | ||||
from ..node import hex, nullid, short | ||||
Gregory Szorc
|
r43357 | from ..pycompat import setattr | ||
Yuya Nishihara
|
r27046 | |||
from .common import ( | ||||
ErrorResponse, | ||||
Denis Laxalde
|
r31665 | HTTP_BAD_REQUEST, | ||
Yuya Nishihara
|
r27046 | HTTP_NOT_FOUND, | ||
paritygen, | ||||
) | ||||
from .. import ( | ||||
context, | ||||
Yuya Nishihara
|
r38607 | diffutil, | ||
Yuya Nishihara
|
r27046 | error, | ||
match, | ||||
Denis Laxalde
|
r31808 | mdiff, | ||
r36973 | obsutil, | |||
Yuya Nishihara
|
r27046 | patch, | ||
pathutil, | ||||
Augie Fackler
|
r34808 | pycompat, | ||
Martin von Zweigbergk
|
r37351 | scmutil, | ||
Yuya Nishihara
|
r27046 | templatefilters, | ||
r35501 | templatekw, | |||
Yuya Nishihara
|
r37533 | templateutil, | ||
Yuya Nishihara
|
r27046 | ui as uimod, | ||
util, | ||||
) | ||||
Dirkjan Ochtman
|
r6392 | |||
Augie Fackler
|
r43346 | from ..utils import stringutil | ||
archivespecs = util.sortdict( | ||||
( | ||||
Augie Fackler
|
r43347 | (b'zip', (b'application/zip', b'zip', b'.zip', None)), | ||
(b'gz', (b'application/x-gzip', b'tgz', b'.tar.gz', None)), | ||||
(b'bz2', (b'application/x-bzip2', b'tbz2', b'.tar.bz2', None)), | ||||
Augie Fackler
|
r43346 | ) | ||
Yuya Nishihara
|
r37102 | ) | ||
Yuya Nishihara
|
r37529 | |||
Yuya Nishihara
|
r37532 | def archivelist(ui, nodeid, url=None): | ||
Augie Fackler
|
r43347 | allowed = ui.configlist(b'web', b'allow-archive', untrusted=True) | ||
Yuya Nishihara
|
r37531 | archives = [] | ||
Gregory Szorc
|
r43376 | for typ, spec in pycompat.iteritems(archivespecs): | ||
Augie Fackler
|
r43346 | if typ in allowed or ui.configbool( | ||
Augie Fackler
|
r43347 | b'web', b'allow' + typ, untrusted=True | ||
Augie Fackler
|
r43346 | ): | ||
archives.append( | ||||
Augie Fackler
|
r43347 | { | ||
b'type': typ, | ||||
b'extension': spec[2], | ||||
b'node': nodeid, | ||||
b'url': url, | ||||
} | ||||
Augie Fackler
|
r43346 | ) | ||
Yuya Nishihara
|
r37531 | |||
Yuya Nishihara
|
r37533 | return templateutil.mappinglist(archives) | ||
Yuya Nishihara
|
r37531 | |||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r6393 | def up(p): | ||
Augie Fackler
|
r43347 | if p[0:1] != b"/": | ||
p = b"/" + p | ||||
if p[-1:] == b"/": | ||||
Dirkjan Ochtman
|
r6393 | p = p[:-1] | ||
up = os.path.dirname(p) | ||||
Augie Fackler
|
r43347 | if up == b"/": | ||
return b"/" | ||||
return up + b"/" | ||||
Dirkjan Ochtman
|
r6393 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r18391 | def _navseq(step, firststep=None): | ||
if firststep: | ||||
yield firststep | ||||
if firststep >= 20 and firststep <= 40: | ||||
Pierre-Yves David
|
r18392 | firststep = 50 | ||
yield firststep | ||||
assert step > 0 | ||||
assert firststep > 0 | ||||
while step <= firststep: | ||||
step *= 10 | ||||
Pierre-Yves David
|
r18390 | while True: | ||
Pierre-Yves David
|
r18391 | yield 1 * step | ||
yield 3 * step | ||||
step *= 10 | ||||
Pierre-Yves David
|
r18389 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r18403 | class revnav(object): | ||
Pierre-Yves David
|
r18409 | def __init__(self, repo): | ||
Pierre-Yves David
|
r18404 | """Navigation generation object | ||
Pierre-Yves David
|
r18409 | :repo: repo object we generate nav for | ||
Pierre-Yves David
|
r18404 | """ | ||
Pierre-Yves David
|
r18409 | # used for hex generation | ||
self._revlog = repo.changelog | ||||
Pierre-Yves David
|
r18404 | |||
Pierre-Yves David
|
r18406 | def __nonzero__(self): | ||
"""return True if any revision to navigate over""" | ||||
Pierre-Yves David
|
r19094 | return self._first() is not None | ||
Gregory Szorc
|
r31476 | __bool__ = __nonzero__ | ||
Pierre-Yves David
|
r19094 | def _first(self): | ||
"""return the minimum non-filtered changeset or None""" | ||||
try: | ||||
timeless
|
r29216 | return next(iter(self._revlog)) | ||
Pierre-Yves David
|
r19094 | except StopIteration: | ||
return None | ||||
Pierre-Yves David
|
r18406 | |||
Pierre-Yves David
|
r18405 | def hex(self, rev): | ||
Pierre-Yves David
|
r18409 | return hex(self._revlog.node(rev)) | ||
Pierre-Yves David
|
r18405 | |||
Pierre-Yves David
|
r18404 | def gen(self, pos, pagelen, limit): | ||
Pierre-Yves David
|
r18403 | """computes label and revision id for navigation link | ||
Pierre-Yves David
|
r18320 | |||
Pierre-Yves David
|
r18403 | :pos: is the revision relative to which we generate navigation. | ||
:pagelen: the size of each navigation page | ||||
:limit: how far shall we link | ||||
Dirkjan Ochtman
|
r6393 | |||
Pierre-Yves David
|
r18403 | The return is: | ||
Yuya Nishihara
|
r37716 | - a single element mappinglist | ||
Pierre-Yves David
|
r18403 | - containing a dictionary with a `before` and `after` key | ||
Yuya Nishihara
|
r37715 | - values are dictionaries with `label` and `node` keys | ||
Pierre-Yves David
|
r18403 | """ | ||
Pierre-Yves David
|
r18406 | if not self: | ||
# empty repo | ||||
Augie Fackler
|
r43346 | return templateutil.mappinglist( | ||
[ | ||||
{ | ||||
Augie Fackler
|
r43347 | b'before': templateutil.mappinglist([]), | ||
b'after': templateutil.mappinglist([]), | ||||
Augie Fackler
|
r43346 | }, | ||
] | ||||
) | ||||
Dirkjan Ochtman
|
r6393 | |||
Pierre-Yves David
|
r18425 | targets = [] | ||
Pierre-Yves David
|
r18403 | for f in _navseq(1, pagelen): | ||
if f > limit: | ||||
break | ||||
Pierre-Yves David
|
r18425 | targets.append(pos + f) | ||
targets.append(pos - f) | ||||
targets.sort() | ||||
Pierre-Yves David
|
r19094 | first = self._first() | ||
Augie Fackler
|
r43347 | navbefore = [{b'label': b'(%i)' % first, b'node': self.hex(first)}] | ||
Pierre-Yves David
|
r18425 | navafter = [] | ||
for rev in targets: | ||||
Pierre-Yves David
|
r18426 | if rev not in self._revlog: | ||
continue | ||||
Pierre-Yves David
|
r18425 | if pos < rev < limit: | ||
Augie Fackler
|
r43346 | navafter.append( | ||
Augie Fackler
|
r43347 | {b'label': b'+%d' % abs(rev - pos), b'node': self.hex(rev)} | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r18425 | if 0 < rev < pos: | ||
Augie Fackler
|
r43346 | navbefore.append( | ||
Augie Fackler
|
r43347 | {b'label': b'-%d' % abs(rev - pos), b'node': self.hex(rev)} | ||
Augie Fackler
|
r43346 | ) | ||
Nicolas Dumazet
|
r10254 | |||
Augie Fackler
|
r43347 | navafter.append({b'label': b'tip', b'node': b'tip'}) | ||
Pierre-Yves David
|
r18403 | |||
Yuya Nishihara
|
r37716 | # TODO: maybe this can be a scalar object supporting tomap() | ||
Augie Fackler
|
r43346 | return templateutil.mappinglist( | ||
[ | ||||
{ | ||||
Augie Fackler
|
r43347 | b'before': templateutil.mappinglist(navbefore), | ||
b'after': templateutil.mappinglist(navafter), | ||||
Augie Fackler
|
r43346 | }, | ||
] | ||||
) | ||||
Dirkjan Ochtman
|
r6393 | |||
Pierre-Yves David
|
r18408 | class filerevnav(revnav): | ||
Pierre-Yves David
|
r18409 | def __init__(self, repo, path): | ||
"""Navigation generation object | ||||
:repo: repo object we generate nav for | ||||
:path: path of the file we generate nav for | ||||
""" | ||||
# used for iteration | ||||
self._changelog = repo.unfiltered().changelog | ||||
# used for hex generation | ||||
self._revlog = repo.file(path) | ||||
def hex(self, rev): | ||||
return hex(self._changelog.node(self._revlog.linkrev(rev))) | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37717 | # TODO: maybe this can be a wrapper class for changectx/filectx list, which | ||
# yields {'ctx': ctx} | ||||
Yuya Nishihara
|
r37718 | def _ctxsgen(context, ctxs): | ||
Yuya Nishihara
|
r37717 | for s in ctxs: | ||
d = { | ||||
Augie Fackler
|
r43347 | b'node': s.hex(), | ||
b'rev': s.rev(), | ||||
b'user': s.user(), | ||||
b'date': s.date(), | ||||
b'description': s.description(), | ||||
b'branch': s.branch(), | ||||
Yuya Nishihara
|
r37717 | } | ||
Augie Fackler
|
r43347 | if util.safehasattr(s, b'path'): | ||
d[b'file'] = s.path() | ||||
Yuya Nishihara
|
r37717 | yield d | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37718 | def _siblings(siblings=None, hiderev=None): | ||
if siblings is None: | ||||
siblings = [] | ||||
siblings = [s for s in siblings if s.node() != nullid] | ||||
if len(siblings) == 1 and siblings[0].rev() == hiderev: | ||||
siblings = [] | ||||
return templateutil.mappinggenerator(_ctxsgen, args=(siblings,)) | ||||
Dirkjan Ochtman
|
r6392 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r34391 | def difffeatureopts(req, ui, section): | ||
Augie Fackler
|
r43346 | diffopts = diffutil.difffeatureopts( | ||
ui, untrusted=True, section=section, whitespace=True | ||||
) | ||||
Gregory Szorc
|
r34391 | |||
Augie Fackler
|
r43347 | for k in ( | ||
b'ignorews', | ||||
b'ignorewsamount', | ||||
b'ignorewseol', | ||||
b'ignoreblanklines', | ||||
): | ||||
Gregory Szorc
|
r36902 | v = req.qsparams.get(k) | ||
Gregory Szorc
|
r34391 | if v is not None: | ||
Yuya Nishihara
|
r37102 | v = stringutil.parsebool(v) | ||
Gregory Szorc
|
r34404 | setattr(diffopts, k, v if v is not None else True) | ||
Gregory Szorc
|
r34391 | |||
return diffopts | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r34391 | def annotate(req, fctx, ui): | ||
Augie Fackler
|
r43347 | diffopts = difffeatureopts(req, ui, b'annotate') | ||
Yuya Nishihara
|
r37083 | return fctx.annotate(follow=True, diffopts=diffopts) | ||
Jun Wu
|
r30081 | |||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r7671 | def parents(ctx, hide=None): | ||
Anton Shestakov
|
r24136 | if isinstance(ctx, context.basefilectx): | ||
introrev = ctx.introrev() | ||||
if ctx.changectx().rev() != introrev: | ||||
Matt Harbison
|
r24340 | return _siblings([ctx.repo()[introrev]], hide) | ||
Dirkjan Ochtman
|
r7671 | return _siblings(ctx.parents(), hide) | ||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r7671 | def children(ctx, hide=None): | ||
return _siblings(ctx.children(), hide) | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r6434 | def renamelink(fctx): | ||
Matt Mackall
|
r6437 | r = fctx.renamed() | ||
Dirkjan Ochtman
|
r6392 | if r: | ||
Augie Fackler
|
r43347 | return templateutil.mappinglist([{b'file': r[0], b'node': hex(r[1])}]) | ||
Yuya Nishihara
|
r37921 | return templateutil.mappinglist([]) | ||
Dirkjan Ochtman
|
r6392 | |||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r6392 | def nodetagsdict(repo, node): | ||
Augie Fackler
|
r43347 | return templateutil.hybridlist(repo.nodetags(node), name=b'name') | ||
Dirkjan Ochtman
|
r6392 | |||
Augie Fackler
|
r43346 | |||
Alexander Solovyov
|
r13596 | def nodebookmarksdict(repo, node): | ||
Augie Fackler
|
r43347 | return templateutil.hybridlist(repo.nodebookmarks(node), name=b'name') | ||
Alexander Solovyov
|
r13596 | |||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r6392 | def nodebranchdict(repo, ctx): | ||
branches = [] | ||||
branch = ctx.branch() | ||||
# If this is an empty repo, ctx.node() == nullid, | ||||
Brodie Rao
|
r16719 | # ctx.branch() == 'default'. | ||
try: | ||||
branchnode = repo.branchtip(branch) | ||||
except error.RepoLookupError: | ||||
branchnode = None | ||||
if branchnode == ctx.node(): | ||||
Yuya Nishihara
|
r37924 | branches.append(branch) | ||
Augie Fackler
|
r43347 | return templateutil.hybridlist(branches, name=b'name') | ||
Dirkjan Ochtman
|
r6392 | |||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r6392 | def nodeinbranch(repo, ctx): | ||
branches = [] | ||||
branch = ctx.branch() | ||||
Brodie Rao
|
r16719 | try: | ||
branchnode = repo.branchtip(branch) | ||||
except error.RepoLookupError: | ||||
branchnode = None | ||||
Augie Fackler
|
r43347 | if branch != b'default' and branchnode != ctx.node(): | ||
Yuya Nishihara
|
r37925 | branches.append(branch) | ||
Augie Fackler
|
r43347 | return templateutil.hybridlist(branches, name=b'name') | ||
Dirkjan Ochtman
|
r6392 | |||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r6392 | def nodebranchnodefault(ctx): | ||
branches = [] | ||||
branch = ctx.branch() | ||||
Augie Fackler
|
r43347 | if branch != b'default': | ||
Yuya Nishihara
|
r37926 | branches.append(branch) | ||
Augie Fackler
|
r43347 | return templateutil.hybridlist(branches, name=b'name') | ||
Dirkjan Ochtman
|
r6392 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37930 | def _nodenamesgen(context, f, node, name): | ||
for t in f(node): | ||||
yield {name: t} | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37931 | def showtag(repo, t1, node=nullid): | ||
Augie Fackler
|
r43347 | args = (repo.nodetags, node, b'tag') | ||
Yuya Nishihara
|
r37930 | return templateutil.mappinggenerator(_nodenamesgen, args=args, name=t1) | ||
Dirkjan Ochtman
|
r6392 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37931 | def showbookmark(repo, t1, node=nullid): | ||
Augie Fackler
|
r43347 | args = (repo.nodebookmarks, node, b'bookmark') | ||
Yuya Nishihara
|
r37930 | return templateutil.mappinggenerator(_nodenamesgen, args=args, name=t1) | ||
Alexander Solovyov
|
r13596 | |||
Augie Fackler
|
r43346 | |||
r26129 | def branchentries(repo, stripecount, limit=0): | |||
tips = [] | ||||
heads = repo.heads() | ||||
parity = paritygen(stripecount) | ||||
sortkey = lambda item: (not item[1], item[0].rev()) | ||||
Yuya Nishihara
|
r37932 | def entries(context): | ||
r26129 | count = 0 | |||
if not tips: | ||||
for tag, hs, tip, closed in repo.branchmap().iterbranches(): | ||||
tips.append((repo[tip], closed)) | ||||
for ctx, closed in sorted(tips, key=sortkey, reverse=True): | ||||
if limit > 0 and count >= limit: | ||||
return | ||||
count += 1 | ||||
if closed: | ||||
Augie Fackler
|
r43347 | status = b'closed' | ||
r26129 | elif ctx.node() not in heads: | |||
Augie Fackler
|
r43347 | status = b'inactive' | ||
r26129 | else: | |||
Augie Fackler
|
r43347 | status = b'open' | ||
r26129 | yield { | |||
Augie Fackler
|
r43347 | b'parity': next(parity), | ||
b'branch': ctx.branch(), | ||||
b'status': status, | ||||
b'node': ctx.hex(), | ||||
b'date': ctx.date(), | ||||
r26129 | } | |||
Yuya Nishihara
|
r37932 | return templateutil.mappinggenerator(entries) | ||
r26129 | ||||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r6392 | def cleanpath(repo, path): | ||
Augie Fackler
|
r43347 | path = path.lstrip(b'/') | ||
Yuya Nishihara
|
r39507 | auditor = pathutil.pathauditor(repo.root, realfs=False) | ||
Augie Fackler
|
r43347 | return pathutil.canonpath(repo.root, b'', path, auditor=auditor) | ||
Dirkjan Ochtman
|
r6392 | |||
Augie Fackler
|
r43346 | |||
r25999 | def changectx(repo, req): | |||
Augie Fackler
|
r43347 | changeid = b"tip" | ||
if b'node' in req.qsparams: | ||||
changeid = req.qsparams[b'node'] | ||||
ipos = changeid.find(b':') | ||||
Weiwen
|
r17991 | if ipos != -1: | ||
Augie Fackler
|
r43346 | changeid = changeid[(ipos + 1) :] | ||
Weiwen
|
r17991 | |||
Martin von Zweigbergk
|
r37351 | return scmutil.revsymbol(repo, changeid) | ||
Weiwen
|
r17991 | |||
Augie Fackler
|
r43346 | |||
Weiwen
|
r17991 | def basechangectx(repo, req): | ||
Augie Fackler
|
r43347 | if b'node' in req.qsparams: | ||
changeid = req.qsparams[b'node'] | ||||
ipos = changeid.find(b':') | ||||
Weiwen
|
r17991 | if ipos != -1: | ||
changeid = changeid[:ipos] | ||||
Martin von Zweigbergk
|
r37351 | return scmutil.revsymbol(repo, changeid) | ||
Weiwen
|
r17991 | |||
return None | ||||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r6392 | def filectx(repo, req): | ||
Augie Fackler
|
r43347 | if b'file' not in req.qsparams: | ||
raise ErrorResponse(HTTP_NOT_FOUND, b'file not given') | ||||
path = cleanpath(repo, req.qsparams[b'file']) | ||||
if b'node' in req.qsparams: | ||||
changeid = req.qsparams[b'node'] | ||||
elif b'filenode' in req.qsparams: | ||||
changeid = req.qsparams[b'filenode'] | ||||
Dirkjan Ochtman
|
r6392 | else: | ||
Augie Fackler
|
r43347 | raise ErrorResponse(HTTP_NOT_FOUND, b'node or filenode not given') | ||
Dirkjan Ochtman
|
r6392 | try: | ||
Martin von Zweigbergk
|
r37351 | fctx = scmutil.revsymbol(repo, changeid)[path] | ||
Matt Mackall
|
r7637 | except error.RepoError: | ||
Dirkjan Ochtman
|
r6392 | fctx = repo.filectx(path, fileid=changeid) | ||
return fctx | ||||
Dirkjan Ochtman
|
r7310 | |||
Augie Fackler
|
r43346 | |||
Denis Laxalde
|
r31665 | def linerange(req): | ||
Augie Fackler
|
r43347 | linerange = req.qsparams.getall(b'linerange') | ||
Gregory Szorc
|
r36881 | if not linerange: | ||
Denis Laxalde
|
r31665 | return None | ||
if len(linerange) > 1: | ||||
Augie Fackler
|
r43347 | raise ErrorResponse(HTTP_BAD_REQUEST, b'redundant linerange parameter') | ||
Denis Laxalde
|
r31665 | try: | ||
Augie Fackler
|
r43347 | fromline, toline = map(int, linerange[0].split(b':', 1)) | ||
Denis Laxalde
|
r31665 | except ValueError: | ||
Augie Fackler
|
r43347 | raise ErrorResponse(HTTP_BAD_REQUEST, b'invalid linerange parameter') | ||
Denis Laxalde
|
r31665 | try: | ||
return util.processlinerange(fromline, toline) | ||||
except error.ParseError as exc: | ||||
Augie Fackler
|
r36272 | raise ErrorResponse(HTTP_BAD_REQUEST, pycompat.bytestr(exc)) | ||
Denis Laxalde
|
r31665 | |||
Augie Fackler
|
r43346 | |||
Denis Laxalde
|
r31665 | def formatlinerange(fromline, toline): | ||
Augie Fackler
|
r43347 | return b'%d:%d' % (fromline + 1, toline) | ||
Denis Laxalde
|
r31665 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37933 | def _succsandmarkersgen(context, mapping): | ||
Augie Fackler
|
r43347 | repo = context.resource(mapping, b'repo') | ||
Yuya Nishihara
|
r37521 | itemmappings = templatekw.showsuccsandmarkers(context, mapping) | ||
for item in itemmappings.tovalue(context, mapping): | ||||
Augie Fackler
|
r43347 | item[b'successors'] = _siblings( | ||
repo[successor] for successor in item[b'successors'] | ||||
Augie Fackler
|
r43346 | ) | ||
r35502 | yield item | |||
r35501 | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37933 | def succsandmarkers(context, mapping): | ||
return templateutil.mappinggenerator(_succsandmarkersgen, args=(mapping,)) | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r36612 | # teach templater succsandmarkers is switched to (context, mapping) API | ||
Augie Fackler
|
r43347 | succsandmarkers._requires = {b'repo', b'ctx'} | ||
Yuya Nishihara
|
r36612 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37934 | def _whyunstablegen(context, mapping): | ||
Augie Fackler
|
r43347 | repo = context.resource(mapping, b'repo') | ||
ctx = context.resource(mapping, b'ctx') | ||||
r36973 | ||||
entries = obsutil.whyunstable(repo, ctx) | ||||
for entry in entries: | ||||
Augie Fackler
|
r43347 | if entry.get(b'divergentnodes'): | ||
entry[b'divergentnodes'] = _siblings(entry[b'divergentnodes']) | ||||
r36973 | yield entry | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37934 | def whyunstable(context, mapping): | ||
return templateutil.mappinggenerator(_whyunstablegen, args=(mapping,)) | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | whyunstable._requires = {b'repo', b'ctx'} | ||
r36973 | ||||
Augie Fackler
|
r43346 | |||
r27294 | def commonentry(repo, ctx): | |||
Yuya Nishihara
|
r39830 | node = scmutil.binnode(ctx) | ||
r27294 | return { | |||
Yuya Nishihara
|
r36535 | # TODO: perhaps ctx.changectx() should be assigned if ctx is a | ||
# filectx, but I'm not pretty sure if that would always work because | ||||
# fctx.parents() != fctx.changectx.parents() for example. | ||||
Augie Fackler
|
r43347 | b'ctx': ctx, | ||
b'rev': ctx.rev(), | ||||
b'node': hex(node), | ||||
b'author': ctx.user(), | ||||
b'desc': ctx.description(), | ||||
b'date': ctx.date(), | ||||
b'extra': ctx.extra(), | ||||
b'phase': ctx.phasestr(), | ||||
b'obsolete': ctx.obsolete(), | ||||
b'succsandmarkers': succsandmarkers, | ||||
b'instabilities': templateutil.hybridlist( | ||||
ctx.instabilities(), name=b'instability' | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | b'whyunstable': whyunstable, | ||
b'branch': nodebranchnodefault(ctx), | ||||
b'inbranch': nodeinbranch(repo, ctx), | ||||
b'branches': nodebranchdict(repo, ctx), | ||||
b'tags': nodetagsdict(repo, node), | ||||
b'bookmarks': nodebookmarksdict(repo, node), | ||||
b'parent': lambda context, mapping: parents(ctx), | ||||
b'child': lambda context, mapping: children(ctx), | ||||
r27294 | } | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r36901 | def changelistentry(web, ctx): | ||
Gregory Szorc
|
r23745 | '''Obtain a dictionary to be used for entries in a changelist. | ||
This function is called when producing items for the "entries" list passed | ||||
to the "shortlog" and "changelog" templates. | ||||
''' | ||||
repo = web.repo | ||||
rev = ctx.rev() | ||||
Yuya Nishihara
|
r39830 | n = scmutil.binnode(ctx) | ||
Augie Fackler
|
r43347 | showtags = showtag(repo, b'changelogtag', n) | ||
Yuya Nishihara
|
r37973 | files = listfilediffs(ctx.files(), n, web.maxfiles) | ||
Gregory Szorc
|
r23745 | |||
r27294 | entry = commonentry(repo, ctx) | |||
Augie Fackler
|
r43346 | entry.update( | ||
{ | ||||
Augie Fackler
|
r43347 | b'allparents': lambda context, mapping: parents(ctx), | ||
b'parent': lambda context, mapping: parents(ctx, rev - 1), | ||||
b'child': lambda context, mapping: children(ctx, rev + 1), | ||||
b'changelogtag': showtags, | ||||
b'files': files, | ||||
Augie Fackler
|
r43346 | } | ||
) | ||||
r27294 | return entry | |||
Gregory Szorc
|
r23745 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r38054 | def changelistentries(web, revs, maxcount, parityfn): | ||
"""Emit up to N records for an iterable of revisions.""" | ||||
repo = web.repo | ||||
count = 0 | ||||
for rev in revs: | ||||
if count >= maxcount: | ||||
break | ||||
count += 1 | ||||
entry = changelistentry(web, repo[rev]) | ||||
Augie Fackler
|
r43347 | entry[b'parity'] = next(parityfn) | ||
Gregory Szorc
|
r38054 | |||
yield entry | ||||
Augie Fackler
|
r43346 | |||
r25602 | def symrevorshortnode(req, ctx): | |||
Augie Fackler
|
r43347 | if b'node' in req.qsparams: | ||
return templatefilters.revescape(req.qsparams[b'node']) | ||||
r25602 | else: | |||
Yuya Nishihara
|
r39830 | return short(scmutil.binnode(ctx)) | ||
r25602 | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37970 | def _listfilesgen(context, ctx, stripecount): | ||
Yuya Nishihara
|
r37968 | parity = paritygen(stripecount) | ||
for blockno, f in enumerate(ctx.files()): | ||||
Augie Fackler
|
r43347 | template = b'filenodelink' if f in ctx else b'filenolink' | ||
Augie Fackler
|
r43346 | yield context.process( | ||
template, | ||||
{ | ||||
Augie Fackler
|
r43347 | b'node': ctx.hex(), | ||
b'file': f, | ||||
b'blockno': blockno + 1, | ||||
b'parity': next(parity), | ||||
Augie Fackler
|
r43346 | }, | ||
) | ||||
Yuya Nishihara
|
r37968 | |||
Gregory Szorc
|
r36902 | def changesetentry(web, ctx): | ||
Gregory Szorc
|
r24177 | '''Obtain a dictionary to be used to render the "changeset" template.''' | ||
Augie Fackler
|
r43347 | showtags = showtag(web.repo, b'changesettag', scmutil.binnode(ctx)) | ||
Augie Fackler
|
r43346 | showbookmarks = showbookmark( | ||
Augie Fackler
|
r43347 | web.repo, b'changesetbookmark', scmutil.binnode(ctx) | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r24177 | showbranch = nodebranchnodefault(ctx) | ||
Gregory Szorc
|
r36902 | basectx = basechangectx(web.repo, web.req) | ||
Gregory Szorc
|
r24177 | if basectx is None: | ||
basectx = ctx.p1() | ||||
Augie Fackler
|
r43347 | style = web.config(b'web', b'style') | ||
if b'style' in web.req.qsparams: | ||||
style = web.req.qsparams[b'style'] | ||||
Gregory Szorc
|
r24177 | |||
Gregory Szorc
|
r36901 | diff = diffs(web, ctx, basectx, None, style) | ||
Gregory Szorc
|
r24177 | |||
parity = paritygen(web.stripecount) | ||||
Yuya Nishihara
|
r38604 | diffstatsgen = diffstatgen(web.repo.ui, ctx, basectx) | ||
Yuya Nishihara
|
r38070 | diffstats = diffstat(ctx, diffstatsgen, parity) | ||
Gregory Szorc
|
r24177 | |||
return dict( | ||||
diff=diff, | ||||
Gregory Szorc
|
r36902 | symrev=symrevorshortnode(web.req, ctx), | ||
Gregory Szorc
|
r24177 | basenode=basectx.hex(), | ||
changesettag=showtags, | ||||
changesetbookmark=showbookmarks, | ||||
changesetbranch=showbranch, | ||||
Augie Fackler
|
r43346 | files=templateutil.mappedgenerator( | ||
_listfilesgen, args=(ctx, web.stripecount) | ||||
), | ||||
Matt Harbison
|
r42524 | diffsummary=lambda context, mapping: diffsummary(diffstatsgen), | ||
Gregory Szorc
|
r24177 | diffstat=diffstats, | ||
archives=web.archivelist(ctx.hex()), | ||||
Augie Fackler
|
r43346 | **pycompat.strkwargs(commonentry(web.repo, ctx)) | ||
) | ||||
Gregory Szorc
|
r24177 | |||
Yuya Nishihara
|
r37972 | def _listfilediffsgen(context, files, node, max): | ||
Dirkjan Ochtman
|
r7311 | for f in files[:max]: | ||
Augie Fackler
|
r43347 | yield context.process(b'filedifflink', {b'node': hex(node), b'file': f}) | ||
Dirkjan Ochtman
|
r7311 | if len(files) > max: | ||
Augie Fackler
|
r43347 | yield context.process(b'fileellipses', {}) | ||
Dirkjan Ochtman
|
r7311 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37973 | def listfilediffs(files, node, max): | ||
Augie Fackler
|
r43346 | return templateutil.mappedgenerator( | ||
_listfilediffsgen, args=(files, node, max) | ||||
) | ||||
Yuya Nishihara
|
r37971 | |||
Yuya Nishihara
|
r38006 | def _prettyprintdifflines(context, lines, blockno, lineidprefix): | ||
Yuya Nishihara
|
r38004 | for lineno, l in enumerate(lines, 1): | ||
Augie Fackler
|
r43347 | difflineno = b"%d.%d" % (blockno, lineno) | ||
if l.startswith(b'+'): | ||||
ltype = b"difflineplus" | ||||
elif l.startswith(b'-'): | ||||
ltype = b"difflineminus" | ||||
elif l.startswith(b'@'): | ||||
ltype = b"difflineat" | ||||
Yuya Nishihara
|
r38004 | else: | ||
Augie Fackler
|
r43347 | ltype = b"diffline" | ||
Augie Fackler
|
r43346 | yield context.process( | ||
ltype, | ||||
{ | ||||
Augie Fackler
|
r43347 | b'line': l, | ||
b'lineno': lineno, | ||||
b'lineid': lineidprefix + b"l%s" % difflineno, | ||||
b'linenumber': b"% 8s" % difflineno, | ||||
Augie Fackler
|
r43346 | }, | ||
) | ||||
Yuya Nishihara
|
r38004 | |||
Augie Fackler
|
r43346 | def _diffsgen( | ||
context, | ||||
repo, | ||||
ctx, | ||||
basectx, | ||||
files, | ||||
style, | ||||
stripecount, | ||||
linerange, | ||||
lineidprefix, | ||||
): | ||||
Dirkjan Ochtman
|
r7310 | if files: | ||
Martin von Zweigbergk
|
r41825 | m = match.exact(files) | ||
Dirkjan Ochtman
|
r7310 | else: | ||
Martin von Zweigbergk
|
r41825 | m = match.always() | ||
Dirkjan Ochtman
|
r7310 | |||
diffopts = patch.diffopts(repo.ui, untrusted=True) | ||||
Yuya Nishihara
|
r38007 | parity = paritygen(stripecount) | ||
Dirkjan Ochtman
|
r7310 | |||
Martin von Zweigbergk
|
r41767 | diffhunks = patch.diffhunks(repo, basectx, ctx, m, opts=diffopts) | ||
Denis Laxalde
|
r34856 | for blockno, (fctx1, fctx2, header, hunks) in enumerate(diffhunks, 1): | ||
Augie Fackler
|
r43347 | if style != b'raw': | ||
Denis Laxalde
|
r31276 | header = header[1:] | ||
Augie Fackler
|
r43347 | lines = [h + b'\n' for h in header] | ||
Denis Laxalde
|
r31276 | for hunkrange, hunklines in hunks: | ||
Denis Laxalde
|
r31666 | if linerange is not None and hunkrange is not None: | ||
s1, l1, s2, l2 = hunkrange | ||||
Denis Laxalde
|
r31808 | if not mdiff.hunkinrange((s2, l2), linerange): | ||
Denis Laxalde
|
r31666 | continue | ||
Denis Laxalde
|
r31276 | lines.extend(hunklines) | ||
if lines: | ||||
Augie Fackler
|
r43346 | l = templateutil.mappedgenerator( | ||
_prettyprintdifflines, args=(lines, blockno, lineidprefix) | ||||
) | ||||
Yuya Nishihara
|
r38007 | yield { | ||
Augie Fackler
|
r43347 | b'parity': next(parity), | ||
b'blockno': blockno, | ||||
b'lines': l, | ||||
Yuya Nishihara
|
r38007 | } | ||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | def diffs(web, ctx, basectx, files, style, linerange=None, lineidprefix=b''): | ||
Augie Fackler
|
r43346 | args = ( | ||
web.repo, | ||||
ctx, | ||||
basectx, | ||||
files, | ||||
style, | ||||
web.stripecount, | ||||
linerange, | ||||
lineidprefix, | ||||
) | ||||
Augie Fackler
|
r43347 | return templateutil.mappinggenerator( | ||
_diffsgen, args=args, name=b'diffblock' | ||||
) | ||||
Dirkjan Ochtman
|
r7345 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r38010 | def _compline(type, leftlineno, leftline, rightlineno, rightline): | ||
Augie Fackler
|
r43347 | lineid = leftlineno and (b"l%d" % leftlineno) or b'' | ||
lineid += rightlineno and (b"r%d" % rightlineno) or b'' | ||||
llno = b'%d' % leftlineno if leftlineno else b'' | ||||
rlno = b'%d' % rightlineno if rightlineno else b'' | ||||
Yuya Nishihara
|
r38010 | return { | ||
Augie Fackler
|
r43347 | b'type': type, | ||
b'lineid': lineid, | ||||
b'leftlineno': leftlineno, | ||||
b'leftlinenumber': b"% 6s" % llno, | ||||
b'leftline': leftline or b'', | ||||
b'rightlineno': rightlineno, | ||||
b'rightlinenumber': b"% 6s" % rlno, | ||||
b'rightline': rightline or b'', | ||||
Yuya Nishihara
|
r38010 | } | ||
Yuya Nishihara
|
r38008 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r38010 | def _getcompblockgen(context, leftlines, rightlines, opcodes): | ||
Yuya Nishihara
|
r38009 | for type, llo, lhi, rlo, rhi in opcodes: | ||
Gregory Szorc
|
r40192 | type = pycompat.sysbytes(type) | ||
Yuya Nishihara
|
r38009 | len1 = lhi - llo | ||
len2 = rhi - rlo | ||||
count = min(len1, len2) | ||||
Gregory Szorc
|
r38806 | for i in pycompat.xrange(count): | ||
Augie Fackler
|
r43346 | yield _compline( | ||
type=type, | ||||
leftlineno=llo + i + 1, | ||||
leftline=leftlines[llo + i], | ||||
rightlineno=rlo + i + 1, | ||||
rightline=rightlines[rlo + i], | ||||
) | ||||
Yuya Nishihara
|
r38009 | if len1 > len2: | ||
Gregory Szorc
|
r38806 | for i in pycompat.xrange(llo + count, lhi): | ||
Augie Fackler
|
r43346 | yield _compline( | ||
type=type, | ||||
leftlineno=i + 1, | ||||
leftline=leftlines[i], | ||||
rightlineno=None, | ||||
rightline=None, | ||||
) | ||||
Yuya Nishihara
|
r38009 | elif len2 > len1: | ||
Gregory Szorc
|
r38806 | for i in pycompat.xrange(rlo + count, rhi): | ||
Augie Fackler
|
r43346 | yield _compline( | ||
type=type, | ||||
leftlineno=None, | ||||
leftline=None, | ||||
rightlineno=i + 1, | ||||
rightline=rightlines[i], | ||||
) | ||||
wujek srujek
|
r17202 | |||
Yuya Nishihara
|
r38010 | def _getcompblock(leftlines, rightlines, opcodes): | ||
args = (leftlines, rightlines, opcodes) | ||||
Augie Fackler
|
r43346 | return templateutil.mappinggenerator( | ||
Augie Fackler
|
r43347 | _getcompblockgen, args=args, name=b'comparisonline' | ||
Augie Fackler
|
r43346 | ) | ||
Yuya Nishihara
|
r38010 | |||
Yuya Nishihara
|
r38012 | def _comparegen(context, contextnum, leftlines, rightlines): | ||
Yuya Nishihara
|
r38009 | '''Generator function that provides side-by-side comparison data.''' | ||
wujek srujek
|
r17202 | s = difflib.SequenceMatcher(None, leftlines, rightlines) | ||
Yuya Nishihara
|
r38011 | if contextnum < 0: | ||
Yuya Nishihara
|
r38010 | l = _getcompblock(leftlines, rightlines, s.get_opcodes()) | ||
Augie Fackler
|
r43347 | yield {b'lines': l} | ||
wujek srujek
|
r17202 | else: | ||
Yuya Nishihara
|
r38011 | for oc in s.get_grouped_opcodes(n=contextnum): | ||
Yuya Nishihara
|
r38010 | l = _getcompblock(leftlines, rightlines, oc) | ||
Augie Fackler
|
r43347 | yield {b'lines': l} | ||
Yuya Nishihara
|
r38012 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r38013 | def compare(contextnum, leftlines, rightlines): | ||
Yuya Nishihara
|
r38012 | args = (contextnum, leftlines, rightlines) | ||
Augie Fackler
|
r43346 | return templateutil.mappinggenerator( | ||
Augie Fackler
|
r43347 | _comparegen, args=args, name=b'comparisonblock' | ||
Augie Fackler
|
r43346 | ) | ||
wujek srujek
|
r17202 | |||
Yuya Nishihara
|
r38604 | def diffstatgen(ui, ctx, basectx): | ||
Steven Brown
|
r14570 | '''Generator function that provides the diffstat data.''' | ||
Steven Brown
|
r14490 | |||
Augie Fackler
|
r43347 | diffopts = patch.diffopts(ui, {b'noprefix': False}) | ||
Augie Fackler
|
r43346 | stats = patch.diffstatdata(util.iterlines(ctx.diff(basectx, opts=diffopts))) | ||
Steven Brown
|
r14490 | maxname, maxtotal, addtotal, removetotal, binary = patch.diffstatsum(stats) | ||
Steven Brown
|
r14570 | while True: | ||
yield stats, maxname, maxtotal, addtotal, removetotal, binary | ||||
Augie Fackler
|
r43346 | |||
Steven Brown
|
r14570 | def diffsummary(statgen): | ||
'''Return a short summary of the diff.''' | ||||
timeless
|
r29216 | stats, maxname, maxtotal, addtotal, removetotal, binary = next(statgen) | ||
Augie Fackler
|
r43347 | return _(b' %d files changed, %d insertions(+), %d deletions(-)\n') % ( | ||
Augie Fackler
|
r43346 | len(stats), | ||
addtotal, | ||||
removetotal, | ||||
) | ||||
Steven Brown
|
r14570 | |||
Yuya Nishihara
|
r38069 | def _diffstattmplgen(context, ctx, statgen, parity): | ||
timeless
|
r29216 | stats, maxname, maxtotal, addtotal, removetotal, binary = next(statgen) | ||
Steven Brown
|
r14561 | files = ctx.files() | ||
Steven Brown
|
r14490 | |||
Steven Brown
|
r14561 | def pct(i): | ||
if maxtotal == 0: | ||||
return 0 | ||||
return (float(i) / maxtotal) * 100 | ||||
Steven Brown
|
r14490 | |||
Steven Brown
|
r14562 | fileno = 0 | ||
Steven Brown
|
r14561 | for filename, adds, removes, isbinary in stats: | ||
Augie Fackler
|
r43347 | template = b'diffstatlink' if filename in files else b'diffstatnolink' | ||
Steven Brown
|
r14561 | total = adds + removes | ||
Steven Brown
|
r14562 | fileno += 1 | ||
Augie Fackler
|
r43346 | yield context.process( | ||
template, | ||||
{ | ||||
Augie Fackler
|
r43347 | b'node': ctx.hex(), | ||
b'file': filename, | ||||
b'fileno': fileno, | ||||
b'total': total, | ||||
b'addpct': pct(adds), | ||||
b'removepct': pct(removes), | ||||
b'parity': next(parity), | ||||
Augie Fackler
|
r43346 | }, | ||
) | ||||
Steven Brown
|
r14490 | |||
Yuya Nishihara
|
r38070 | def diffstat(ctx, statgen, parity): | ||
Yuya Nishihara
|
r38068 | '''Return a diffstat template for each file in the diff.''' | ||
Yuya Nishihara
|
r38069 | args = (ctx, statgen, parity) | ||
Yuya Nishihara
|
r38068 | return templateutil.mappedgenerator(_diffstattmplgen, args=args) | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37714 | class sessionvars(templateutil.wrapped): | ||
Augie Fackler
|
r43347 | def __init__(self, vars, start=b'?'): | ||
Yuya Nishihara
|
r37712 | self._start = start | ||
self._vars = vars | ||||
Yuya Nishihara
|
r37713 | |||
Dirkjan Ochtman
|
r7345 | def __getitem__(self, key): | ||
Yuya Nishihara
|
r37712 | return self._vars[key] | ||
Yuya Nishihara
|
r37713 | |||
Dirkjan Ochtman
|
r7345 | def __setitem__(self, key, value): | ||
Yuya Nishihara
|
r37712 | self._vars[key] = value | ||
Yuya Nishihara
|
r37713 | |||
Dirkjan Ochtman
|
r7345 | def __copy__(self): | ||
Yuya Nishihara
|
r37712 | return sessionvars(copy.copy(self._vars), self._start) | ||
Yuya Nishihara
|
r37713 | |||
Yuya Nishihara
|
r38286 | def contains(self, context, mapping, item): | ||
item = templateutil.unwrapvalue(context, mapping, item) | ||||
return item in self._vars | ||||
Yuya Nishihara
|
r38261 | def getmember(self, context, mapping, key): | ||
Yuya Nishihara
|
r38262 | key = templateutil.unwrapvalue(context, mapping, key) | ||
Yuya Nishihara
|
r38261 | return self._vars.get(key) | ||
Yuya Nishihara
|
r38284 | def getmin(self, context, mapping): | ||
Augie Fackler
|
r43347 | raise error.ParseError(_(b'not comparable')) | ||
Yuya Nishihara
|
r38284 | |||
def getmax(self, context, mapping): | ||||
Augie Fackler
|
r43347 | raise error.ParseError(_(b'not comparable')) | ||
Yuya Nishihara
|
r38284 | |||
Yuya Nishihara
|
r38467 | def filter(self, context, mapping, select): | ||
# implement if necessary | ||||
Augie Fackler
|
r43347 | raise error.ParseError(_(b'not filterable')) | ||
Yuya Nishihara
|
r38467 | |||
Yuya Nishihara
|
r37714 | def itermaps(self, context): | ||
Yuya Nishihara
|
r37712 | separator = self._start | ||
Gregory Szorc
|
r43376 | for key, value in sorted(pycompat.iteritems(self._vars)): | ||
Augie Fackler
|
r43346 | yield { | ||
Augie Fackler
|
r43347 | b'name': key, | ||
b'value': pycompat.bytestr(value), | ||||
b'separator': separator, | ||||
Augie Fackler
|
r34808 | } | ||
Augie Fackler
|
r43347 | separator = b'&' | ||
Augie Fackler
|
r12691 | |||
Yuya Nishihara
|
r37714 | def join(self, context, mapping, sep): | ||
# could be '{separator}{name}={value|urlescape}' | ||||
Augie Fackler
|
r43347 | raise error.ParseError(_(b'not displayable without template')) | ||
Yuya Nishihara
|
r37714 | |||
def show(self, context, mapping): | ||||
Augie Fackler
|
r43347 | return self.join(context, b'') | ||
Yuya Nishihara
|
r37714 | |||
Yuya Nishihara
|
r38308 | def tobool(self, context, mapping): | ||
return bool(self._vars) | ||||
Yuya Nishihara
|
r37714 | def tovalue(self, context, mapping): | ||
return self._vars | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r27007 | class wsgiui(uimod.ui): | ||
Augie Fackler
|
r12691 | # default termwidth breaks under mod_wsgi | ||
def termwidth(self): | ||||
return 80 | ||||
Gregory Szorc
|
r26162 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r26162 | def getwebsubs(repo): | ||
websubtable = [] | ||||
Augie Fackler
|
r43347 | websubdefs = repo.ui.configitems(b'websub') | ||
Gregory Szorc
|
r26162 | # we must maintain interhg backwards compatibility | ||
Augie Fackler
|
r43347 | websubdefs += repo.ui.configitems(b'interhg') | ||
Gregory Szorc
|
r26162 | for key, pattern in websubdefs: | ||
# grab the delimiter from the character after the "s" | ||||
Pulkit Goyal
|
r36198 | unesc = pattern[1:2] | ||
Augie Fackler
|
r38494 | delim = stringutil.reescape(unesc) | ||
Gregory Szorc
|
r26162 | |||
# identify portions of the pattern, taking care to avoid escaped | ||||
# delimiters. the replace format and flags are optional, but | ||||
# delimiters are required. | ||||
match = re.match( | ||||
Pulkit Goyal
|
r36201 | br'^s%s(.+)(?:(?<=\\\\)|(?<!\\))%s(.*)%s([ilmsux])*$' | ||
Augie Fackler
|
r43346 | % (delim, delim, delim), | ||
pattern, | ||||
) | ||||
Gregory Szorc
|
r26162 | if not match: | ||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
Augie Fackler
|
r43347 | _(b"websub: invalid pattern for %s: %s\n") % (key, pattern) | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r26162 | continue | ||
# we need to unescape the delimiter for regexp and format | ||||
Pulkit Goyal
|
r36201 | delim_re = re.compile(br'(?<!\\)\\%s' % delim) | ||
Gregory Szorc
|
r26162 | regexp = delim_re.sub(unesc, match.group(1)) | ||
format = delim_re.sub(unesc, match.group(2)) | ||||
# the pattern allows for 6 regexp flags, so set them if necessary | ||||
flagin = match.group(3) | ||||
flags = 0 | ||||
if flagin: | ||||
Connor Sheehan
|
r43189 | for flag in pycompat.sysstr(flagin.upper()): | ||
Gregory Szorc
|
r26162 | flags |= re.__dict__[flag] | ||
try: | ||||
regexp = re.compile(regexp, flags) | ||||
websubtable.append((regexp, format)) | ||||
except re.error: | ||||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
Augie Fackler
|
r43347 | _(b"websub: invalid regexp for %s: %s\n") % (key, regexp) | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r26162 | return websubtable | ||
r37928 | ||||
Augie Fackler
|
r43346 | |||
r37928 | def getgraphnode(repo, ctx): | |||
Augie Fackler
|
r43346 | return templatekw.getgraphnodecurrent( | ||
repo, ctx | ||||
) + templatekw.getgraphnodesymbol(ctx) | ||||