webutil.py
499 lines
| 15.9 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 | |||
wujek srujek
|
r17302 | import os, copy | ||
FUJIWARA Katsunori
|
r21122 | from mercurial import match, patch, error, ui, util, pathutil, context | ||
Steven Brown
|
r14570 | from mercurial.i18n import _ | ||
Dirkjan Ochtman
|
r6392 | from mercurial.node import hex, nullid | ||
Gregory Szorc
|
r24177 | from common import ErrorResponse, paritygen | ||
Ross Lagerwall
|
r17289 | from common import HTTP_NOT_FOUND | ||
wujek srujek
|
r17202 | import difflib | ||
Dirkjan Ochtman
|
r6392 | |||
Dirkjan Ochtman
|
r6393 | def up(p): | ||
if p[0] != "/": | ||||
p = "/" + p | ||||
if p[-1] == "/": | ||||
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 | ||
def _first(self): | ||||
"""return the minimum non-filtered changeset or None""" | ||||
try: | ||||
return iter(self._revlog).next() | ||||
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: | ||
- a single element tuple | ||||
- containing a dictionary with a `before` and `after` key | ||||
- values are generator functions taking arbitrary number of kwargs | ||||
- yield items are dictionaries with `label` and `node` keys | ||||
""" | ||||
Pierre-Yves David
|
r18406 | if not self: | ||
# empty repo | ||||
return ({'before': (), 'after': ()},) | ||||
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() | ||
navbefore = [("(%i)" % first, 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: | ||
Pierre-Yves David
|
r18503 | navafter.append(("+%d" % abs(rev - pos), self.hex(rev))) | ||
Pierre-Yves David
|
r18425 | if 0 < rev < pos: | ||
Pierre-Yves David
|
r18503 | navbefore.append(("-%d" % abs(rev - pos), self.hex(rev))) | ||
Pierre-Yves David
|
r18425 | |||
Nicolas Dumazet
|
r10254 | |||
Pierre-Yves David
|
r18403 | navafter.append(("tip", "tip")) | ||
data = lambda i: {"label": i[0], "node": i[1]} | ||||
return ({'before': lambda **map: (data(i) for i in navbefore), | ||||
'after': lambda **map: (data(i) for i in 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))) | ||||
Pierre-Yves David
|
r18408 | |||
Dirkjan Ochtman
|
r7671 | def _siblings(siblings=[], hiderev=None): | ||
Dirkjan Ochtman
|
r6392 | siblings = [s for s in siblings if s.node() != nullid] | ||
if len(siblings) == 1 and siblings[0].rev() == hiderev: | ||||
return | ||||
for s in siblings: | ||||
Alexander Solovyov
|
r14055 | d = {'node': s.hex(), 'rev': s.rev()} | ||
Dirkjan Ochtman
|
r7294 | d['user'] = s.user() | ||
d['date'] = s.date() | ||||
d['description'] = s.description() | ||||
Dirkjan Ochtman
|
r7717 | d['branch'] = s.branch() | ||
Augie Fackler
|
r14957 | if util.safehasattr(s, 'path'): | ||
Dirkjan Ochtman
|
r6392 | d['file'] = s.path() | ||
yield d | ||||
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: | ||
Augie Fackler
|
r20681 | return [{'file': r[0], 'node': hex(r[1])}] | ||
Dirkjan Ochtman
|
r6392 | return [] | ||
def nodetagsdict(repo, node): | ||||
return [{"name": i} for i in repo.nodetags(node)] | ||||
Alexander Solovyov
|
r13596 | def nodebookmarksdict(repo, node): | ||
return [{"name": i} for i in repo.nodebookmarks(node)] | ||||
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(): | ||||
Dirkjan Ochtman
|
r6392 | branches.append({"name": branch}) | ||
return branches | ||||
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(): | ||||
Dirkjan Ochtman
|
r6392 | branches.append({"name": branch}) | ||
return branches | ||||
def nodebranchnodefault(ctx): | ||||
branches = [] | ||||
branch = ctx.branch() | ||||
if branch != 'default': | ||||
branches.append({"name": branch}) | ||||
return branches | ||||
def showtag(repo, tmpl, t1, node=nullid, **args): | ||||
for t in repo.nodetags(node): | ||||
yield tmpl(t1, tag=t, **args) | ||||
Alexander Solovyov
|
r13596 | def showbookmark(repo, tmpl, t1, node=nullid, **args): | ||
for t in repo.nodebookmarks(node): | ||||
yield tmpl(t1, bookmark=t, **args) | ||||
Dirkjan Ochtman
|
r6392 | def cleanpath(repo, path): | ||
path = path.lstrip('/') | ||||
Augie Fackler
|
r20033 | return pathutil.canonpath(repo.root, '', path) | ||
Dirkjan Ochtman
|
r6392 | |||
Weiwen
|
r17991 | def changeidctx (repo, changeid): | ||
Dirkjan Ochtman
|
r6392 | try: | ||
Matt Mackall
|
r6747 | ctx = repo[changeid] | ||
Matt Mackall
|
r7637 | except error.RepoError: | ||
Dirkjan Ochtman
|
r6392 | man = repo.manifest | ||
Matt Mackall
|
r7361 | ctx = repo[man.linkrev(man.rev(man.lookup(changeid)))] | ||
Dirkjan Ochtman
|
r6392 | |||
return ctx | ||||
Weiwen
|
r17991 | def changectx (repo, req): | ||
changeid = "tip" | ||||
if 'node' in req.form: | ||||
changeid = req.form['node'][0] | ||||
ipos=changeid.find(':') | ||||
if ipos != -1: | ||||
changeid = changeid[(ipos + 1):] | ||||
elif 'manifest' in req.form: | ||||
changeid = req.form['manifest'][0] | ||||
return changeidctx(repo, changeid) | ||||
def basechangectx(repo, req): | ||||
if 'node' in req.form: | ||||
changeid = req.form['node'][0] | ||||
ipos=changeid.find(':') | ||||
if ipos != -1: | ||||
changeid = changeid[:ipos] | ||||
return changeidctx(repo, changeid) | ||||
return None | ||||
Dirkjan Ochtman
|
r6392 | def filectx(repo, req): | ||
Ross Lagerwall
|
r17289 | if 'file' not in req.form: | ||
raise ErrorResponse(HTTP_NOT_FOUND, 'file not given') | ||||
Dirkjan Ochtman
|
r6392 | path = cleanpath(repo, req.form['file'][0]) | ||
if 'node' in req.form: | ||||
changeid = req.form['node'][0] | ||||
Ross Lagerwall
|
r17289 | elif 'filenode' in req.form: | ||
changeid = req.form['filenode'][0] | ||||
Dirkjan Ochtman
|
r6392 | else: | ||
Ross Lagerwall
|
r17289 | raise ErrorResponse(HTTP_NOT_FOUND, 'node or filenode not given') | ||
Dirkjan Ochtman
|
r6392 | try: | ||
Matt Mackall
|
r6747 | fctx = repo[changeid][path] | ||
Matt Mackall
|
r7637 | except error.RepoError: | ||
Dirkjan Ochtman
|
r6392 | fctx = repo.filectx(path, fileid=changeid) | ||
return fctx | ||||
Dirkjan Ochtman
|
r7310 | |||
Gregory Szorc
|
r23745 | def changelistentry(web, ctx, tmpl): | ||
'''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() | ||||
showtags = showtag(repo, tmpl, 'changelogtag', n) | ||||
files = listfilediffs(tmpl, ctx.files(), n, web.maxfiles) | ||||
return { | ||||
"author": ctx.user(), | ||||
"parent": parents(ctx, rev - 1), | ||||
"child": children(ctx, rev + 1), | ||||
"changelogtag": showtags, | ||||
"desc": ctx.description(), | ||||
"extra": ctx.extra(), | ||||
"date": ctx.date(), | ||||
"files": files, | ||||
"rev": rev, | ||||
"node": hex(n), | ||||
"tags": nodetagsdict(repo, n), | ||||
"bookmarks": nodebookmarksdict(repo, n), | ||||
"inbranch": nodeinbranch(repo, ctx), | ||||
"branches": nodebranchdict(repo, ctx) | ||||
} | ||||
Gregory Szorc
|
r24177 | def changesetentry(web, req, tmpl, ctx): | ||
'''Obtain a dictionary to be used to render the "changeset" template.''' | ||||
showtags = showtag(web.repo, tmpl, 'changesettag', ctx.node()) | ||||
showbookmarks = showbookmark(web.repo, tmpl, 'changesetbookmark', | ||||
ctx.node()) | ||||
showbranch = nodebranchnodefault(ctx) | ||||
files = [] | ||||
parity = paritygen(web.stripecount) | ||||
for blockno, f in enumerate(ctx.files()): | ||||
template = f in ctx and 'filenodelink' or 'filenolink' | ||||
files.append(tmpl(template, | ||||
node=ctx.hex(), file=f, blockno=blockno + 1, | ||||
parity=parity.next())) | ||||
basectx = basechangectx(web.repo, req) | ||||
if basectx is None: | ||||
basectx = ctx.p1() | ||||
style = web.config('web', 'style', 'paper') | ||||
if 'style' in req.form: | ||||
style = req.form['style'][0] | ||||
parity = paritygen(web.stripecount) | ||||
diff = diffs(web.repo, tmpl, ctx, basectx, None, parity, style) | ||||
parity = paritygen(web.stripecount) | ||||
diffstatsgen = diffstatgen(ctx, basectx) | ||||
diffstats = diffstat(tmpl, ctx, diffstatsgen, parity) | ||||
return dict( | ||||
diff=diff, | ||||
rev=ctx.rev(), | ||||
node=ctx.hex(), | ||||
parent=tuple(parents(ctx)), | ||||
child=children(ctx), | ||||
basenode=basectx.hex(), | ||||
changesettag=showtags, | ||||
changesetbookmark=showbookmarks, | ||||
changesetbranch=showbranch, | ||||
author=ctx.user(), | ||||
desc=ctx.description(), | ||||
extra=ctx.extra(), | ||||
date=ctx.date(), | ||||
files=files, | ||||
diffsummary=lambda **x: diffsummary(diffstatsgen), | ||||
diffstat=diffstats, | ||||
archives=web.archivelist(ctx.hex()), | ||||
tags=nodetagsdict(web.repo, ctx.node()), | ||||
bookmarks=nodebookmarksdict(web.repo, ctx.node()), | ||||
branch=nodebranchnodefault(ctx), | ||||
inbranch=nodeinbranch(web.repo, ctx), | ||||
branches=nodebranchdict(web.repo, ctx)) | ||||
Dirkjan Ochtman
|
r7311 | def listfilediffs(tmpl, files, node, max): | ||
for f in files[:max]: | ||||
yield tmpl('filedifflink', node=hex(node), file=f) | ||||
if len(files) > max: | ||||
yield tmpl('fileellipses') | ||||
Weiwen
|
r17991 | def diffs(repo, tmpl, ctx, basectx, files, parity, style): | ||
Dirkjan Ochtman
|
r7310 | |||
def countgen(): | ||||
start = 1 | ||||
while True: | ||||
yield start | ||||
start += 1 | ||||
blockcount = countgen() | ||||
Paul Boddie
|
r16308 | def prettyprintlines(diff, blockno): | ||
Dirkjan Ochtman
|
r7310 | for lineno, l in enumerate(diff.splitlines(True)): | ||
lineno = "%d.%d" % (blockno, lineno + 1) | ||||
if l.startswith('+'): | ||||
ltype = "difflineplus" | ||||
elif l.startswith('-'): | ||||
ltype = "difflineminus" | ||||
elif l.startswith('@'): | ||||
ltype = "difflineat" | ||||
else: | ||||
ltype = "diffline" | ||||
yield tmpl(ltype, | ||||
line=l, | ||||
lineid="l%s" % lineno, | ||||
linenumber="% 8s" % lineno) | ||||
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) | ||||
Weiwen
|
r17991 | if basectx is None: | ||
parents = ctx.parents() | ||||
Jordi Gutiérrez Hermoso
|
r24306 | if parents: | ||
node1 = parents[0].node() | ||||
else: | ||||
node1 = nullid | ||||
Weiwen
|
r17991 | else: | ||
node1 = basectx.node() | ||||
Dirkjan Ochtman
|
r7310 | node2 = ctx.node() | ||
block = [] | ||||
for chunk in patch.diff(repo, node1, node2, m, opts=diffopts): | ||||
if chunk.startswith('diff') and block: | ||||
Paul Boddie
|
r16308 | blockno = blockcount.next() | ||
yield tmpl('diffblock', parity=parity.next(), blockno=blockno, | ||||
lines=prettyprintlines(''.join(block), blockno)) | ||||
Dirkjan Ochtman
|
r7310 | block = [] | ||
Dirkjan Ochtman
|
r9402 | if chunk.startswith('diff') and style != 'raw': | ||
Dirkjan Ochtman
|
r7310 | chunk = ''.join(chunk.splitlines(True)[1:]) | ||
block.append(chunk) | ||||
Paul Boddie
|
r16308 | blockno = blockcount.next() | ||
yield tmpl('diffblock', parity=parity.next(), blockno=blockno, | ||||
lines=prettyprintlines(''.join(block), blockno)) | ||||
Dirkjan Ochtman
|
r7345 | |||
wujek srujek
|
r17302 | def compare(tmpl, context, leftlines, rightlines): | ||
wujek srujek
|
r17202 | '''Generator function that provides side-by-side comparison data.''' | ||
def compline(type, leftlineno, leftline, rightlineno, rightline): | ||||
lineid = leftlineno and ("l%s" % leftlineno) or '' | ||||
lineid += rightlineno and ("r%s" % rightlineno) or '' | ||||
return tmpl('comparisonline', | ||||
type=type, | ||||
lineid=lineid, | ||||
leftlinenumber="% 6s" % (leftlineno or ''), | ||||
leftline=leftline or '', | ||||
rightlinenumber="% 6s" % (rightlineno or ''), | ||||
rightline=rightline or '') | ||||
def getblock(opcodes): | ||||
for type, llo, lhi, rlo, rhi in opcodes: | ||||
len1 = lhi - llo | ||||
len2 = rhi - rlo | ||||
count = min(len1, len2) | ||||
for i in xrange(count): | ||||
yield compline(type=type, | ||||
leftlineno=llo + i + 1, | ||||
leftline=leftlines[llo + i], | ||||
rightlineno=rlo + i + 1, | ||||
rightline=rightlines[rlo + i]) | ||||
if len1 > len2: | ||||
for i in xrange(llo + count, lhi): | ||||
yield compline(type=type, | ||||
leftlineno=i + 1, | ||||
leftline=leftlines[i], | ||||
rightlineno=None, | ||||
rightline=None) | ||||
elif len2 > len1: | ||||
for i in xrange(rlo + count, rhi): | ||||
yield compline(type=type, | ||||
leftlineno=None, | ||||
leftline=None, | ||||
rightlineno=i + 1, | ||||
rightline=rightlines[i]) | ||||
s = difflib.SequenceMatcher(None, leftlines, rightlines) | ||||
if context < 0: | ||||
wujek srujek
|
r17302 | yield tmpl('comparisonblock', lines=getblock(s.get_opcodes())) | ||
wujek srujek
|
r17202 | else: | ||
wujek srujek
|
r17302 | for oc in s.get_grouped_opcodes(n=context): | ||
yield tmpl('comparisonblock', lines=getblock(oc)) | ||||
wujek srujek
|
r17202 | |||
Weiwen
|
r17991 | def diffstatgen(ctx, basectx): | ||
Steven Brown
|
r14570 | '''Generator function that provides the diffstat data.''' | ||
Steven Brown
|
r14490 | |||
Weiwen
|
r17991 | stats = patch.diffstatdata(util.iterlines(ctx.diff(basectx))) | ||
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.''' | ||||
stats, maxname, maxtotal, addtotal, removetotal, binary = statgen.next() | ||||
return _(' %d files changed, %d insertions(+), %d deletions(-)\n') % ( | ||||
len(stats), addtotal, removetotal) | ||||
def diffstat(tmpl, ctx, statgen, parity): | ||||
'''Return a diffstat template for each file in the diff.''' | ||||
stats, maxname, maxtotal, addtotal, removetotal, binary = statgen.next() | ||||
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: | ||
template = filename in files and 'diffstatlink' or 'diffstatnolink' | ||||
total = adds + removes | ||||
Steven Brown
|
r14562 | fileno += 1 | ||
yield tmpl(template, node=ctx.hex(), file=filename, fileno=fileno, | ||||
Steven Brown
|
r14561 | total=total, addpct=pct(adds), removepct=pct(removes), | ||
parity=parity.next()) | ||||
Steven Brown
|
r14490 | |||
Dirkjan Ochtman
|
r7345 | class sessionvars(object): | ||
def __init__(self, vars, start='?'): | ||||
self.start = start | ||||
self.vars = vars | ||||
def __getitem__(self, key): | ||||
return self.vars[key] | ||||
def __setitem__(self, key, value): | ||||
self.vars[key] = value | ||||
def __copy__(self): | ||||
return sessionvars(copy.copy(self.vars), self.start) | ||||
def __iter__(self): | ||||
separator = self.start | ||||
Mads Kiilerich
|
r18367 | for key, value in sorted(self.vars.iteritems()): | ||
Dirkjan Ochtman
|
r7345 | yield {'name': key, 'value': str(value), 'separator': separator} | ||
separator = '&' | ||||
Augie Fackler
|
r12691 | |||
class wsgiui(ui.ui): | ||||
# default termwidth breaks under mod_wsgi | ||||
def termwidth(self): | ||||
return 80 | ||||