webcommands.py
651 lines
| 21.6 KiB
| text/x-python
|
PythonLexer
Dirkjan Ochtman
|
r5591 | # | ||
# Copyright 21 May 2005 - (c) 2005 Jake Edge <jake@edge2.net> | ||||
# Copyright 2005-2007 Matt Mackall <mpm@selenic.com> | ||||
# | ||||
# This software may be used and distributed according to the terms | ||||
# of the GNU General Public License, incorporated herein by reference. | ||||
Dirkjan Ochtman
|
r6691 | import os, mimetypes, re, cgi | ||
Dirkjan Ochtman
|
r6392 | import webutil | ||
Dirkjan Ochtman
|
r6691 | from mercurial import revlog, archival, templatefilters | ||
Benoit Boissinot
|
r6410 | from mercurial.node import short, hex, nullid | ||
Dirkjan Ochtman
|
r6691 | from mercurial.util import binary, datestr | ||
Joel Rosdahl
|
r6217 | from mercurial.repo import RepoError | ||
Dirkjan Ochtman
|
r6393 | from common import paritygen, staticfile, get_contact, ErrorResponse | ||
Rocco Rutte
|
r7029 | from common import HTTP_OK, HTTP_FORBIDDEN, HTTP_NOT_FOUND | ||
Matt Mackall
|
r6762 | from mercurial import graphmod, util | ||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5963 | # __all__ is populated with the allowed commands. Be sure to add to it if | ||
# you're adding a new command, or the new command won't work. | ||||
__all__ = [ | ||||
'log', 'rawfile', 'file', 'changelog', 'shortlog', 'changeset', 'rev', | ||||
'manifest', 'tags', 'summary', 'filediff', 'diff', 'annotate', 'filelog', | ||||
Dirkjan Ochtman
|
r6691 | 'archive', 'static', 'graph', | ||
Dirkjan Ochtman
|
r5963 | ] | ||
Dirkjan Ochtman
|
r5600 | def log(web, req, tmpl): | ||
Christian Ebert
|
r5915 | if 'file' in req.form and req.form['file'][0]: | ||
Dirkjan Ochtman
|
r5964 | return filelog(web, req, tmpl) | ||
Dirkjan Ochtman
|
r5591 | else: | ||
Dirkjan Ochtman
|
r5964 | return changelog(web, req, tmpl) | ||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5890 | def rawfile(web, req, tmpl): | ||
Dirkjan Ochtman
|
r6392 | path = webutil.cleanpath(web.repo, req.form.get('file', [''])[0]) | ||
Dirkjan Ochtman
|
r5890 | if not path: | ||
Dirkjan Ochtman
|
r6393 | content = manifest(web, req, tmpl) | ||
Dirkjan Ochtman
|
r5993 | req.respond(HTTP_OK, web.ctype) | ||
return content | ||||
Dirkjan Ochtman
|
r5890 | |||
try: | ||||
Dirkjan Ochtman
|
r6392 | fctx = webutil.filectx(web.repo, req) | ||
Dirkjan Ochtman
|
r6368 | except revlog.LookupError, inst: | ||
try: | ||||
Dirkjan Ochtman
|
r6393 | content = manifest(web, req, tmpl) | ||
Dirkjan Ochtman
|
r6368 | req.respond(HTTP_OK, web.ctype) | ||
return content | ||||
except ErrorResponse: | ||||
raise inst | ||||
Dirkjan Ochtman
|
r5890 | |||
path = fctx.path() | ||||
text = fctx.data() | ||||
mt = mimetypes.guess_type(path)[0] | ||||
Rocco Rutte
|
r6981 | if mt is None: | ||
mt = binary(text) and 'application/octet-stream' or 'text/plain' | ||||
Dirkjan Ochtman
|
r5890 | |||
Dirkjan Ochtman
|
r5993 | req.respond(HTTP_OK, mt, path, len(text)) | ||
Dirkjan Ochtman
|
r5964 | return [text] | ||
Dirkjan Ochtman
|
r5890 | |||
Dirkjan Ochtman
|
r6393 | def _filerevision(web, tmpl, fctx): | ||
f = fctx.path() | ||||
text = fctx.data() | ||||
parity = paritygen(web.stripecount) | ||||
if binary(text): | ||||
mt = mimetypes.guess_type(f)[0] or 'application/octet-stream' | ||||
text = '(binary:%s)' % mt | ||||
def lines(): | ||||
for lineno, t in enumerate(text.splitlines(1)): | ||||
yield {"line": t, | ||||
"lineid": "l%d" % (lineno + 1), | ||||
"linenumber": "% 6d" % (lineno + 1), | ||||
"parity": parity.next()} | ||||
return tmpl("filerevision", | ||||
file=f, | ||||
path=webutil.up(f), | ||||
text=lines(), | ||||
rev=fctx.rev(), | ||||
node=hex(fctx.node()), | ||||
author=fctx.user(), | ||||
date=fctx.date(), | ||||
desc=fctx.description(), | ||||
branch=webutil.nodebranchnodefault(fctx), | ||||
parent=webutil.siblings(fctx.parents()), | ||||
child=webutil.siblings(fctx.children()), | ||||
Matt Mackall
|
r6434 | rename=webutil.renamelink(fctx), | ||
Dirkjan Ochtman
|
r6393 | permissions=fctx.manifest().flags(f)) | ||
Dirkjan Ochtman
|
r5600 | def file(web, req, tmpl): | ||
Dirkjan Ochtman
|
r6392 | path = webutil.cleanpath(web.repo, req.form.get('file', [''])[0]) | ||
Benoit Boissinot
|
r6853 | if not path: | ||
Patrick Mezard
|
r6857 | return manifest(web, req, tmpl) | ||
Benoit Boissinot
|
r6853 | try: | ||
Patrick Mezard
|
r6857 | return _filerevision(web, tmpl, webutil.filectx(web.repo, req)) | ||
Benoit Boissinot
|
r6853 | except revlog.LookupError, inst: | ||
Dirkjan Ochtman
|
r5591 | try: | ||
Patrick Mezard
|
r6857 | return manifest(web, req, tmpl) | ||
Benoit Boissinot
|
r6853 | except ErrorResponse: | ||
raise inst | ||||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r6393 | def _search(web, tmpl, query): | ||
def changelist(**map): | ||||
cl = web.repo.changelog | ||||
count = 0 | ||||
qw = query.lower().split() | ||||
def revgen(): | ||||
Matt Mackall
|
r6750 | for i in xrange(len(cl) - 1, 0, -100): | ||
Dirkjan Ochtman
|
r6393 | l = [] | ||
for j in xrange(max(0, i - 100), i + 1): | ||||
Matt Mackall
|
r6747 | ctx = web.repo[j] | ||
Dirkjan Ochtman
|
r6393 | l.append(ctx) | ||
l.reverse() | ||||
for e in l: | ||||
yield e | ||||
for ctx in revgen(): | ||||
miss = 0 | ||||
for q in qw: | ||||
if not (q in ctx.user().lower() or | ||||
q in ctx.description().lower() or | ||||
q in " ".join(ctx.files()).lower()): | ||||
miss = 1 | ||||
break | ||||
if miss: | ||||
continue | ||||
Andrew Beekhof
|
r6659 | count += 1 | ||
Dirkjan Ochtman
|
r6393 | n = ctx.node() | ||
showtags = webutil.showtag(web.repo, tmpl, 'changelogtag', n) | ||||
yield tmpl('searchentry', | ||||
parity=parity.next(), | ||||
author=ctx.user(), | ||||
parent=webutil.siblings(ctx.parents()), | ||||
child=webutil.siblings(ctx.children()), | ||||
changelogtag=showtags, | ||||
desc=ctx.description(), | ||||
date=ctx.date(), | ||||
files=web.listfilediffs(tmpl, ctx.files(), n), | ||||
rev=ctx.rev(), | ||||
node=hex(n), | ||||
tags=webutil.nodetagsdict(web.repo, n), | ||||
inbranch=webutil.nodeinbranch(web.repo, ctx), | ||||
branches=webutil.nodebranchdict(web.repo, ctx)) | ||||
if count >= web.maxchanges: | ||||
break | ||||
cl = web.repo.changelog | ||||
parity = paritygen(web.stripecount) | ||||
return tmpl('search', | ||||
query=query, | ||||
node=hex(cl.tip()), | ||||
entries=changelist, | ||||
archives=web.archivelist("tip")) | ||||
Dirkjan Ochtman
|
r5600 | def changelog(web, req, tmpl, shortlog = False): | ||
Christian Ebert
|
r5915 | if 'node' in req.form: | ||
Dirkjan Ochtman
|
r6392 | ctx = webutil.changectx(web.repo, req) | ||
Dirkjan Ochtman
|
r5591 | else: | ||
Christian Ebert
|
r5915 | if 'rev' in req.form: | ||
Dirkjan Ochtman
|
r5591 | hi = req.form['rev'][0] | ||
else: | ||||
Matt Mackall
|
r6750 | hi = len(web.repo) - 1 | ||
Dirkjan Ochtman
|
r5591 | try: | ||
Matt Mackall
|
r6747 | ctx = web.repo[hi] | ||
Joel Rosdahl
|
r6217 | except RepoError: | ||
Dirkjan Ochtman
|
r6393 | return _search(web, tmpl, hi) # XXX redirect to 404 page? | ||
def changelist(limit=0, **map): | ||||
cl = web.repo.changelog | ||||
l = [] # build a list in forward order for efficiency | ||||
for i in xrange(start, end): | ||||
Matt Mackall
|
r6747 | ctx = web.repo[i] | ||
Dirkjan Ochtman
|
r6393 | n = ctx.node() | ||
showtags = webutil.showtag(web.repo, tmpl, 'changelogtag', n) | ||||
l.insert(0, {"parity": parity.next(), | ||||
"author": ctx.user(), | ||||
"parent": webutil.siblings(ctx.parents(), i - 1), | ||||
"child": webutil.siblings(ctx.children(), i + 1), | ||||
"changelogtag": showtags, | ||||
"desc": ctx.description(), | ||||
"date": ctx.date(), | ||||
"files": web.listfilediffs(tmpl, ctx.files(), n), | ||||
"rev": i, | ||||
"node": hex(n), | ||||
"tags": webutil.nodetagsdict(web.repo, n), | ||||
"inbranch": webutil.nodeinbranch(web.repo, ctx), | ||||
"branches": webutil.nodebranchdict(web.repo, ctx) | ||||
}) | ||||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r6393 | if limit > 0: | ||
l = l[:limit] | ||||
for e in l: | ||||
yield e | ||||
maxchanges = shortlog and web.maxshortchanges or web.maxchanges | ||||
cl = web.repo.changelog | ||||
Matt Mackall
|
r6750 | count = len(cl) | ||
Dirkjan Ochtman
|
r6393 | pos = ctx.rev() | ||
start = max(0, pos - maxchanges + 1) | ||||
end = min(count, start + maxchanges) | ||||
pos = end - 1 | ||||
parity = paritygen(web.stripecount, offset=start-end) | ||||
changenav = webutil.revnavgen(pos, maxchanges, count, web.repo.changectx) | ||||
return tmpl(shortlog and 'shortlog' or 'changelog', | ||||
changenav=changenav, | ||||
Matt Mackall
|
r6434 | node=hex(ctx.node()), | ||
Dirkjan Ochtman
|
r6393 | rev=pos, changesets=count, | ||
entries=lambda **x: changelist(limit=0,**x), | ||||
latestentry=lambda **x: changelist(limit=1,**x), | ||||
archives=web.archivelist("tip")) | ||||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5600 | def shortlog(web, req, tmpl): | ||
Dirkjan Ochtman
|
r5964 | return changelog(web, req, tmpl, shortlog = True) | ||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5600 | def changeset(web, req, tmpl): | ||
Dirkjan Ochtman
|
r6393 | ctx = webutil.changectx(web.repo, req) | ||
n = ctx.node() | ||||
showtags = webutil.showtag(web.repo, tmpl, 'changesettag', n) | ||||
parents = ctx.parents() | ||||
p1 = parents[0].node() | ||||
files = [] | ||||
parity = paritygen(web.stripecount) | ||||
for f in ctx.files(): | ||||
Dirkjan Ochtman
|
r7182 | template = f in ctx and 'filenodelink' or 'filenolink' | ||
files.append(tmpl(template, | ||||
Dirkjan Ochtman
|
r6393 | node=hex(n), file=f, | ||
parity=parity.next())) | ||||
diffs = web.diff(tmpl, p1, n, None) | ||||
return tmpl('changeset', | ||||
diff=diffs, | ||||
rev=ctx.rev(), | ||||
node=hex(n), | ||||
parent=webutil.siblings(parents), | ||||
child=webutil.siblings(ctx.children()), | ||||
changesettag=showtags, | ||||
author=ctx.user(), | ||||
desc=ctx.description(), | ||||
date=ctx.date(), | ||||
files=files, | ||||
archives=web.archivelist(hex(n)), | ||||
tags=webutil.nodetagsdict(web.repo, n), | ||||
branch=webutil.nodebranchnodefault(ctx), | ||||
inbranch=webutil.nodeinbranch(web.repo, ctx), | ||||
branches=webutil.nodebranchdict(web.repo, ctx)) | ||||
Dirkjan Ochtman
|
r5591 | |||
rev = changeset | ||||
Dirkjan Ochtman
|
r5600 | def manifest(web, req, tmpl): | ||
Dirkjan Ochtman
|
r6393 | ctx = webutil.changectx(web.repo, req) | ||
path = webutil.cleanpath(web.repo, req.form.get('file', [''])[0]) | ||||
mf = ctx.manifest() | ||||
node = ctx.node() | ||||
files = {} | ||||
Ry4an Brase
|
r7305 | dirs = {} | ||
Dirkjan Ochtman
|
r6393 | parity = paritygen(web.stripecount) | ||
if path and path[-1] != "/": | ||||
path += "/" | ||||
l = len(path) | ||||
abspath = "/" + path | ||||
for f, n in mf.items(): | ||||
if f[:l] != path: | ||||
continue | ||||
remain = f[l:] | ||||
Ry4an Brase
|
r7305 | elements = remain.split('/') | ||
if len(elements) == 1: | ||||
files[remain] = f | ||||
else: | ||||
h = dirs # need to retain ref to dirs (root) | ||||
for elem in elements[0:-1]: | ||||
if elem not in h: | ||||
h[elem] = {} | ||||
h = h[elem] | ||||
if len(h) > 1: | ||||
break | ||||
h[None] = None # denotes files present | ||||
Dirkjan Ochtman
|
r6393 | |||
Ry4an Brase
|
r7305 | if not files and not dirs: | ||
Dirkjan Ochtman
|
r6393 | raise ErrorResponse(HTTP_NOT_FOUND, 'path not found: ' + path) | ||
def filelist(**map): | ||||
Matt Mackall
|
r6762 | for f in util.sort(files): | ||
Ry4an Brase
|
r7305 | full = files[f] | ||
Dirkjan Ochtman
|
r6393 | |||
fctx = ctx.filectx(full) | ||||
yield {"file": full, | ||||
"parity": parity.next(), | ||||
"basename": f, | ||||
Matt Mackall
|
r6747 | "date": fctx.date(), | ||
Dirkjan Ochtman
|
r6393 | "size": fctx.size(), | ||
"permissions": mf.flags(full)} | ||||
def dirlist(**map): | ||||
Ry4an Brase
|
r7305 | for d in util.sort(dirs): | ||
Dirkjan Ochtman
|
r6393 | |||
Ry4an Brase
|
r7305 | emptydirs = [] | ||
h = dirs[d] | ||||
while isinstance(h, dict) and len(h) == 1: | ||||
k,v = h.items()[0] | ||||
if v: | ||||
emptydirs.append(k) | ||||
h = v | ||||
path = "%s%s" % (abspath, d) | ||||
Dirkjan Ochtman
|
r6393 | yield {"parity": parity.next(), | ||
Ry4an Brase
|
r7305 | "path": path, | ||
"emptydirs": "/".join(emptydirs), | ||||
"basename": d} | ||||
Dirkjan Ochtman
|
r6393 | |||
return tmpl("manifest", | ||||
rev=ctx.rev(), | ||||
node=hex(node), | ||||
path=abspath, | ||||
up=webutil.up(abspath), | ||||
upparity=parity.next(), | ||||
fentries=filelist, | ||||
dentries=dirlist, | ||||
archives=web.archivelist(hex(node)), | ||||
tags=webutil.nodetagsdict(web.repo, node), | ||||
inbranch=webutil.nodeinbranch(web.repo, ctx), | ||||
branches=webutil.nodebranchdict(web.repo, ctx)) | ||||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5600 | def tags(web, req, tmpl): | ||
Dirkjan Ochtman
|
r6393 | i = web.repo.tagslist() | ||
i.reverse() | ||||
parity = paritygen(web.stripecount) | ||||
def entries(notip=False,limit=0, **map): | ||||
count = 0 | ||||
for k, n in i: | ||||
if notip and k == "tip": | ||||
continue | ||||
if limit > 0 and count >= limit: | ||||
continue | ||||
count = count + 1 | ||||
yield {"parity": parity.next(), | ||||
"tag": k, | ||||
Matt Mackall
|
r6747 | "date": web.repo[n].date(), | ||
Dirkjan Ochtman
|
r6393 | "node": hex(n)} | ||
return tmpl("tags", | ||||
node=hex(web.repo.changelog.tip()), | ||||
entries=lambda **x: entries(False,0, **x), | ||||
entriesnotip=lambda **x: entries(True,0, **x), | ||||
latestentry=lambda **x: entries(True,1, **x)) | ||||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5600 | def summary(web, req, tmpl): | ||
Dirkjan Ochtman
|
r6393 | i = web.repo.tagslist() | ||
i.reverse() | ||||
def tagentries(**map): | ||||
parity = paritygen(web.stripecount) | ||||
count = 0 | ||||
for k, n in i: | ||||
if k == "tip": # skip tip | ||||
continue | ||||
Andrew Beekhof
|
r6659 | count += 1 | ||
Dirkjan Ochtman
|
r6393 | if count > 10: # limit to 10 tags | ||
break | ||||
yield tmpl("tagentry", | ||||
parity=parity.next(), | ||||
tag=k, | ||||
node=hex(n), | ||||
Matt Mackall
|
r6747 | date=web.repo[n].date()) | ||
Dirkjan Ochtman
|
r6393 | |||
def branches(**map): | ||||
parity = paritygen(web.stripecount) | ||||
b = web.repo.branchtags() | ||||
l = [(-web.repo.changelog.rev(n), n, t) for t, n in b.items()] | ||||
Matt Mackall
|
r6762 | for r,n,t in util.sort(l): | ||
Dirkjan Ochtman
|
r6393 | yield {'parity': parity.next(), | ||
'branch': t, | ||||
'node': hex(n), | ||||
Matt Mackall
|
r6747 | 'date': web.repo[n].date()} | ||
Dirkjan Ochtman
|
r6393 | |||
def changelist(**map): | ||||
parity = paritygen(web.stripecount, offset=start-end) | ||||
l = [] # build a list in forward order for efficiency | ||||
for i in xrange(start, end): | ||||
Matt Mackall
|
r6747 | ctx = web.repo[i] | ||
Dirkjan Ochtman
|
r6393 | n = ctx.node() | ||
hn = hex(n) | ||||
l.insert(0, tmpl( | ||||
'shortlogentry', | ||||
parity=parity.next(), | ||||
author=ctx.user(), | ||||
desc=ctx.description(), | ||||
date=ctx.date(), | ||||
rev=i, | ||||
node=hn, | ||||
tags=webutil.nodetagsdict(web.repo, n), | ||||
inbranch=webutil.nodeinbranch(web.repo, ctx), | ||||
branches=webutil.nodebranchdict(web.repo, ctx))) | ||||
yield l | ||||
cl = web.repo.changelog | ||||
Matt Mackall
|
r6750 | count = len(cl) | ||
Dirkjan Ochtman
|
r6393 | start = max(0, count - web.maxchanges) | ||
end = min(count, start + web.maxchanges) | ||||
return tmpl("summary", | ||||
desc=web.config("web", "description", "unknown"), | ||||
owner=get_contact(web.config) or "unknown", | ||||
lastchange=cl.read(cl.tip())[2], | ||||
tags=tagentries, | ||||
branches=branches, | ||||
shortlog=changelist, | ||||
node=hex(cl.tip()), | ||||
archives=web.archivelist("tip")) | ||||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5600 | def filediff(web, req, tmpl): | ||
Dirkjan Ochtman
|
r7183 | fctx, ctx = None, None | ||
try: | ||||
fctx = webutil.filectx(web.repo, req) | ||||
Benoit Boissinot
|
r7280 | except LookupError: | ||
Dirkjan Ochtman
|
r7183 | ctx = webutil.changectx(web.repo, req) | ||
path = webutil.cleanpath(web.repo, req.form['file'][0]) | ||||
if path not in ctx.files(): | ||||
raise | ||||
if fctx is not None: | ||||
n = fctx.node() | ||||
path = fctx.path() | ||||
parents = fctx.parents() | ||||
p1 = parents and parents[0].node() or nullid | ||||
else: | ||||
n = ctx.node() | ||||
# path already defined in except clause | ||||
parents = ctx.parents() | ||||
p1 = parents and parents[0].node() or nullid | ||||
Dirkjan Ochtman
|
r6393 | |||
diffs = web.diff(tmpl, p1, n, [path]) | ||||
Dirkjan Ochtman
|
r7183 | rename = fctx and webutil.renamelink(fctx) or [] | ||
ctx = fctx and fctx or ctx | ||||
Dirkjan Ochtman
|
r6393 | return tmpl("filediff", | ||
file=path, | ||||
node=hex(n), | ||||
Dirkjan Ochtman
|
r7183 | rev=ctx.rev(), | ||
date=ctx.date(), | ||||
desc=ctx.description(), | ||||
author=ctx.user(), | ||||
rename=rename, | ||||
branch=webutil.nodebranchnodefault(ctx), | ||||
Dirkjan Ochtman
|
r6393 | parent=webutil.siblings(parents), | ||
Dirkjan Ochtman
|
r7183 | child=webutil.siblings(ctx.children()), | ||
Dirkjan Ochtman
|
r6393 | diff=diffs) | ||
Dirkjan Ochtman
|
r5591 | |||
diff = filediff | ||||
Dirkjan Ochtman
|
r5600 | def annotate(web, req, tmpl): | ||
Dirkjan Ochtman
|
r6393 | fctx = webutil.filectx(web.repo, req) | ||
f = fctx.path() | ||||
parity = paritygen(web.stripecount) | ||||
def annotate(**map): | ||||
last = None | ||||
if binary(fctx.data()): | ||||
mt = (mimetypes.guess_type(fctx.path())[0] | ||||
or 'application/octet-stream') | ||||
lines = enumerate([((fctx.filectx(fctx.filerev()), 1), | ||||
'(binary:%s)' % mt)]) | ||||
else: | ||||
lines = enumerate(fctx.annotate(follow=True, linenumber=True)) | ||||
for lineno, ((f, targetline), l) in lines: | ||||
fnode = f.filenode() | ||||
if last != fnode: | ||||
last = fnode | ||||
yield {"parity": parity.next(), | ||||
"node": hex(f.node()), | ||||
"rev": f.rev(), | ||||
Patrick Mezard
|
r6564 | "author": f.user(), | ||
Dirkjan Ochtman
|
r6657 | "desc": f.description(), | ||
Dirkjan Ochtman
|
r6393 | "file": f.path(), | ||
"targetline": targetline, | ||||
"line": l, | ||||
"lineid": "l%d" % (lineno + 1), | ||||
"linenumber": "% 6d" % (lineno + 1)} | ||||
return tmpl("fileannotate", | ||||
file=f, | ||||
annotate=annotate, | ||||
path=webutil.up(f), | ||||
rev=fctx.rev(), | ||||
node=hex(fctx.node()), | ||||
author=fctx.user(), | ||||
date=fctx.date(), | ||||
desc=fctx.description(), | ||||
Matt Mackall
|
r6434 | rename=webutil.renamelink(fctx), | ||
Dirkjan Ochtman
|
r6393 | branch=webutil.nodebranchnodefault(fctx), | ||
parent=webutil.siblings(fctx.parents()), | ||||
child=webutil.siblings(fctx.children()), | ||||
permissions=fctx.manifest().flags(f)) | ||||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5600 | def filelog(web, req, tmpl): | ||
Dirkjan Ochtman
|
r7300 | |||
try: | ||||
fctx = webutil.filectx(web.repo, req) | ||||
f = fctx.path() | ||||
fl = fctx.filelog() | ||||
except revlog.LookupError: | ||||
f = webutil.cleanpath(web.repo, req.form['file'][0]) | ||||
fl = web.repo.file(f) | ||||
numrevs = len(fl) | ||||
if not numrevs: # file doesn't exist at all | ||||
raise | ||||
rev = webutil.changectx(web.repo, req).rev() | ||||
first = fl.linkrev(fl.node(0)) | ||||
if rev < first: # current rev is from before file existed | ||||
raise | ||||
frev = numrevs - 1 | ||||
while fl.linkrev(fl.node(frev)) > rev: | ||||
frev -= 1 | ||||
fctx = web.repo.filectx(f, fl.linkrev(fl.node(frev))) | ||||
count = fctx.filerev() + 1 | ||||
Dirkjan Ochtman
|
r6393 | pagelen = web.maxshortchanges | ||
Dirkjan Ochtman
|
r7300 | start = max(0, fctx.filerev() - pagelen + 1) # first rev on this page | ||
end = min(count, start + pagelen) # last rev on this page | ||||
Dirkjan Ochtman
|
r6393 | parity = paritygen(web.stripecount, offset=start-end) | ||
def entries(limit=0, **map): | ||||
l = [] | ||||
for i in xrange(start, end): | ||||
ctx = fctx.filectx(i) | ||||
l.insert(0, {"parity": parity.next(), | ||||
"filerev": i, | ||||
"file": f, | ||||
"node": hex(ctx.node()), | ||||
"author": ctx.user(), | ||||
"date": ctx.date(), | ||||
Matt Mackall
|
r6434 | "rename": webutil.renamelink(fctx), | ||
Dirkjan Ochtman
|
r6393 | "parent": webutil.siblings(fctx.parents()), | ||
"child": webutil.siblings(fctx.children()), | ||||
"desc": ctx.description()}) | ||||
if limit > 0: | ||||
l = l[:limit] | ||||
for e in l: | ||||
yield e | ||||
nodefunc = lambda x: fctx.filectx(fileid=x) | ||||
Dirkjan Ochtman
|
r7300 | nav = webutil.revnavgen(end - 1, pagelen, count, nodefunc) | ||
Dirkjan Ochtman
|
r6393 | return tmpl("filelog", file=f, node=hex(fctx.node()), nav=nav, | ||
entries=lambda **x: entries(limit=0, **x), | ||||
latestentry=lambda **x: entries(limit=1, **x)) | ||||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5600 | def archive(web, req, tmpl): | ||
Ali Saidi
|
r6669 | type_ = req.form.get('type', [None])[0] | ||
Dirkjan Ochtman
|
r5591 | allowed = web.configlist("web", "allow_archive") | ||
Dirkjan Ochtman
|
r6393 | key = req.form['node'][0] | ||
Rocco Rutte
|
r7029 | if type_ not in web.archives: | ||
Dirkjan Ochtman
|
r6393 | msg = 'Unsupported archive type: %s' % type_ | ||
raise ErrorResponse(HTTP_NOT_FOUND, msg) | ||||
Rocco Rutte
|
r7029 | if not ((type_ in allowed or | ||
web.configbool("web", "allow" + type_, False))): | ||||
msg = 'Archive type not allowed: %s' % type_ | ||||
raise ErrorResponse(HTTP_FORBIDDEN, msg) | ||||
Dirkjan Ochtman
|
r6393 | reponame = re.sub(r"\W+", "-", os.path.basename(web.reponame)) | ||
cnode = web.repo.lookup(key) | ||||
arch_version = key | ||||
if cnode == key or key == 'tip': | ||||
arch_version = short(cnode) | ||||
name = "%s-%s" % (reponame, arch_version) | ||||
mimetype, artype, extension, encoding = web.archive_specs[type_] | ||||
headers = [ | ||||
('Content-Type', mimetype), | ||||
('Content-Disposition', 'attachment; filename=%s%s' % (name, extension)) | ||||
] | ||||
if encoding: | ||||
headers.append(('Content-Encoding', encoding)) | ||||
req.header(headers) | ||||
req.respond(HTTP_OK) | ||||
archival.archive(web.repo, req, cnode, artype, prefix=name) | ||||
return [] | ||||
Dirkjan Ochtman
|
r5591 | |||
Dirkjan Ochtman
|
r5600 | def static(web, req, tmpl): | ||
Dirkjan Ochtman
|
r5591 | fname = req.form['file'][0] | ||
# a repo owner may set web.static in .hg/hgrc to get any file | ||||
# readable by the user running the CGI script | ||||
Brendan Cully
|
r7107 | static = web.config("web", "static", None, untrusted=False) | ||
if not static: | ||||
tp = web.templatepath | ||||
if isinstance(tp, str): | ||||
tp = [tp] | ||||
Brendan Cully
|
r7288 | static = [os.path.join(p, 'static') for p in tp] | ||
Dirkjan Ochtman
|
r5964 | return [staticfile(static, fname, req)] | ||
Dirkjan Ochtman
|
r6691 | |||
def graph(web, req, tmpl): | ||||
rev = webutil.changectx(web.repo, req).rev() | ||||
bg_height = 39 | ||||
Matt Mackall
|
r6750 | max_rev = len(web.repo) - 1 | ||
Benoit Allard
|
r6704 | revcount = min(max_rev, int(req.form.get('revcount', [25])[0])) | ||
Dirkjan Ochtman
|
r6691 | revnode = web.repo.changelog.node(rev) | ||
revnode_hex = hex(revnode) | ||||
uprev = min(max_rev, rev + revcount) | ||||
downrev = max(0, rev - revcount) | ||||
lessrev = max(0, rev - revcount / 2) | ||||
maxchanges = web.maxshortchanges or web.maxchanges | ||||
Matt Mackall
|
r6750 | count = len(web.repo) | ||
Dirkjan Ochtman
|
r6691 | changenav = webutil.revnavgen(rev, maxchanges, count, web.repo.changectx) | ||
Dirkjan Ochtman
|
r7030 | tree = list(graphmod.graph(web.repo, rev, downrev)) | ||
Dirkjan Ochtman
|
r6691 | canvasheight = (len(tree) + 1) * bg_height - 27; | ||
data = [] | ||||
for i, (ctx, vtx, edges) in enumerate(tree): | ||||
node = short(ctx.node()) | ||||
age = templatefilters.age(ctx.date()) | ||||
desc = templatefilters.firstline(ctx.description()) | ||||
desc = cgi.escape(desc) | ||||
user = cgi.escape(templatefilters.person(ctx.user())) | ||||
Dirkjan Ochtman
|
r6720 | branch = ctx.branch() | ||
branch = branch, web.repo.branchtags().get(branch) == ctx.node() | ||||
data.append((node, vtx, edges, desc, user, age, branch, ctx.tags())) | ||||
Dirkjan Ochtman
|
r6691 | |||
return tmpl('graph', rev=rev, revcount=revcount, uprev=uprev, | ||||
lessrev=lessrev, revcountmore=revcount and 2 * revcount or 1, | ||||
revcountless=revcount / 2, downrev=downrev, | ||||
canvasheight=canvasheight, bg_height=bg_height, | ||||
jsdata=data, node=revnode_hex, changenav=changenav) | ||||