hgweb_mod.py
1057 lines
| 36.0 KiB
| text/x-python
|
PythonLexer
Eric Hopper
|
r2391 | # hgweb/hgweb_mod.py - Web interface for a repository. | ||
Eric Hopper
|
r2356 | # | ||
# Copyright 21 May 2005 - (c) 2005 Jake Edge <jake@edge2.net> | ||||
Vadim Gelfer
|
r2859 | # Copyright 2005, 2006 Matt Mackall <mpm@selenic.com> | ||
Eric Hopper
|
r2356 | # | ||
# This software may be used and distributed according to the terms | ||||
# of the GNU General Public License, incorporated herein by reference. | ||||
import os | ||||
import os.path | ||||
import mimetypes | ||||
from mercurial.demandload import demandload | ||||
Eric Hopper
|
r2514 | demandload(globals(), "re zlib ConfigParser mimetools cStringIO sys tempfile") | ||
Brendan Cully
|
r3270 | demandload(globals(), 'urllib') | ||
Matt Mackall
|
r2888 | demandload(globals(), "mercurial:mdiff,ui,hg,util,archival,streamclone,patch") | ||
Vadim Gelfer
|
r2612 | demandload(globals(), "mercurial:templater") | ||
Eric Hopper
|
r2356 | demandload(globals(), "mercurial.hgweb.common:get_mtime,staticfile") | ||
from mercurial.node import * | ||||
from mercurial.i18n import gettext as _ | ||||
def _up(p): | ||||
if p[0] != "/": | ||||
p = "/" + p | ||||
if p[-1] == "/": | ||||
p = p[:-1] | ||||
up = os.path.dirname(p) | ||||
if up == "/": | ||||
return "/" | ||||
return up + "/" | ||||
class hgweb(object): | ||||
def __init__(self, repo, name=None): | ||||
if type(repo) == type(""): | ||||
self.repo = hg.repository(ui.ui(), repo) | ||||
else: | ||||
self.repo = repo | ||||
self.mtime = -1 | ||||
self.reponame = name | ||||
self.archives = 'zip', 'gz', 'bz2' | ||||
Frank Kingswood
|
r2666 | self.stripecount = 1 | ||
Vadim Gelfer
|
r2436 | self.templatepath = self.repo.ui.config("web", "templates", | ||
templater.templatepath()) | ||||
Eric Hopper
|
r2356 | |||
def refresh(self): | ||||
mtime = get_mtime(self.repo.root) | ||||
if mtime != self.mtime: | ||||
self.mtime = mtime | ||||
self.repo = hg.repository(self.repo.ui, self.repo.root) | ||||
self.maxchanges = int(self.repo.ui.config("web", "maxchanges", 10)) | ||||
Frank Kingswood
|
r2666 | self.stripecount = int(self.repo.ui.config("web", "stripes", 1)) | ||
Josef "Jeff" Sipek
|
r2684 | self.maxshortchanges = int(self.repo.ui.config("web", "maxshortchanges", 60)) | ||
Eric Hopper
|
r2356 | self.maxfiles = int(self.repo.ui.config("web", "maxfiles", 10)) | ||
self.allowpull = self.repo.ui.configbool("web", "allowpull", True) | ||||
def archivelist(self, nodeid): | ||||
Thomas Arendsen Hein
|
r2500 | allowed = self.repo.ui.configlist("web", "allow_archive") | ||
Brendan Cully
|
r3260 | for i, spec in self.archive_specs.iteritems(): | ||
Thomas Arendsen Hein
|
r2359 | if i in allowed or self.repo.ui.configbool("web", "allow" + i): | ||
Brendan Cully
|
r3260 | yield {"type" : i, "extension" : spec[2], "node" : nodeid} | ||
Eric Hopper
|
r2356 | |||
def listfiles(self, files, mf): | ||||
for f in files[:self.maxfiles]: | ||||
yield self.t("filenodelink", node=hex(mf[f]), file=f) | ||||
if len(files) > self.maxfiles: | ||||
yield self.t("fileellipses") | ||||
def listfilediffs(self, files, changeset): | ||||
for f in files[:self.maxfiles]: | ||||
yield self.t("filedifflink", node=hex(changeset), file=f) | ||||
if len(files) > self.maxfiles: | ||||
yield self.t("fileellipses") | ||||
Brendan Cully
|
r3208 | def siblings(self, siblings=[], hiderev=None, **args): | ||
siblings = [s for s in siblings if s.node() != nullid] | ||||
if len(siblings) == 1 and siblings[0].rev() == hiderev: | ||||
Eric Hopper
|
r2356 | return | ||
for s in siblings: | ||||
Brendan Cully
|
r3208 | yield dict(node=hex(s.node()), rev=s.rev(), **args) | ||
Eric Hopper
|
r2356 | |||
def renamelink(self, fl, node): | ||||
r = fl.renamed(node) | ||||
if r: | ||||
return [dict(file=r[0], node=hex(r[1]))] | ||||
return [] | ||||
def showtag(self, t1, node=nullid, **args): | ||||
for t in self.repo.nodetags(node): | ||||
yield self.t(t1, tag=t, **args) | ||||
def diff(self, node1, node2, files): | ||||
def filterfiles(filters, files): | ||||
l = [x for x in files if x in filters] | ||||
for t in filters: | ||||
if t and t[-1] != os.sep: | ||||
t += os.sep | ||||
l += [x for x in files if x.startswith(t)] | ||||
return l | ||||
parity = [0] | ||||
def diffblock(diff, f, fn): | ||||
yield self.t("diffblock", | ||||
lines=prettyprintlines(diff), | ||||
parity=parity[0], | ||||
file=f, | ||||
filenode=hex(fn or nullid)) | ||||
parity[0] = 1 - parity[0] | ||||
def prettyprintlines(diff): | ||||
for l in diff.splitlines(1): | ||||
if l.startswith('+'): | ||||
yield self.t("difflineplus", line=l) | ||||
elif l.startswith('-'): | ||||
yield self.t("difflineminus", line=l) | ||||
elif l.startswith('@'): | ||||
yield self.t("difflineat", line=l) | ||||
else: | ||||
yield self.t("diffline", line=l) | ||||
r = self.repo | ||||
cl = r.changelog | ||||
mf = r.manifest | ||||
change1 = cl.read(node1) | ||||
change2 = cl.read(node2) | ||||
mmap1 = mf.read(change1[0]) | ||||
mmap2 = mf.read(change2[0]) | ||||
date1 = util.datestr(change1[2]) | ||||
date2 = util.datestr(change2[2]) | ||||
Giorgos Keramidas
|
r2876 | modified, added, removed, deleted, unknown = r.status(node1, node2)[:5] | ||
Eric Hopper
|
r2356 | if files: | ||
modified, added, removed = map(lambda x: filterfiles(files, x), | ||||
(modified, added, removed)) | ||||
Thomas Arendsen Hein
|
r2927 | diffopts = patch.diffopts(self.repo.ui) | ||
Eric Hopper
|
r2356 | for f in modified: | ||
to = r.file(f).read(mmap1[f]) | ||||
tn = r.file(f).read(mmap2[f]) | ||||
yield diffblock(mdiff.unidiff(to, date1, tn, date2, f, | ||||
Vadim Gelfer
|
r2874 | opts=diffopts), f, tn) | ||
Eric Hopper
|
r2356 | for f in added: | ||
to = None | ||||
tn = r.file(f).read(mmap2[f]) | ||||
yield diffblock(mdiff.unidiff(to, date1, tn, date2, f, | ||||
Vadim Gelfer
|
r2874 | opts=diffopts), f, tn) | ||
Eric Hopper
|
r2356 | for f in removed: | ||
to = r.file(f).read(mmap1[f]) | ||||
tn = None | ||||
yield diffblock(mdiff.unidiff(to, date1, tn, date2, f, | ||||
Vadim Gelfer
|
r2874 | opts=diffopts), f, tn) | ||
Eric Hopper
|
r2356 | |||
Brendan Cully
|
r3220 | def changelog(self, ctx, shortlog=False): | ||
pos = ctx.rev() | ||||
Eric Hopper
|
r2356 | def changenav(**map): | ||
def seq(factor, maxchanges=None): | ||||
if maxchanges: | ||||
yield maxchanges | ||||
if maxchanges >= 20 and maxchanges <= 40: | ||||
yield 50 | ||||
else: | ||||
yield 1 * factor | ||||
yield 3 * factor | ||||
for f in seq(factor * 10): | ||||
yield f | ||||
l = [] | ||||
last = 0 | ||||
Josef "Jeff" Sipek
|
r2684 | maxchanges = shortlog and self.maxshortchanges or self.maxchanges | ||
for f in seq(1, maxchanges): | ||||
if f < maxchanges or f <= last: | ||||
Eric Hopper
|
r2356 | continue | ||
if f > count: | ||||
break | ||||
last = f | ||||
r = "%d" % f | ||||
if pos + f < count: | ||||
l.append(("+" + r, pos + f)) | ||||
if pos - f >= 0: | ||||
l.insert(0, ("-" + r, pos - f)) | ||||
yield {"rev": 0, "label": "(0)"} | ||||
for label, rev in l: | ||||
yield {"label": label, "rev": rev} | ||||
yield {"label": "tip", "rev": "tip"} | ||||
def changelist(**map): | ||||
parity = (start - end) & 1 | ||||
cl = self.repo.changelog | ||||
l = [] # build a list in forward order for efficiency | ||||
for i in range(start, end): | ||||
Brendan Cully
|
r3208 | ctx = self.repo.changectx(i) | ||
n = ctx.node() | ||||
Eric Hopper
|
r2356 | |||
l.insert(0, {"parity": parity, | ||||
Brendan Cully
|
r3208 | "author": ctx.user(), | ||
"parent": self.siblings(ctx.parents(), i - 1), | ||||
"child": self.siblings(ctx.children(), i + 1), | ||||
Eric Hopper
|
r2356 | "changelogtag": self.showtag("changelogtag",n), | ||
Brendan Cully
|
r3208 | "desc": ctx.description(), | ||
"date": ctx.date(), | ||||
"files": self.listfilediffs(ctx.files(), n), | ||||
Eric Hopper
|
r2356 | "rev": i, | ||
Brendan Cully
|
r3208 | "node": hex(n)}) | ||
Eric Hopper
|
r2356 | parity = 1 - parity | ||
for e in l: | ||||
yield e | ||||
Josef "Jeff" Sipek
|
r2684 | maxchanges = shortlog and self.maxshortchanges or self.maxchanges | ||
Eric Hopper
|
r2356 | cl = self.repo.changelog | ||
count = cl.count() | ||||
Josef "Jeff" Sipek
|
r2684 | start = max(0, pos - maxchanges + 1) | ||
end = min(count, start + maxchanges) | ||||
Eric Hopper
|
r2356 | pos = end - 1 | ||
Josef "Jeff" Sipek
|
r2684 | yield self.t(shortlog and 'shortlog' or 'changelog', | ||
Eric Hopper
|
r2356 | changenav=changenav, | ||
Brendan Cully
|
r3205 | node=hex(cl.tip()), | ||
Eric Hopper
|
r2356 | rev=pos, changesets=count, entries=changelist, | ||
archives=self.archivelist("tip")) | ||||
def search(self, query): | ||||
def changelist(**map): | ||||
cl = self.repo.changelog | ||||
count = 0 | ||||
qw = query.lower().split() | ||||
def revgen(): | ||||
for i in range(cl.count() - 1, 0, -100): | ||||
l = [] | ||||
for j in range(max(0, i - 100), i): | ||||
Brendan Cully
|
r3208 | ctx = self.repo.changectx(j) | ||
l.append(ctx) | ||||
Eric Hopper
|
r2356 | l.reverse() | ||
for e in l: | ||||
yield e | ||||
Brendan Cully
|
r3208 | for ctx in revgen(): | ||
Eric Hopper
|
r2356 | miss = 0 | ||
for q in qw: | ||||
Brendan Cully
|
r3208 | if not (q in ctx.user().lower() or | ||
q in ctx.description().lower() or | ||||
q in " ".join(ctx.files()[:20]).lower()): | ||||
Eric Hopper
|
r2356 | miss = 1 | ||
break | ||||
if miss: | ||||
continue | ||||
count += 1 | ||||
Brendan Cully
|
r3208 | n = ctx.node() | ||
Eric Hopper
|
r2356 | |||
yield self.t('searchentry', | ||||
Frank Kingswood
|
r2666 | parity=self.stripes(count), | ||
Brendan Cully
|
r3208 | author=ctx.user(), | ||
parent=self.siblings(ctx.parents()), | ||||
child=self.siblings(ctx.children()), | ||||
Eric Hopper
|
r2356 | changelogtag=self.showtag("changelogtag",n), | ||
Brendan Cully
|
r3208 | desc=ctx.description(), | ||
date=ctx.date(), | ||||
files=self.listfilediffs(ctx.files(), n), | ||||
rev=ctx.rev(), | ||||
node=hex(n)) | ||||
Eric Hopper
|
r2356 | |||
if count >= self.maxchanges: | ||||
break | ||||
cl = self.repo.changelog | ||||
yield self.t('search', | ||||
query=query, | ||||
Brendan Cully
|
r3205 | node=hex(cl.tip()), | ||
Eric Hopper
|
r2356 | entries=changelist) | ||
Brendan Cully
|
r3220 | def changeset(self, ctx): | ||
Brendan Cully
|
r3208 | n = ctx.node() | ||
parents = ctx.parents() | ||||
p1 = parents[0].node() | ||||
Eric Hopper
|
r2356 | |||
files = [] | ||||
Brendan Cully
|
r3179 | parity = 0 | ||
Brendan Cully
|
r3208 | for f in ctx.files(): | ||
Eric Hopper
|
r2356 | files.append(self.t("filenodelink", | ||
Brendan Cully
|
r3206 | node=hex(n), file=f, | ||
Brendan Cully
|
r3179 | parity=parity)) | ||
parity = 1 - parity | ||||
Eric Hopper
|
r2356 | |||
def diff(**map): | ||||
yield self.diff(p1, n, None) | ||||
yield self.t('changeset', | ||||
diff=diff, | ||||
Brendan Cully
|
r3208 | rev=ctx.rev(), | ||
node=hex(n), | ||||
parent=self.siblings(parents), | ||||
child=self.siblings(ctx.children()), | ||||
Eric Hopper
|
r2356 | changesettag=self.showtag("changesettag",n), | ||
Brendan Cully
|
r3208 | author=ctx.user(), | ||
desc=ctx.description(), | ||||
date=ctx.date(), | ||||
Eric Hopper
|
r2356 | files=files, | ||
Brendan Cully
|
r3220 | archives=self.archivelist(hex(n))) | ||
Eric Hopper
|
r2356 | |||
Brendan Cully
|
r3206 | def filelog(self, fctx): | ||
f = fctx.path() | ||||
fl = fctx.filelog() | ||||
Eric Hopper
|
r2356 | count = fl.count() | ||
def entries(**map): | ||||
l = [] | ||||
parity = (count - 1) & 1 | ||||
for i in range(count): | ||||
Brendan Cully
|
r3208 | ctx = fctx.filectx(i) | ||
Eric Hopper
|
r2356 | n = fl.node(i) | ||
l.insert(0, {"parity": parity, | ||||
"filerev": i, | ||||
"file": f, | ||||
Brendan Cully
|
r3206 | "node": hex(ctx.node()), | ||
"author": ctx.user(), | ||||
"date": ctx.date(), | ||||
Eric Hopper
|
r2356 | "rename": self.renamelink(fl, n), | ||
Brendan Cully
|
r3208 | "parent": self.siblings(fctx.parents(), file=f), | ||
"child": self.siblings(fctx.children(), file=f), | ||||
Brendan Cully
|
r3206 | "desc": ctx.description()}) | ||
Eric Hopper
|
r2356 | parity = 1 - parity | ||
for e in l: | ||||
yield e | ||||
Brendan Cully
|
r3206 | yield self.t("filelog", file=f, node=hex(fctx.node()), entries=entries) | ||
Eric Hopper
|
r2356 | |||
Brendan Cully
|
r3206 | def filerevision(self, fctx): | ||
f = fctx.path() | ||||
text = fctx.data() | ||||
fl = fctx.filelog() | ||||
n = fctx.filenode() | ||||
Eric Hopper
|
r2356 | |||
mt = mimetypes.guess_type(f)[0] | ||||
rawtext = text | ||||
if util.binary(text): | ||||
mt = mt or 'application/octet-stream' | ||||
text = "(binary:%s)" % mt | ||||
mt = mt or 'text/plain' | ||||
def lines(): | ||||
for l, t in enumerate(text.splitlines(1)): | ||||
yield {"line": t, | ||||
"linenumber": "% 6d" % (l + 1), | ||||
Frank Kingswood
|
r2666 | "parity": self.stripes(l)} | ||
Eric Hopper
|
r2356 | |||
yield self.t("filerevision", | ||||
file=f, | ||||
path=_up(f), | ||||
text=lines(), | ||||
raw=rawtext, | ||||
mimetype=mt, | ||||
Brendan Cully
|
r3206 | rev=fctx.rev(), | ||
node=hex(fctx.node()), | ||||
author=fctx.user(), | ||||
date=fctx.date(), | ||||
Brendan Cully
|
r3208 | parent=self.siblings(fctx.parents(), file=f), | ||
child=self.siblings(fctx.children(), file=f), | ||||
Eric Hopper
|
r2356 | rename=self.renamelink(fl, n), | ||
Brendan Cully
|
r3206 | permissions=fctx.manifest().execf(f)) | ||
Eric Hopper
|
r2356 | |||
Brendan Cully
|
r3206 | def fileannotate(self, fctx): | ||
f = fctx.path() | ||||
Brendan Cully
|
r3173 | n = fctx.filenode() | ||
fl = fctx.filelog() | ||||
Eric Hopper
|
r2356 | |||
def annotate(**map): | ||||
Frank Kingswood
|
r2666 | parity = 0 | ||
Eric Hopper
|
r2356 | last = None | ||
Brendan Cully
|
r3175 | for f, l in fctx.annotate(follow=True): | ||
fnode = f.filenode() | ||||
Brendan Cully
|
r3173 | name = self.repo.ui.shortuser(f.user()) | ||
Eric Hopper
|
r2356 | |||
Brendan Cully
|
r3175 | if last != fnode: | ||
Eric Hopper
|
r2356 | parity = 1 - parity | ||
Brendan Cully
|
r3175 | last = fnode | ||
Eric Hopper
|
r2356 | |||
yield {"parity": parity, | ||||
Brendan Cully
|
r3178 | "node": hex(f.node()), | ||
Brendan Cully
|
r3173 | "rev": f.rev(), | ||
Eric Hopper
|
r2356 | "author": name, | ||
Brendan Cully
|
r3173 | "file": f.path(), | ||
Eric Hopper
|
r2356 | "line": l} | ||
yield self.t("fileannotate", | ||||
file=f, | ||||
annotate=annotate, | ||||
path=_up(f), | ||||
Brendan Cully
|
r3173 | rev=fctx.rev(), | ||
Brendan Cully
|
r3177 | node=hex(fctx.node()), | ||
Brendan Cully
|
r3173 | author=fctx.user(), | ||
date=fctx.date(), | ||||
Eric Hopper
|
r2356 | rename=self.renamelink(fl, n), | ||
Brendan Cully
|
r3208 | parent=self.siblings(fctx.parents(), file=f), | ||
child=self.siblings(fctx.children(), file=f), | ||||
Brendan Cully
|
r3173 | permissions=fctx.manifest().execf(f)) | ||
Eric Hopper
|
r2356 | |||
Brendan Cully
|
r3205 | def manifest(self, ctx, path): | ||
mf = ctx.manifest() | ||||
node = ctx.node() | ||||
Eric Hopper
|
r2356 | |||
files = {} | ||||
p = path[1:] | ||||
if p and p[-1] != "/": | ||||
p += "/" | ||||
l = len(p) | ||||
for f,n in mf.items(): | ||||
if f[:l] != p: | ||||
continue | ||||
remain = f[l:] | ||||
if "/" in remain: | ||||
Benoit Boissinot
|
r2579 | short = remain[:remain.index("/") + 1] # bleah | ||
Eric Hopper
|
r2356 | files[short] = (f, None) | ||
else: | ||||
short = os.path.basename(remain) | ||||
files[short] = (f, n) | ||||
def filelist(**map): | ||||
parity = 0 | ||||
fl = files.keys() | ||||
fl.sort() | ||||
for f in fl: | ||||
full, fnode = files[f] | ||||
if not fnode: | ||||
continue | ||||
yield {"file": full, | ||||
"filenode": hex(fnode), | ||||
Frank Kingswood
|
r2666 | "parity": self.stripes(parity), | ||
Eric Hopper
|
r2356 | "basename": f, | ||
Alexis S. L. Carvalho
|
r2857 | "permissions": mf.execf(full)} | ||
Frank Kingswood
|
r2666 | parity += 1 | ||
Eric Hopper
|
r2356 | |||
def dirlist(**map): | ||||
parity = 0 | ||||
fl = files.keys() | ||||
fl.sort() | ||||
for f in fl: | ||||
full, fnode = files[f] | ||||
if fnode: | ||||
continue | ||||
Frank Kingswood
|
r2666 | yield {"parity": self.stripes(parity), | ||
Eric Hopper
|
r2356 | "path": os.path.join(path, f), | ||
"basename": f[:-1]} | ||||
Frank Kingswood
|
r2666 | parity += 1 | ||
Eric Hopper
|
r2356 | |||
yield self.t("manifest", | ||||
Brendan Cully
|
r3205 | rev=ctx.rev(), | ||
Eric Hopper
|
r2356 | node=hex(node), | ||
path=path, | ||||
up=_up(path), | ||||
fentries=filelist, | ||||
dentries=dirlist, | ||||
archives=self.archivelist(hex(node))) | ||||
def tags(self): | ||||
cl = self.repo.changelog | ||||
i = self.repo.tagslist() | ||||
i.reverse() | ||||
def entries(notip=False, **map): | ||||
parity = 0 | ||||
for k,n in i: | ||||
if notip and k == "tip": continue | ||||
Frank Kingswood
|
r2666 | yield {"parity": self.stripes(parity), | ||
Eric Hopper
|
r2356 | "tag": k, | ||
"date": cl.read(n)[2], | ||||
"node": hex(n)} | ||||
Frank Kingswood
|
r2666 | parity += 1 | ||
Eric Hopper
|
r2356 | |||
yield self.t("tags", | ||||
Brendan Cully
|
r3205 | node=hex(self.repo.changelog.tip()), | ||
Eric Hopper
|
r2356 | entries=lambda **x: entries(False, **x), | ||
entriesnotip=lambda **x: entries(True, **x)) | ||||
def summary(self): | ||||
cl = self.repo.changelog | ||||
i = self.repo.tagslist() | ||||
i.reverse() | ||||
def tagentries(**map): | ||||
parity = 0 | ||||
count = 0 | ||||
for k,n in i: | ||||
if k == "tip": # skip tip | ||||
continue; | ||||
count += 1 | ||||
if count > 10: # limit to 10 tags | ||||
break; | ||||
c = cl.read(n) | ||||
t = c[2] | ||||
yield self.t("tagentry", | ||||
Frank Kingswood
|
r2666 | parity = self.stripes(parity), | ||
Eric Hopper
|
r2356 | tag = k, | ||
node = hex(n), | ||||
Brendan Cully
|
r3205 | date = t) | ||
Frank Kingswood
|
r2666 | parity += 1 | ||
Eric Hopper
|
r2356 | |||
def changelist(**map): | ||||
parity = 0 | ||||
cl = self.repo.changelog | ||||
l = [] # build a list in forward order for efficiency | ||||
for i in range(start, end): | ||||
n = cl.node(i) | ||||
changes = cl.read(n) | ||||
hn = hex(n) | ||||
t = changes[2] | ||||
l.insert(0, self.t( | ||||
'shortlogentry', | ||||
parity = parity, | ||||
author = changes[1], | ||||
desc = changes[4], | ||||
date = t, | ||||
rev = i, | ||||
node = hn)) | ||||
parity = 1 - parity | ||||
yield l | ||||
count = cl.count() | ||||
start = max(0, count - self.maxchanges) | ||||
end = min(count, start + self.maxchanges) | ||||
yield self.t("summary", | ||||
desc = self.repo.ui.config("web", "description", "unknown"), | ||||
owner = (self.repo.ui.config("ui", "username") or # preferred | ||||
self.repo.ui.config("web", "contact") or # deprecated | ||||
self.repo.ui.config("web", "author", "unknown")), # also | ||||
lastchange = (0, 0), # FIXME | ||||
tags = tagentries, | ||||
Josef "Jeff" Sipek
|
r2683 | shortlog = changelist, | ||
Brendan Cully
|
r3205 | node = hex(self.repo.changelog.tip()), | ||
Josef "Jeff" Sipek
|
r2683 | archives=self.archivelist("tip")) | ||
Eric Hopper
|
r2356 | |||
Brendan Cully
|
r3220 | def filediff(self, fctx): | ||
n = fctx.node() | ||||
path = fctx.path() | ||||
parents = fctx.changectx().parents() | ||||
Brendan Cully
|
r3208 | p1 = parents[0].node() | ||
Eric Hopper
|
r2356 | |||
def diff(**map): | ||||
Brendan Cully
|
r3220 | yield self.diff(p1, n, [path]) | ||
Eric Hopper
|
r2356 | |||
yield self.t("filediff", | ||||
Brendan Cully
|
r3220 | file=path, | ||
Brendan Cully
|
r3208 | node=hex(n), | ||
Brendan Cully
|
r3220 | rev=fctx.rev(), | ||
Brendan Cully
|
r3208 | parent=self.siblings(parents), | ||
Brendan Cully
|
r3220 | child=self.siblings(fctx.children()), | ||
Eric Hopper
|
r2356 | diff=diff) | ||
archive_specs = { | ||||
Thomas Arendsen Hein
|
r2361 | 'bz2': ('application/x-tar', 'tbz2', '.tar.bz2', None), | ||
'gz': ('application/x-tar', 'tgz', '.tar.gz', None), | ||||
Eric Hopper
|
r2356 | 'zip': ('application/zip', 'zip', '.zip', None), | ||
} | ||||
Benoit Boissinot
|
r2394 | def archive(self, req, cnode, type_): | ||
Eric Hopper
|
r2356 | reponame = re.sub(r"\W+", "-", os.path.basename(self.reponame)) | ||
name = "%s-%s" % (reponame, short(cnode)) | ||||
Benoit Boissinot
|
r2394 | mimetype, artype, extension, encoding = self.archive_specs[type_] | ||
Eric Hopper
|
r2356 | headers = [('Content-type', mimetype), | ||
('Content-disposition', 'attachment; filename=%s%s' % | ||||
(name, extension))] | ||||
if encoding: | ||||
headers.append(('Content-encoding', encoding)) | ||||
req.header(headers) | ||||
archival.archive(self.repo, req.out, cnode, artype, prefix=name) | ||||
# add tags to things | ||||
# tags -> list of changesets corresponding to tags | ||||
# find tag, changeset, file | ||||
Vadim Gelfer
|
r2436 | def cleanpath(self, path): | ||
p = util.normpath(path) | ||||
if p[:2] == "..": | ||||
raise Exception("suspicious path") | ||||
return p | ||||
Eric Hopper
|
r2535 | def run(self): | ||
Eric Hopper
|
r2538 | if not os.environ.get('GATEWAY_INTERFACE', '').startswith("CGI/1."): | ||
Eric Hopper
|
r2535 | raise RuntimeError("This function is only intended to be called while running as a CGI script.") | ||
import mercurial.hgweb.wsgicgi as wsgicgi | ||||
from request import wsgiapplication | ||||
def make_web_app(): | ||||
Eric Hopper
|
r2538 | return self | ||
Eric Hopper
|
r2535 | wsgicgi.launch(wsgiapplication(make_web_app)) | ||
def run_wsgi(self, req): | ||||
Eric Hopper
|
r2356 | def header(**map): | ||
Eric Hopper
|
r2514 | header_file = cStringIO.StringIO(''.join(self.t("header", **map))) | ||
msg = mimetools.Message(header_file, 0) | ||||
req.header(msg.items()) | ||||
yield header_file.read() | ||||
Eric Hopper
|
r2356 | |||
Eric Hopper
|
r2534 | def rawfileheader(**map): | ||
req.header([('Content-type', map['mimetype']), | ||||
('Content-disposition', 'filename=%s' % map['file']), | ||||
('Content-length', str(len(map['raw'])))]) | ||||
yield '' | ||||
Eric Hopper
|
r2356 | def footer(**map): | ||
yield self.t("footer", | ||||
motd=self.repo.ui.config("web", "motd", ""), | ||||
**map) | ||||
def expand_form(form): | ||||
shortcuts = { | ||||
'cl': [('cmd', ['changelog']), ('rev', None)], | ||||
Josef "Jeff" Sipek
|
r2684 | 'sl': [('cmd', ['shortlog']), ('rev', None)], | ||
Eric Hopper
|
r2356 | 'cs': [('cmd', ['changeset']), ('node', None)], | ||
'f': [('cmd', ['file']), ('filenode', None)], | ||||
'fl': [('cmd', ['filelog']), ('filenode', None)], | ||||
'fd': [('cmd', ['filediff']), ('node', None)], | ||||
'fa': [('cmd', ['annotate']), ('filenode', None)], | ||||
'mf': [('cmd', ['manifest']), ('manifest', None)], | ||||
'ca': [('cmd', ['archive']), ('node', None)], | ||||
'tags': [('cmd', ['tags'])], | ||||
'tip': [('cmd', ['changeset']), ('node', ['tip'])], | ||||
'static': [('cmd', ['static']), ('file', None)] | ||||
} | ||||
for k in shortcuts.iterkeys(): | ||||
if form.has_key(k): | ||||
for name, value in shortcuts[k]: | ||||
if value is None: | ||||
value = form[k] | ||||
form[name] = value | ||||
del form[k] | ||||
Brendan Cully
|
r3260 | def rewrite_request(req): | ||
'''translate new web interface to traditional format''' | ||||
Brendan Cully
|
r3263 | def spliturl(req): | ||
def firstitem(query): | ||||
return query.split('&', 1)[0].split(';', 1)[0] | ||||
base = '' | ||||
if req.env.has_key('REPO_NAME'): | ||||
base = '/' + req.env['REPO_NAME'] | ||||
elif req.env.get('SCRIPT_NAME'): | ||||
base = req.env['SCRIPT_NAME'] | ||||
pi = req.env.get('PATH_INFO') | ||||
if pi: | ||||
Thomas Arendsen Hein
|
r3267 | while pi.startswith('//'): | ||
pi = pi[1:] | ||||
Brendan Cully
|
r3263 | if pi.startswith(base): | ||
if len(pi) > len(base): | ||||
base += '/' | ||||
query = pi[len(base):] | ||||
else: | ||||
if req.env.has_key('REPO_NAME'): | ||||
# We are using hgwebdir | ||||
base += '/' | ||||
else: | ||||
base += '?' | ||||
query = firstitem(req.env['QUERY_STRING']) | ||||
else: | ||||
base += '/' | ||||
query = pi[1:] | ||||
else: | ||||
base += '?' | ||||
query = firstitem(req.env['QUERY_STRING']) | ||||
return (base, query) | ||||
req.url, query = spliturl(req) | ||||
Brendan Cully
|
r3260 | if req.form.has_key('cmd'): | ||
# old style | ||||
return | ||||
args = query.split('/', 2) | ||||
if not args or not args[0]: | ||||
return | ||||
cmd = args.pop(0) | ||||
Brendan Cully
|
r3261 | style = cmd.rfind('-') | ||
if style != -1: | ||||
req.form['style'] = [cmd[:style]] | ||||
cmd = cmd[style+1:] | ||||
Brendan Cully
|
r3263 | # avoid accepting e.g. style parameter as command | ||
if hasattr(self, 'do_' + cmd): | ||||
req.form['cmd'] = [cmd] | ||||
Brendan Cully
|
r3260 | |||
if args and args[0]: | ||||
node = args.pop(0) | ||||
req.form['node'] = [node] | ||||
if args: | ||||
req.form['file'] = args | ||||
if cmd == 'static': | ||||
req.form['file'] = req.form['node'] | ||||
elif cmd == 'archive': | ||||
fn = req.form['node'][0] | ||||
for type_, spec in self.archive_specs.iteritems(): | ||||
ext = spec[2] | ||||
if fn.endswith(ext): | ||||
req.form['node'] = [fn[:-len(ext)]] | ||||
req.form['type'] = [type_] | ||||
Brendan Cully
|
r3270 | def queryprefix(**map): | ||
return req.url[-1] == '?' and ';' or '?' | ||||
def getentries(**map): | ||||
fields = {} | ||||
if req.form.has_key('style'): | ||||
style = req.form['style'][0] | ||||
if style != self.repo.ui.config('web', 'style', ''): | ||||
fields['style'] = style | ||||
if fields: | ||||
fields = ['%s=%s' % (k, urllib.quote(v)) | ||||
for k, v in fields.iteritems()] | ||||
yield '%s%s' % (queryprefix(), ';'.join(fields)) | ||||
else: | ||||
yield '' | ||||
Eric Hopper
|
r2356 | self.refresh() | ||
expand_form(req.form) | ||||
Brendan Cully
|
r3260 | rewrite_request(req) | ||
Eric Hopper
|
r2356 | |||
Vadim Gelfer
|
r2436 | m = os.path.join(self.templatepath, "map") | ||
Eric Hopper
|
r2356 | style = self.repo.ui.config("web", "style", "") | ||
if req.form.has_key('style'): | ||||
style = req.form['style'][0] | ||||
if style: | ||||
b = os.path.basename("map-" + style) | ||||
Vadim Gelfer
|
r2436 | p = os.path.join(self.templatepath, b) | ||
Eric Hopper
|
r2356 | if os.path.isfile(p): | ||
m = p | ||||
Brendan Cully
|
r3263 | if not req.url: | ||
port = req.env["SERVER_PORT"] | ||||
port = port != "80" and (":" + port) or "" | ||||
uri = req.env["REQUEST_URI"] | ||||
if "?" in uri: | ||||
uri = uri.split("?")[0] | ||||
req.url = "http://%s%s%s" % (req.env["SERVER_NAME"], port, uri) | ||||
Eric Hopper
|
r2356 | if not self.reponame: | ||
self.reponame = (self.repo.ui.config("web", "name") | ||||
Brendan Cully
|
r3263 | or req.env.get('REPO_NAME') | ||
or req.url.strip('/') or self.repo.root) | ||||
Eric Hopper
|
r2356 | |||
self.t = templater.templater(m, templater.common_filters, | ||||
Brendan Cully
|
r3263 | defaults={"url": req.url, | ||
Eric Hopper
|
r2356 | "repo": self.reponame, | ||
"header": header, | ||||
"footer": footer, | ||||
Eric Hopper
|
r2534 | "rawfileheader": rawfileheader, | ||
Brendan Cully
|
r3270 | "queryprefix": queryprefix, | ||
"getentries": getentries | ||||
Eric Hopper
|
r2356 | }) | ||
if not req.form.has_key('cmd'): | ||||
req.form['cmd'] = [self.t.cache['default'],] | ||||
cmd = req.form['cmd'][0] | ||||
Vadim Gelfer
|
r2436 | method = getattr(self, 'do_' + cmd, None) | ||
if method: | ||||
method(req) | ||||
Eric Hopper
|
r2356 | else: | ||
req.write(self.t("error")) | ||||
Vadim Gelfer
|
r2436 | |||
Brendan Cully
|
r3226 | def changectx(self, req): | ||
if req.form.has_key('node'): | ||||
changeid = req.form['node'][0] | ||||
else: | ||||
changeid = req.form['manifest'][0] | ||||
try: | ||||
ctx = self.repo.changectx(changeid) | ||||
except hg.RepoError: | ||||
man = self.repo.manifest | ||||
mn = man.lookup(changeid) | ||||
ctx = self.repo.changectx(man.linkrev(mn)) | ||||
return ctx | ||||
def filectx(self, req): | ||||
path = self.cleanpath(req.form['file'][0]) | ||||
if req.form.has_key('node'): | ||||
changeid = req.form['node'][0] | ||||
else: | ||||
changeid = req.form['filenode'][0] | ||||
try: | ||||
ctx = self.repo.changectx(changeid) | ||||
fctx = ctx.filectx(path) | ||||
except hg.RepoError: | ||||
fctx = self.repo.filectx(path, fileid=changeid) | ||||
return fctx | ||||
Frank Kingswood
|
r2666 | def stripes(self, parity): | ||
"make horizontal stripes for easier reading" | ||||
if self.stripecount: | ||||
return (1 + parity / self.stripecount) & 1 | ||||
else: | ||||
return 0 | ||||
Brendan Cully
|
r3260 | def do_log(self, req): | ||
if req.form.has_key('file') and req.form['file'][0]: | ||||
self.do_filelog(req) | ||||
else: | ||||
self.do_changelog(req) | ||||
def do_rev(self, req): | ||||
self.do_changeset(req) | ||||
def do_file(self, req): | ||||
path = req.form.get('file', [''])[0] | ||||
if path: | ||||
try: | ||||
req.write(self.filerevision(self.filectx(req))) | ||||
return | ||||
except hg.RepoError: | ||||
pass | ||||
path = self.cleanpath(path) | ||||
req.write(self.manifest(self.changectx(req), '/' + path)) | ||||
def do_diff(self, req): | ||||
self.do_filediff(req) | ||||
Brendan Cully
|
r3226 | def do_changelog(self, req, shortlog = False): | ||
if req.form.has_key('node'): | ||||
ctx = self.changectx(req) | ||||
Brendan Cully
|
r3220 | else: | ||
Brendan Cully
|
r3226 | if req.form.has_key('rev'): | ||
hi = req.form['rev'][0] | ||||
else: | ||||
hi = self.repo.changelog.count() - 1 | ||||
try: | ||||
ctx = self.repo.changectx(hi) | ||||
except hg.RepoError: | ||||
req.write(self.search(hi)) # XXX redirect to 404 page? | ||||
return | ||||
Vadim Gelfer
|
r2436 | |||
Brendan Cully
|
r3226 | req.write(self.changelog(ctx, shortlog = shortlog)) | ||
Vadim Gelfer
|
r2436 | |||
Josef "Jeff" Sipek
|
r2684 | def do_shortlog(self, req): | ||
Brendan Cully
|
r3226 | self.do_changelog(req, shortlog = True) | ||
Josef "Jeff" Sipek
|
r2684 | |||
Vadim Gelfer
|
r2436 | def do_changeset(self, req): | ||
Brendan Cully
|
r3220 | ctx = self.repo.changectx(req.form['node'][0]) | ||
req.write(self.changeset(ctx)) | ||||
Vadim Gelfer
|
r2436 | |||
def do_manifest(self, req): | ||||
Brendan Cully
|
r3226 | req.write(self.manifest(self.changectx(req), | ||
Vadim Gelfer
|
r2436 | self.cleanpath(req.form['path'][0]))) | ||
def do_tags(self, req): | ||||
req.write(self.tags()) | ||||
def do_summary(self, req): | ||||
req.write(self.summary()) | ||||
def do_filediff(self, req): | ||||
Brendan Cully
|
r3226 | req.write(self.filediff(self.filectx(req))) | ||
Vadim Gelfer
|
r2436 | |||
def do_annotate(self, req): | ||||
Brendan Cully
|
r3226 | req.write(self.fileannotate(self.filectx(req))) | ||
Vadim Gelfer
|
r2436 | |||
def do_filelog(self, req): | ||||
Brendan Cully
|
r3226 | req.write(self.filelog(self.filectx(req))) | ||
Vadim Gelfer
|
r2436 | |||
def do_heads(self, req): | ||||
resp = " ".join(map(hex, self.repo.heads())) + "\n" | ||||
req.httphdr("application/mercurial-0.1", length=len(resp)) | ||||
req.write(resp) | ||||
def do_branches(self, req): | ||||
nodes = [] | ||||
if req.form.has_key('nodes'): | ||||
nodes = map(bin, req.form['nodes'][0].split(" ")) | ||||
resp = cStringIO.StringIO() | ||||
for b in self.repo.branches(nodes): | ||||
resp.write(" ".join(map(hex, b)) + "\n") | ||||
resp = resp.getvalue() | ||||
req.httphdr("application/mercurial-0.1", length=len(resp)) | ||||
req.write(resp) | ||||
def do_between(self, req): | ||||
if req.form.has_key('pairs'): | ||||
pairs = [map(bin, p.split("-")) | ||||
for p in req.form['pairs'][0].split(" ")] | ||||
resp = cStringIO.StringIO() | ||||
for b in self.repo.between(pairs): | ||||
resp.write(" ".join(map(hex, b)) + "\n") | ||||
resp = resp.getvalue() | ||||
req.httphdr("application/mercurial-0.1", length=len(resp)) | ||||
req.write(resp) | ||||
def do_changegroup(self, req): | ||||
req.httphdr("application/mercurial-0.1") | ||||
nodes = [] | ||||
if not self.allowpull: | ||||
return | ||||
if req.form.has_key('roots'): | ||||
nodes = map(bin, req.form['roots'][0].split(" ")) | ||||
z = zlib.compressobj() | ||||
f = self.repo.changegroup(nodes, 'serve') | ||||
while 1: | ||||
chunk = f.read(4096) | ||||
if not chunk: | ||||
break | ||||
req.write(z.compress(chunk)) | ||||
req.write(z.flush()) | ||||
def do_archive(self, req): | ||||
changeset = self.repo.lookup(req.form['node'][0]) | ||||
type_ = req.form['type'][0] | ||||
Thomas Arendsen Hein
|
r2500 | allowed = self.repo.ui.configlist("web", "allow_archive") | ||
Vadim Gelfer
|
r2436 | if (type_ in self.archives and (type_ in allowed or | ||
self.repo.ui.configbool("web", "allow" + type_, False))): | ||||
self.archive(req, changeset, type_) | ||||
return | ||||
req.write(self.t("error")) | ||||
def do_static(self, req): | ||||
fname = req.form['file'][0] | ||||
static = self.repo.ui.config("web", "static", | ||||
os.path.join(self.templatepath, | ||||
"static")) | ||||
Eric Hopper
|
r2514 | req.write(staticfile(static, fname, req) | ||
Vadim Gelfer
|
r2436 | or self.t("error", error="%r not found" % fname)) | ||
Vadim Gelfer
|
r2442 | |||
def do_capabilities(self, req): | ||||
Vadim Gelfer
|
r2621 | caps = ['unbundle'] | ||
Vadim Gelfer
|
r2622 | if self.repo.ui.configbool('server', 'uncompressed'): | ||
Vadim Gelfer
|
r2621 | caps.append('stream=%d' % self.repo.revlogversion) | ||
resp = ' '.join(caps) | ||||
Vadim Gelfer
|
r2442 | req.httphdr("application/mercurial-0.1", length=len(resp)) | ||
req.write(resp) | ||||
Vadim Gelfer
|
r2466 | def check_perm(self, req, op, default): | ||
'''check permission for operation based on user auth. | ||||
return true if op allowed, else false. | ||||
default is policy to use if no config given.''' | ||||
user = req.env.get('REMOTE_USER') | ||||
Thomas Arendsen Hein
|
r2501 | deny = self.repo.ui.configlist('web', 'deny_' + op) | ||
Vadim Gelfer
|
r2466 | if deny and (not user or deny == ['*'] or user in deny): | ||
return False | ||||
Thomas Arendsen Hein
|
r2501 | allow = self.repo.ui.configlist('web', 'allow_' + op) | ||
Vadim Gelfer
|
r2466 | return (allow and (allow == ['*'] or user in allow)) or default | ||
Vadim Gelfer
|
r2464 | def do_unbundle(self, req): | ||
Vadim Gelfer
|
r2466 | def bail(response, headers={}): | ||
length = int(req.env['CONTENT_LENGTH']) | ||||
for s in util.filechunkiter(req, limit=length): | ||||
# drain incoming bundle, else client will not see | ||||
# response when run outside cgi script | ||||
pass | ||||
req.httphdr("application/mercurial-0.1", headers=headers) | ||||
req.write('0\n') | ||||
req.write(response) | ||||
# require ssl by default, auth info cannot be sniffed and | ||||
# replayed | ||||
ssl_req = self.repo.ui.configbool('web', 'push_ssl', True) | ||||
Vadim Gelfer
|
r2673 | if ssl_req: | ||
if not req.env.get('HTTPS'): | ||||
bail(_('ssl required\n')) | ||||
return | ||||
proto = 'https' | ||||
else: | ||||
proto = 'http' | ||||
Vadim Gelfer
|
r2466 | |||
# do not allow push unless explicitly allowed | ||||
if not self.check_perm(req, 'push', False): | ||||
bail(_('push not authorized\n'), | ||||
headers={'status': '401 Unauthorized'}) | ||||
return | ||||
req.httphdr("application/mercurial-0.1") | ||||
Vadim Gelfer
|
r2464 | their_heads = req.form['heads'][0].split(' ') | ||
def check_heads(): | ||||
heads = map(hex, self.repo.heads()) | ||||
return their_heads == [hex('force')] or their_heads == heads | ||||
# fail early if possible | ||||
if not check_heads(): | ||||
Vadim Gelfer
|
r2466 | bail(_('unsynced changes\n')) | ||
Vadim Gelfer
|
r2464 | return | ||
# do not lock repo until all changegroup data is | ||||
# streamed. save to temporary file. | ||||
fd, tempname = tempfile.mkstemp(prefix='hg-unbundle-') | ||||
fp = os.fdopen(fd, 'wb+') | ||||
try: | ||||
length = int(req.env['CONTENT_LENGTH']) | ||||
for s in util.filechunkiter(req, limit=length): | ||||
fp.write(s) | ||||
lock = self.repo.lock() | ||||
try: | ||||
if not check_heads(): | ||||
req.write('0\n') | ||||
req.write(_('unsynced changes\n')) | ||||
return | ||||
fp.seek(0) | ||||
# send addchangegroup output to client | ||||
old_stdout = sys.stdout | ||||
sys.stdout = cStringIO.StringIO() | ||||
try: | ||||
Vadim Gelfer
|
r2673 | url = 'remote:%s:%s' % (proto, | ||
req.env.get('REMOTE_HOST', '')) | ||||
ret = self.repo.addchangegroup(fp, 'serve', url) | ||||
Vadim Gelfer
|
r2464 | finally: | ||
Alexis S. L. Carvalho
|
r2558 | val = sys.stdout.getvalue() | ||
Vadim Gelfer
|
r2464 | sys.stdout = old_stdout | ||
Alexis S. L. Carvalho
|
r2558 | req.write('%d\n' % ret) | ||
req.write(val) | ||||
Vadim Gelfer
|
r2464 | finally: | ||
lock.release() | ||||
finally: | ||||
fp.close() | ||||
os.unlink(tempname) | ||||
Vadim Gelfer
|
r2612 | |||
def do_stream_out(self, req): | ||||
req.httphdr("application/mercurial-0.1") | ||||
streamclone.stream_out(self.repo, req) | ||||