webcommands.py
606 lines
| 20.0 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 | ||
from common import HTTP_OK, 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] | ||||
Dirkjan Ochtman
|
r6392 | if mt is None or binary(text): | ||
Dirkjan Ochtman
|
r5890 | mt = mt or 'application/octet-stream' | ||
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() | ||||
fl = fctx.filelog() | ||||
n = fctx.filenode() | ||||
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]) | ||
Dirkjan Ochtman
|
r5591 | if path: | ||
try: | ||||
Dirkjan Ochtman
|
r6393 | return _filerevision(web, tmpl, webutil.filectx(web.repo, req)) | ||
Dirkjan Ochtman
|
r6368 | except revlog.LookupError, inst: | ||
Dirkjan Ochtman
|
r5591 | pass | ||
Dirkjan Ochtman
|
r6368 | try: | ||
Dirkjan Ochtman
|
r6393 | return manifest(web, req, tmpl) | ||
Dirkjan Ochtman
|
r6368 | 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(): | ||||
files.append(tmpl("filenodelink", | ||||
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 = {} | ||||
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:] | ||||
if "/" in remain: | ||||
short = remain[:remain.index("/") + 1] # bleah | ||||
files[short] = (f, None) | ||||
else: | ||||
short = os.path.basename(remain) | ||||
files[short] = (f, n) | ||||
if not files: | ||||
raise ErrorResponse(HTTP_NOT_FOUND, 'path not found: ' + path) | ||||
def filelist(**map): | ||||
Matt Mackall
|
r6762 | for f in util.sort(files): | ||
Dirkjan Ochtman
|
r6393 | full, fnode = files[f] | ||
if not fnode: | ||||
continue | ||||
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): | ||||
Matt Mackall
|
r6762 | for f in util.sort(files): | ||
Dirkjan Ochtman
|
r6393 | full, fnode = files[f] | ||
if fnode: | ||||
continue | ||||
yield {"parity": parity.next(), | ||||
"path": "%s%s" % (abspath, f), | ||||
"basename": f[:-1]} | ||||
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
|
r6393 | fctx = webutil.filectx(web.repo, req) | ||
n = fctx.node() | ||||
path = fctx.path() | ||||
parents = fctx.parents() | ||||
p1 = parents and parents[0].node() or nullid | ||||
diffs = web.diff(tmpl, p1, n, [path]) | ||||
return tmpl("filediff", | ||||
file=path, | ||||
node=hex(n), | ||||
rev=fctx.rev(), | ||||
Matt Mackall
|
r6434 | date=fctx.date(), | ||
desc=fctx.description(), | ||||
author=fctx.user(), | ||||
Matt Mackall
|
r6437 | rename=webutil.renamelink(fctx), | ||
Dirkjan Ochtman
|
r6393 | branch=webutil.nodebranchnodefault(fctx), | ||
parent=webutil.siblings(parents), | ||||
child=webutil.siblings(fctx.children()), | ||||
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() | ||||
n = fctx.filenode() | ||||
fl = fctx.filelog() | ||||
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
|
r6393 | fctx = webutil.filectx(web.repo, req) | ||
f = fctx.path() | ||||
fl = fctx.filelog() | ||||
Matt Mackall
|
r6750 | count = len(fl) | ||
Dirkjan Ochtman
|
r6393 | pagelen = web.maxshortchanges | ||
pos = fctx.filerev() | ||||
start = max(0, pos - pagelen + 1) | ||||
end = min(count, start + pagelen) | ||||
pos = end - 1 | ||||
parity = paritygen(web.stripecount, offset=start-end) | ||||
def entries(limit=0, **map): | ||||
l = [] | ||||
for i in xrange(start, end): | ||||
ctx = fctx.filectx(i) | ||||
n = fl.node(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) | ||||
nav = webutil.revnavgen(pos, pagelen, count, nodefunc) | ||||
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] | ||
if not (type_ in web.archives and (type_ in allowed or | ||||
Dirkjan Ochtman
|
r5591 | web.configbool("web", "allow" + type_, False))): | ||
Dirkjan Ochtman
|
r6393 | msg = 'Unsupported archive type: %s' % type_ | ||
raise ErrorResponse(HTTP_NOT_FOUND, msg) | ||||
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 | ||||
static = web.config("web", "static", | ||||
os.path.join(web.templatepath, "static"), | ||||
untrusted=False) | ||||
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) | ||
tree = list(graphmod.graph(web.repo, rev, rev - revcount)) | ||||
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) | ||||