webutil.py
812 lines
| 26.2 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 | ||||
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 | |||
Yuya Nishihara
|
r37102 | from ..utils import ( | ||
stringutil, | ||||
) | ||||
Yuya Nishihara
|
r37529 | archivespecs = util.sortdict(( | ||
('zip', ('application/zip', 'zip', '.zip', None)), | ||||
('gz', ('application/x-gzip', 'tgz', '.tar.gz', None)), | ||||
('bz2', ('application/x-bzip2', 'tbz2', '.tar.bz2', None)), | ||||
)) | ||||
Yuya Nishihara
|
r37532 | def archivelist(ui, nodeid, url=None): | ||
David Demelier
|
r38215 | allowed = ui.configlist('web', 'allow-archive', untrusted=True) | ||
Yuya Nishihara
|
r37531 | archives = [] | ||
for typ, spec in archivespecs.iteritems(): | ||||
if typ in allowed or ui.configbool('web', 'allow' + typ, | ||||
untrusted=True): | ||||
archives.append({ | ||||
'type': typ, | ||||
'extension': spec[2], | ||||
'node': nodeid, | ||||
'url': url, | ||||
}) | ||||
Yuya Nishihara
|
r37533 | return templateutil.mappinglist(archives) | ||
Yuya Nishihara
|
r37531 | |||
Dirkjan Ochtman
|
r6393 | def up(p): | ||
Augie Fackler
|
r36731 | if p[0:1] != "/": | ||
Dirkjan Ochtman
|
r6393 | p = "/" + p | ||
Augie Fackler
|
r36731 | if p[-1:] == "/": | ||
Dirkjan Ochtman
|
r6393 | p = p[:-1] | ||
up = os.path.dirname(p) | ||||
if up == "/": | ||||
return "/" | ||||
return up + "/" | ||||
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 | |||
Pierre-Yves David
|
r18403 | class revnav(object): | ||
Pierre-Yves David
|
r18320 | |||
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 | ||||
Yuya Nishihara
|
r37716 | return templateutil.mappinglist([ | ||
{'before': templateutil.mappinglist([]), | ||||
'after': templateutil.mappinglist([])}, | ||||
]) | ||||
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() | ||
Yuya Nishihara
|
r37715 | navbefore = [{'label': '(%i)' % first, '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: | ||
Yuya Nishihara
|
r37715 | navafter.append({'label': '+%d' % abs(rev - pos), | ||
'node': self.hex(rev)}) | ||||
Pierre-Yves David
|
r18425 | if 0 < rev < pos: | ||
Yuya Nishihara
|
r37715 | navbefore.append({'label': '-%d' % abs(rev - pos), | ||
'node': self.hex(rev)}) | ||||
Nicolas Dumazet
|
r10254 | |||
Yuya Nishihara
|
r37715 | navafter.append({'label': 'tip', 'node': 'tip'}) | ||
Pierre-Yves David
|
r18403 | |||
Yuya Nishihara
|
r37716 | # TODO: maybe this can be a scalar object supporting tomap() | ||
return templateutil.mappinglist([ | ||||
{'before': templateutil.mappinglist(navbefore), | ||||
'after': templateutil.mappinglist(navafter)}, | ||||
]) | ||||
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))) | ||||
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 = { | ||||
'node': s.hex(), | ||||
'rev': s.rev(), | ||||
'user': s.user(), | ||||
'date': s.date(), | ||||
'description': s.description(), | ||||
'branch': s.branch(), | ||||
} | ||||
if util.safehasattr(s, 'path'): | ||||
d['file'] = s.path() | ||||
yield d | ||||
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 | |||
Gregory Szorc
|
r34391 | def difffeatureopts(req, ui, section): | ||
Boris Feld
|
r38585 | diffopts = diffutil.difffeatureopts(ui, untrusted=True, | ||
section=section, whitespace=True) | ||||
Gregory Szorc
|
r34391 | |||
for k in ('ignorews', 'ignorewsamount', 'ignorewseol', '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 | ||||
def annotate(req, fctx, ui): | ||||
diffopts = difffeatureopts(req, ui, 'annotate') | ||||
Yuya Nishihara
|
r37083 | return fctx.annotate(follow=True, diffopts=diffopts) | ||
Jun Wu
|
r30081 | |||
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) | ||
def children(ctx, hide=None): | ||||
return _siblings(ctx.children(), hide) | ||||
Matt Mackall
|
r6434 | def renamelink(fctx): | ||
Matt Mackall
|
r6437 | r = fctx.renamed() | ||
Dirkjan Ochtman
|
r6392 | if r: | ||
Yuya Nishihara
|
r37921 | return templateutil.mappinglist([{'file': r[0], 'node': hex(r[1])}]) | ||
return templateutil.mappinglist([]) | ||||
Dirkjan Ochtman
|
r6392 | |||
def nodetagsdict(repo, node): | ||||
Yuya Nishihara
|
r37922 | return templateutil.hybridlist(repo.nodetags(node), name='name') | ||
Dirkjan Ochtman
|
r6392 | |||
Alexander Solovyov
|
r13596 | def nodebookmarksdict(repo, node): | ||
Yuya Nishihara
|
r37923 | return templateutil.hybridlist(repo.nodebookmarks(node), name='name') | ||
Alexander Solovyov
|
r13596 | |||
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) | ||
return templateutil.hybridlist(branches, name='name') | ||||
Dirkjan Ochtman
|
r6392 | |||
def nodeinbranch(repo, ctx): | ||||
branches = [] | ||||
branch = ctx.branch() | ||||
Brodie Rao
|
r16719 | try: | ||
branchnode = repo.branchtip(branch) | ||||
except error.RepoLookupError: | ||||
branchnode = None | ||||
if branch != 'default' and branchnode != ctx.node(): | ||||
Yuya Nishihara
|
r37925 | branches.append(branch) | ||
return templateutil.hybridlist(branches, name='name') | ||||
Dirkjan Ochtman
|
r6392 | |||
def nodebranchnodefault(ctx): | ||||
branches = [] | ||||
branch = ctx.branch() | ||||
if branch != 'default': | ||||
Yuya Nishihara
|
r37926 | branches.append(branch) | ||
return templateutil.hybridlist(branches, name='name') | ||||
Dirkjan Ochtman
|
r6392 | |||
Yuya Nishihara
|
r37930 | def _nodenamesgen(context, f, node, name): | ||
for t in f(node): | ||||
yield {name: t} | ||||
Yuya Nishihara
|
r37931 | def showtag(repo, t1, node=nullid): | ||
Yuya Nishihara
|
r37930 | args = (repo.nodetags, node, 'tag') | ||
return templateutil.mappinggenerator(_nodenamesgen, args=args, name=t1) | ||||
Dirkjan Ochtman
|
r6392 | |||
Yuya Nishihara
|
r37931 | def showbookmark(repo, t1, node=nullid): | ||
Yuya Nishihara
|
r37930 | args = (repo.nodebookmarks, node, 'bookmark') | ||
return templateutil.mappinggenerator(_nodenamesgen, args=args, name=t1) | ||||
Alexander Solovyov
|
r13596 | |||
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: | ||||
status = 'closed' | ||||
elif ctx.node() not in heads: | ||||
status = 'inactive' | ||||
else: | ||||
status = 'open' | ||||
yield { | ||||
timeless
|
r29216 | 'parity': next(parity), | ||
r26129 | 'branch': ctx.branch(), | |||
'status': status, | ||||
'node': ctx.hex(), | ||||
'date': ctx.date() | ||||
} | ||||
Yuya Nishihara
|
r37932 | return templateutil.mappinggenerator(entries) | ||
r26129 | ||||
Dirkjan Ochtman
|
r6392 | def cleanpath(repo, path): | ||
path = path.lstrip('/') | ||||
Augie Fackler
|
r20033 | return pathutil.canonpath(repo.root, '', path) | ||
Dirkjan Ochtman
|
r6392 | |||
r25999 | def changectx(repo, req): | |||
Weiwen
|
r17991 | changeid = "tip" | ||
Gregory Szorc
|
r36902 | if 'node' in req.qsparams: | ||
changeid = req.qsparams['node'] | ||||
r25999 | ipos = changeid.find(':') | |||
Weiwen
|
r17991 | if ipos != -1: | ||
changeid = changeid[(ipos + 1):] | ||||
Martin von Zweigbergk
|
r37351 | return scmutil.revsymbol(repo, changeid) | ||
Weiwen
|
r17991 | |||
def basechangectx(repo, req): | ||||
Gregory Szorc
|
r36902 | if 'node' in req.qsparams: | ||
changeid = req.qsparams['node'] | ||||
r25999 | ipos = changeid.find(':') | |||
Weiwen
|
r17991 | if ipos != -1: | ||
changeid = changeid[:ipos] | ||||
Martin von Zweigbergk
|
r37351 | return scmutil.revsymbol(repo, changeid) | ||
Weiwen
|
r17991 | |||
return None | ||||
Dirkjan Ochtman
|
r6392 | def filectx(repo, req): | ||
Gregory Szorc
|
r36902 | if 'file' not in req.qsparams: | ||
Ross Lagerwall
|
r17289 | raise ErrorResponse(HTTP_NOT_FOUND, 'file not given') | ||
Gregory Szorc
|
r36902 | path = cleanpath(repo, req.qsparams['file']) | ||
if 'node' in req.qsparams: | ||||
changeid = req.qsparams['node'] | ||||
elif 'filenode' in req.qsparams: | ||||
changeid = req.qsparams['filenode'] | ||||
Dirkjan Ochtman
|
r6392 | else: | ||
Ross Lagerwall
|
r17289 | raise ErrorResponse(HTTP_NOT_FOUND, '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 | |||
Denis Laxalde
|
r31665 | def linerange(req): | ||
Gregory Szorc
|
r36902 | linerange = req.qsparams.getall('linerange') | ||
Gregory Szorc
|
r36881 | if not linerange: | ||
Denis Laxalde
|
r31665 | return None | ||
if len(linerange) > 1: | ||||
raise ErrorResponse(HTTP_BAD_REQUEST, | ||||
'redundant linerange parameter') | ||||
try: | ||||
fromline, toline = map(int, linerange[0].split(':', 1)) | ||||
except ValueError: | ||||
raise ErrorResponse(HTTP_BAD_REQUEST, | ||||
'invalid linerange parameter') | ||||
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 | |||
def formatlinerange(fromline, toline): | ||||
return '%d:%d' % (fromline + 1, toline) | ||||
Yuya Nishihara
|
r37933 | def _succsandmarkersgen(context, mapping): | ||
Yuya Nishihara
|
r36612 | repo = context.resource(mapping, 'repo') | ||
Yuya Nishihara
|
r37521 | itemmappings = templatekw.showsuccsandmarkers(context, mapping) | ||
for item in itemmappings.tovalue(context, mapping): | ||||
r35502 | item['successors'] = _siblings(repo[successor] | |||
for successor in item['successors']) | ||||
yield item | ||||
r35501 | ||||
Yuya Nishihara
|
r37933 | def succsandmarkers(context, mapping): | ||
return templateutil.mappinggenerator(_succsandmarkersgen, args=(mapping,)) | ||||
Yuya Nishihara
|
r36612 | # teach templater succsandmarkers is switched to (context, mapping) API | ||
Yuya Nishihara
|
r37087 | succsandmarkers._requires = {'repo', 'ctx'} | ||
Yuya Nishihara
|
r36612 | |||
Yuya Nishihara
|
r37934 | def _whyunstablegen(context, mapping): | ||
r36973 | repo = context.resource(mapping, 'repo') | |||
ctx = context.resource(mapping, 'ctx') | ||||
entries = obsutil.whyunstable(repo, ctx) | ||||
for entry in entries: | ||||
if entry.get('divergentnodes'): | ||||
entry['divergentnodes'] = _siblings(entry['divergentnodes']) | ||||
yield entry | ||||
Yuya Nishihara
|
r37934 | def whyunstable(context, mapping): | ||
return templateutil.mappinggenerator(_whyunstablegen, args=(mapping,)) | ||||
Yuya Nishihara
|
r37087 | whyunstable._requires = {'repo', 'ctx'} | ||
r36973 | ||||
Yuya Nishihara
|
r38965 | # helper to mark a function as a new-style template keyword; can be removed | ||
# once old-style function gets unsupported and new-style becomes the default | ||||
def _kwfunc(f): | ||||
f._requires = () | ||||
return f | ||||
r27294 | def commonentry(repo, ctx): | |||
node = ctx.node() | ||||
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. | ||||
'ctx': ctx, | ||||
r27294 | 'rev': ctx.rev(), | |||
'node': hex(node), | ||||
'author': ctx.user(), | ||||
'desc': ctx.description(), | ||||
'date': ctx.date(), | ||||
'extra': ctx.extra(), | ||||
'phase': ctx.phasestr(), | ||||
r35087 | 'obsolete': ctx.obsolete(), | |||
Yuya Nishihara
|
r36535 | 'succsandmarkers': succsandmarkers, | ||
Yuya Nishihara
|
r37935 | 'instabilities': templateutil.hybridlist(ctx.instabilities(), | ||
name='instability'), | ||||
r36973 | 'whyunstable': whyunstable, | |||
r27294 | 'branch': nodebranchnodefault(ctx), | |||
'inbranch': nodeinbranch(repo, ctx), | ||||
'branches': nodebranchdict(repo, ctx), | ||||
'tags': nodetagsdict(repo, node), | ||||
'bookmarks': nodebookmarksdict(repo, node), | ||||
Yuya Nishihara
|
r38965 | 'parent': _kwfunc(lambda context, mapping: parents(ctx)), | ||
'child': _kwfunc(lambda context, mapping: children(ctx)), | ||||
r27294 | } | |||
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() | ||||
n = ctx.node() | ||||
Yuya Nishihara
|
r37931 | showtags = showtag(repo, 'changelogtag', n) | ||
Yuya Nishihara
|
r37973 | files = listfilediffs(ctx.files(), n, web.maxfiles) | ||
Gregory Szorc
|
r23745 | |||
r27294 | entry = commonentry(repo, ctx) | |||
entry.update( | ||||
Yuya Nishihara
|
r38965 | allparents=_kwfunc(lambda context, mapping: parents(ctx)), | ||
parent=_kwfunc(lambda context, mapping: parents(ctx, rev - 1)), | ||||
child=_kwfunc(lambda context, mapping: children(ctx, rev + 1)), | ||||
r27294 | changelogtag=showtags, | |||
files=files, | ||||
) | ||||
return entry | ||||
Gregory Szorc
|
r23745 | |||
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]) | ||||
entry['parity'] = next(parityfn) | ||||
yield entry | ||||
r25602 | def symrevorshortnode(req, ctx): | |||
Gregory Szorc
|
r36902 | if 'node' in req.qsparams: | ||
return templatefilters.revescape(req.qsparams['node']) | ||||
r25602 | else: | |||
return short(ctx.node()) | ||||
Yuya Nishihara
|
r37970 | def _listfilesgen(context, ctx, stripecount): | ||
Yuya Nishihara
|
r37968 | parity = paritygen(stripecount) | ||
for blockno, f in enumerate(ctx.files()): | ||||
template = 'filenodelink' if f in ctx else 'filenolink' | ||||
Yuya Nishihara
|
r37970 | yield context.process(template, { | ||
Yuya Nishihara
|
r37968 | 'node': ctx.hex(), | ||
'file': f, | ||||
'blockno': blockno + 1, | ||||
'parity': next(parity), | ||||
}) | ||||
Gregory Szorc
|
r36902 | def changesetentry(web, ctx): | ||
Gregory Szorc
|
r24177 | '''Obtain a dictionary to be used to render the "changeset" template.''' | ||
Yuya Nishihara
|
r37931 | showtags = showtag(web.repo, 'changesettag', ctx.node()) | ||
showbookmarks = showbookmark(web.repo, 'changesetbookmark', ctx.node()) | ||||
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() | ||||
Boris Feld
|
r34243 | style = web.config('web', 'style') | ||
Gregory Szorc
|
r36902 | if 'style' in web.req.qsparams: | ||
style = web.req.qsparams['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, | ||||
Yuya Nishihara
|
r37969 | files=templateutil.mappedgenerator(_listfilesgen, | ||
Yuya Nishihara
|
r37970 | args=(ctx, web.stripecount)), | ||
Yuya Nishihara
|
r38965 | diffsummary=_kwfunc(lambda context, mapping: diffsummary(diffstatsgen)), | ||
Gregory Szorc
|
r24177 | diffstat=diffstats, | ||
archives=web.archivelist(ctx.hex()), | ||||
Pulkit Goyal
|
r36452 | **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]: | ||
Yuya Nishihara
|
r37972 | yield context.process('filedifflink', {'node': hex(node), 'file': f}) | ||
Dirkjan Ochtman
|
r7311 | if len(files) > max: | ||
Yuya Nishihara
|
r37972 | yield context.process('fileellipses', {}) | ||
Dirkjan Ochtman
|
r7311 | |||
Yuya Nishihara
|
r37973 | def listfilediffs(files, node, max): | ||
Yuya Nishihara
|
r37971 | return templateutil.mappedgenerator(_listfilediffsgen, | ||
Yuya Nishihara
|
r37972 | 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): | ||
difflineno = "%d.%d" % (blockno, lineno) | ||||
if l.startswith('+'): | ||||
ltype = "difflineplus" | ||||
elif l.startswith('-'): | ||||
ltype = "difflineminus" | ||||
elif l.startswith('@'): | ||||
ltype = "difflineat" | ||||
else: | ||||
ltype = "diffline" | ||||
Yuya Nishihara
|
r38006 | yield context.process(ltype, { | ||
Yuya Nishihara
|
r38004 | 'line': l, | ||
'lineno': lineno, | ||||
'lineid': lineidprefix + "l%s" % difflineno, | ||||
'linenumber': "% 8s" % difflineno, | ||||
}) | ||||
Yuya Nishihara
|
r38007 | def _diffsgen(context, repo, ctx, basectx, files, style, stripecount, | ||
linerange, lineidprefix): | ||||
Dirkjan Ochtman
|
r7310 | if files: | ||
m = match.exact(repo.root, repo.getcwd(), files) | ||||
else: | ||||
m = match.always(repo.root, repo.getcwd()) | ||||
diffopts = patch.diffopts(repo.ui, untrusted=True) | ||||
Denis Laxalde
|
r31082 | node1 = basectx.node() | ||
Dirkjan Ochtman
|
r7310 | node2 = ctx.node() | ||
Yuya Nishihara
|
r38007 | parity = paritygen(stripecount) | ||
Dirkjan Ochtman
|
r7310 | |||
Denis Laxalde
|
r31276 | diffhunks = patch.diffhunks(repo, node1, node2, m, opts=diffopts) | ||
Denis Laxalde
|
r34856 | for blockno, (fctx1, fctx2, header, hunks) in enumerate(diffhunks, 1): | ||
Denis Laxalde
|
r31276 | if style != 'raw': | ||
header = header[1:] | ||||
lines = [h + '\n' for h in header] | ||||
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: | ||||
Yuya Nishihara
|
r38005 | l = templateutil.mappedgenerator(_prettyprintdifflines, | ||
Yuya Nishihara
|
r38006 | args=(lines, blockno, | ||
Yuya Nishihara
|
r38005 | lineidprefix)) | ||
Yuya Nishihara
|
r38007 | yield { | ||
Yuya Nishihara
|
r37037 | 'parity': next(parity), | ||
'blockno': blockno, | ||||
Yuya Nishihara
|
r38005 | 'lines': l, | ||
Yuya Nishihara
|
r38007 | } | ||
def diffs(web, ctx, basectx, files, style, linerange=None, lineidprefix=''): | ||||
args = (web.repo, ctx, basectx, files, style, web.stripecount, | ||||
linerange, lineidprefix) | ||||
return templateutil.mappinggenerator(_diffsgen, args=args, name='diffblock') | ||||
Dirkjan Ochtman
|
r7345 | |||
Yuya Nishihara
|
r38010 | def _compline(type, leftlineno, leftline, rightlineno, rightline): | ||
Yuya Nishihara
|
r38008 | lineid = leftlineno and ("l%d" % leftlineno) or '' | ||
lineid += rightlineno and ("r%d" % rightlineno) or '' | ||||
llno = '%d' % leftlineno if leftlineno else '' | ||||
rlno = '%d' % rightlineno if rightlineno else '' | ||||
Yuya Nishihara
|
r38010 | return { | ||
Yuya Nishihara
|
r38008 | 'type': type, | ||
'lineid': lineid, | ||||
'leftlineno': leftlineno, | ||||
'leftlinenumber': "% 6s" % llno, | ||||
'leftline': leftline or '', | ||||
'rightlineno': rightlineno, | ||||
'rightlinenumber': "% 6s" % rlno, | ||||
'rightline': rightline or '', | ||||
Yuya Nishihara
|
r38010 | } | ||
Yuya Nishihara
|
r38008 | |||
Yuya Nishihara
|
r38010 | def _getcompblockgen(context, leftlines, rightlines, opcodes): | ||
Yuya Nishihara
|
r38009 | for type, llo, lhi, rlo, rhi in opcodes: | ||
len1 = lhi - llo | ||||
len2 = rhi - rlo | ||||
count = min(len1, len2) | ||||
Gregory Szorc
|
r38806 | for i in pycompat.xrange(count): | ||
Yuya Nishihara
|
r38010 | yield _compline(type=type, | ||
Yuya Nishihara
|
r38009 | leftlineno=llo + i + 1, | ||
leftline=leftlines[llo + i], | ||||
rightlineno=rlo + i + 1, | ||||
rightline=rightlines[rlo + i]) | ||||
if len1 > len2: | ||||
Gregory Szorc
|
r38806 | for i in pycompat.xrange(llo + count, lhi): | ||
Yuya Nishihara
|
r38010 | yield _compline(type=type, | ||
Yuya Nishihara
|
r38009 | leftlineno=i + 1, | ||
leftline=leftlines[i], | ||||
rightlineno=None, | ||||
rightline=None) | ||||
elif len2 > len1: | ||||
Gregory Szorc
|
r38806 | for i in pycompat.xrange(rlo + count, rhi): | ||
Yuya Nishihara
|
r38010 | yield _compline(type=type, | ||
Yuya Nishihara
|
r38009 | 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) | ||||
return templateutil.mappinggenerator(_getcompblockgen, args=args, | ||||
name='comparisonline') | ||||
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()) | ||
Yuya Nishihara
|
r38012 | yield {'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) | ||
Yuya Nishihara
|
r38012 | yield {'lines': l} | ||
Yuya Nishihara
|
r38013 | def compare(contextnum, leftlines, rightlines): | ||
Yuya Nishihara
|
r38012 | args = (contextnum, leftlines, rightlines) | ||
return templateutil.mappinggenerator(_comparegen, args=args, | ||||
name='comparisonblock') | ||||
wujek srujek
|
r17202 | |||
Yuya Nishihara
|
r38604 | def diffstatgen(ui, ctx, basectx): | ||
Steven Brown
|
r14570 | '''Generator function that provides the diffstat data.''' | ||
Steven Brown
|
r14490 | |||
Yuya Nishihara
|
r38604 | diffopts = patch.diffopts(ui, {'noprefix': False}) | ||
Yuya Nishihara
|
r35445 | stats = patch.diffstatdata( | ||
Boris Feld
|
r38585 | 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 | ||||
def diffsummary(statgen): | ||||
'''Return a short summary of the diff.''' | ||||
timeless
|
r29216 | stats, maxname, maxtotal, addtotal, removetotal, binary = next(statgen) | ||
Steven Brown
|
r14570 | return _(' %d files changed, %d insertions(+), %d deletions(-)\n') % ( | ||
len(stats), addtotal, removetotal) | ||||
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: | ||
r35315 | template = 'diffstatlink' if filename in files else 'diffstatnolink' | |||
Steven Brown
|
r14561 | total = adds + removes | ||
Steven Brown
|
r14562 | fileno += 1 | ||
Yuya Nishihara
|
r38069 | yield context.process(template, { | ||
Yuya Nishihara
|
r37037 | 'node': ctx.hex(), | ||
'file': filename, | ||||
'fileno': fileno, | ||||
'total': total, | ||||
'addpct': pct(adds), | ||||
'removepct': pct(removes), | ||||
'parity': next(parity), | ||||
}) | ||||
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) | ||
Yuya Nishihara
|
r37714 | class sessionvars(templateutil.wrapped): | ||
Dirkjan Ochtman
|
r7345 | def __init__(self, vars, start='?'): | ||
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): | ||
raise error.ParseError(_('not comparable')) | ||||
def getmax(self, context, mapping): | ||||
raise error.ParseError(_('not comparable')) | ||||
Yuya Nishihara
|
r38467 | def filter(self, context, mapping, select): | ||
# implement if necessary | ||||
raise error.ParseError(_('not filterable')) | ||||
Yuya Nishihara
|
r37714 | def itermaps(self, context): | ||
Yuya Nishihara
|
r37712 | separator = self._start | ||
for key, value in sorted(self._vars.iteritems()): | ||||
Augie Fackler
|
r34808 | yield {'name': key, | ||
'value': pycompat.bytestr(value), | ||||
'separator': separator, | ||||
} | ||||
Dirkjan Ochtman
|
r7345 | separator = '&' | ||
Augie Fackler
|
r12691 | |||
Yuya Nishihara
|
r37714 | def join(self, context, mapping, sep): | ||
# could be '{separator}{name}={value|urlescape}' | ||||
raise error.ParseError(_('not displayable without template')) | ||||
def show(self, context, mapping): | ||||
return self.join(context, '') | ||||
Yuya Nishihara
|
r38308 | def tobool(self, context, mapping): | ||
return bool(self._vars) | ||||
Yuya Nishihara
|
r37714 | def tovalue(self, context, mapping): | ||
return self._vars | ||||
Yuya Nishihara
|
r27007 | class wsgiui(uimod.ui): | ||
Augie Fackler
|
r12691 | # default termwidth breaks under mod_wsgi | ||
def termwidth(self): | ||||
return 80 | ||||
Gregory Szorc
|
r26162 | |||
def getwebsubs(repo): | ||||
websubtable = [] | ||||
websubdefs = repo.ui.configitems('websub') | ||||
# we must maintain interhg backwards compatibility | ||||
websubdefs += repo.ui.configitems('interhg') | ||||
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])*$' | ||
Gregory Szorc
|
r26162 | % (delim, delim, delim), pattern) | ||
if not match: | ||||
repo.ui.warn(_("websub: invalid pattern for %s: %s\n") | ||||
% (key, pattern)) | ||||
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: | ||||
for flag in flagin.upper(): | ||||
flags |= re.__dict__[flag] | ||||
try: | ||||
regexp = re.compile(regexp, flags) | ||||
websubtable.append((regexp, format)) | ||||
except re.error: | ||||
repo.ui.warn(_("websub: invalid regexp for %s: %s\n") | ||||
% (key, regexp)) | ||||
return websubtable | ||||
r37928 | ||||
def getgraphnode(repo, ctx): | ||||
return (templatekw.getgraphnodecurrent(repo, ctx) + | ||||
templatekw.getgraphnodesymbol(ctx)) | ||||