webutil.py
269 lines
| 8.0 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 | |||
Dirkjan Ochtman
|
r7345 | import os, copy | ||
Steven Brown
|
r14490 | from mercurial import match, patch, scmutil, error, ui, util | ||
Steven Brown
|
r14570 | from mercurial.i18n import _ | ||
Dirkjan Ochtman
|
r6392 | from mercurial.node import hex, nullid | ||
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 + "/" | ||||
def revnavgen(pos, pagelen, limit, nodefunc): | ||||
def seq(factor, limit=None): | ||||
if limit: | ||||
yield limit | ||||
if limit >= 20 and limit <= 40: | ||||
yield 50 | ||||
else: | ||||
yield 1 * factor | ||||
yield 3 * factor | ||||
for f in seq(factor * 10): | ||||
yield f | ||||
Nicolas Dumazet
|
r10254 | navbefore = [] | ||
navafter = [] | ||||
Dirkjan Ochtman
|
r6393 | |||
Nicolas Dumazet
|
r10254 | last = 0 | ||
for f in seq(1, pagelen): | ||||
if f < pagelen or f <= last: | ||||
continue | ||||
if f > limit: | ||||
break | ||||
last = f | ||||
if pos + f < limit: | ||||
navafter.append(("+%d" % f, hex(nodefunc(pos + f).node()))) | ||||
if pos - f >= 0: | ||||
navbefore.insert(0, ("-%d" % f, hex(nodefunc(pos - f).node()))) | ||||
Dirkjan Ochtman
|
r6393 | |||
Nicolas Dumazet
|
r10254 | navafter.append(("tip", "tip")) | ||
try: | ||||
navbefore.insert(0, ("(0)", hex(nodefunc('0').node()))) | ||||
except error.RepoError: | ||||
pass | ||||
def gen(l): | ||||
def f(**map): | ||||
Dirkjan Ochtman
|
r6393 | for label, node in l: | ||
yield {"label": label, "node": node} | ||||
Nicolas Dumazet
|
r10254 | return f | ||
Dirkjan Ochtman
|
r6393 | |||
Matt Mackall
|
r10282 | return (dict(before=gen(navbefore), after=gen(navafter)),) | ||
Dirkjan Ochtman
|
r6393 | |||
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): | ||
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: | ||
return [dict(file=r[0], node=hex(r[1]))] | ||||
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, | ||||
# ctx.branch() == 'default', but branchtags() is | ||||
# an empty dict. Using dict.get avoids a traceback. | ||||
if repo.branchtags().get(branch) == ctx.node(): | ||||
branches.append({"name": branch}) | ||||
return branches | ||||
def nodeinbranch(repo, ctx): | ||||
branches = [] | ||||
branch = ctx.branch() | ||||
if branch != 'default' and repo.branchtags().get(branch) != ctx.node(): | ||||
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('/') | ||||
Adrian Buehlmann
|
r13971 | return scmutil.canonpath(repo.root, '', path) | ||
Dirkjan Ochtman
|
r6392 | |||
def changectx(repo, req): | ||||
Matt Mackall
|
r6750 | changeid = "tip" | ||
Dirkjan Ochtman
|
r6392 | if 'node' in req.form: | ||
changeid = req.form['node'][0] | ||||
elif 'manifest' in req.form: | ||||
changeid = req.form['manifest'][0] | ||||
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 | ||||
def filectx(repo, req): | ||||
path = cleanpath(repo, req.form['file'][0]) | ||||
if 'node' in req.form: | ||||
changeid = req.form['node'][0] | ||||
else: | ||||
changeid = req.form['filenode'][0] | ||||
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 | |||
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') | ||||
Dirkjan Ochtman
|
r9402 | def diffs(repo, tmpl, ctx, files, parity, style): | ||
Dirkjan Ochtman
|
r7310 | |||
def countgen(): | ||||
start = 1 | ||||
while True: | ||||
yield start | ||||
start += 1 | ||||
blockcount = countgen() | ||||
def prettyprintlines(diff): | ||||
blockno = blockcount.next() | ||||
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) | ||||
parents = ctx.parents() | ||||
node1 = parents and parents[0].node() or nullid | ||||
node2 = ctx.node() | ||||
block = [] | ||||
for chunk in patch.diff(repo, node1, node2, m, opts=diffopts): | ||||
if chunk.startswith('diff') and block: | ||||
yield tmpl('diffblock', parity=parity.next(), | ||||
lines=prettyprintlines(''.join(block))) | ||||
block = [] | ||||
Dirkjan Ochtman
|
r9402 | if chunk.startswith('diff') and style != 'raw': | ||
Dirkjan Ochtman
|
r7310 | chunk = ''.join(chunk.splitlines(True)[1:]) | ||
block.append(chunk) | ||||
yield tmpl('diffblock', parity=parity.next(), | ||||
lines=prettyprintlines(''.join(block))) | ||||
Dirkjan Ochtman
|
r7345 | |||
Steven Brown
|
r14570 | def diffstatgen(ctx): | ||
'''Generator function that provides the diffstat data.''' | ||||
Steven Brown
|
r14490 | |||
stats = patch.diffstatdata(util.iterlines(ctx.diff())) | ||||
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 | ||||
for key, value in self.vars.iteritems(): | ||||
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 | ||||