hgwebdir_mod.py
276 lines
| 10.8 KiB
| text/x-python
|
PythonLexer
Eric Hopper
|
r2391 | # hgweb/hgwebdir_mod.py - Web interface for a directory of repositories. | ||
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. | ||||
Matt Mackall
|
r3877 | import os, mimetools, cStringIO | ||
Eric Hopper
|
r2356 | from mercurial.i18n import gettext as _ | ||
Matt Mackall
|
r3877 | from mercurial import ui, hg, util, templater | ||
Thomas Arendsen Hein
|
r5779 | from common import ErrorResponse, get_mtime, staticfile, style_map, paritygen, \ | ||
get_contact | ||||
Matt Mackall
|
r3877 | from hgweb_mod import hgweb | ||
Dirkjan Ochtman
|
r5566 | from request import wsgirequest | ||
Eric Hopper
|
r2356 | |||
# This is a stopgap | ||||
class hgwebdir(object): | ||||
Alexis S. L. Carvalho
|
r4079 | def __init__(self, config, parentui=None): | ||
Eric Hopper
|
r2356 | def cleannames(items): | ||
Patrick Mezard
|
r5584 | return [(util.pconvert(name).strip('/'), path) | ||
Alexis S. L. Carvalho
|
r5063 | for name, path in items] | ||
Eric Hopper
|
r2356 | |||
Dirkjan Ochtman
|
r5601 | self.parentui = parentui or ui.ui(report_untrusted=False, | ||
interactive = False) | ||||
Alexis S. L. Carvalho
|
r4080 | self.motd = None | ||
self.style = None | ||||
Thomas Arendsen Hein
|
r4462 | self.stripecount = None | ||
Eric Hopper
|
r2356 | self.repos_sorted = ('name', False) | ||
if isinstance(config, (list, tuple)): | ||||
self.repos = cleannames(config) | ||||
self.repos_sorted = ('', False) | ||||
elif isinstance(config, dict): | ||||
self.repos = cleannames(config.items()) | ||||
self.repos.sort() | ||||
else: | ||||
Michael Gebetsroither
|
r4051 | if isinstance(config, util.configparser): | ||
cp = config | ||||
else: | ||||
cp = util.configparser() | ||||
cp.read(config) | ||||
Eric Hopper
|
r2356 | self.repos = [] | ||
Edouard Gomez
|
r3221 | if cp.has_section('web'): | ||
if cp.has_option('web', 'motd'): | ||||
self.motd = cp.get('web', 'motd') | ||||
if cp.has_option('web', 'style'): | ||||
Thomas Arendsen Hein
|
r3223 | self.style = cp.get('web', 'style') | ||
Thomas Arendsen Hein
|
r4462 | if cp.has_option('web', 'stripes'): | ||
self.stripecount = int(cp.get('web', 'stripes')) | ||||
Eric Hopper
|
r2356 | if cp.has_section('paths'): | ||
self.repos.extend(cleannames(cp.items('paths'))) | ||||
if cp.has_section('collections'): | ||||
for prefix, root in cp.items('collections'): | ||||
for path in util.walkrepos(root): | ||||
repo = os.path.normpath(path) | ||||
name = repo | ||||
if name.startswith(prefix): | ||||
name = name[len(prefix):] | ||||
self.repos.append((name.lstrip(os.sep), repo)) | ||||
self.repos.sort() | ||||
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 | ||||
Dirkjan Ochtman
|
r5566 | wsgicgi.launch(self) | ||
def __call__(self, env, respond): | ||||
req = wsgirequest(env, respond) | ||||
self.run_wsgi(req) | ||||
return req | ||||
Eric Hopper
|
r2535 | |||
def run_wsgi(self, req): | ||||
Eric Hopper
|
r2356 | |||
Dirkjan Ochtman
|
r5601 | try: | ||
try: | ||||
Dirkjan Ochtman
|
r5603 | |||
Dirkjan Ochtman
|
r5601 | virtual = req.env.get("PATH_INFO", "").strip('/') | ||
Thomas Arendsen Hein
|
r5760 | |||
Dirkjan Ochtman
|
r5603 | # a static file | ||
if virtual.startswith('static/') or 'static' in req.form: | ||||
Dirkjan Ochtman
|
r5601 | static = os.path.join(templater.templatepath(), 'static') | ||
Dirkjan Ochtman
|
r5603 | if virtual.startswith('static/'): | ||
fname = virtual[7:] | ||||
else: | ||||
fname = req.form['static'][0] | ||||
Dirkjan Ochtman
|
r5601 | req.write(staticfile(static, fname, req)) | ||
Dirkjan Ochtman
|
r5603 | return | ||
# top-level index | ||||
elif not virtual: | ||||
tmpl = self.templater(req) | ||||
self.makeindex(req, tmpl) | ||||
return | ||||
Dirkjan Ochtman
|
r5601 | |||
Dirkjan Ochtman
|
r5603 | # nested indexes and hgwebs | ||
repos = dict(self.repos) | ||||
while virtual: | ||||
real = repos.get(virtual) | ||||
if real: | ||||
req.env['REPO_NAME'] = virtual | ||||
try: | ||||
repo = hg.repository(self.parentui, real) | ||||
hgweb(repo).run_wsgi(req) | ||||
Dirkjan Ochtman
|
r5601 | return | ||
Dirkjan Ochtman
|
r5603 | except IOError, inst: | ||
raise ErrorResponse(500, inst.strerror) | ||||
except hg.RepoError, inst: | ||||
raise ErrorResponse(500, str(inst)) | ||||
Dirkjan Ochtman
|
r5601 | |||
Dirkjan Ochtman
|
r5603 | # browse subdirectories | ||
subdir = virtual + '/' | ||||
if [r for r in repos if r.startswith(subdir)]: | ||||
Dirkjan Ochtman
|
r5602 | tmpl = self.templater(req) | ||
Dirkjan Ochtman
|
r5603 | self.makeindex(req, tmpl, subdir) | ||
return | ||||
up = virtual.rfind('/') | ||||
if up < 0: | ||||
break | ||||
virtual = virtual[:up] | ||||
# prefixes not found | ||||
tmpl = self.templater(req) | ||||
req.respond(404, tmpl("notfound", repo=virtual)) | ||||
Thomas Arendsen Hein
|
r5760 | |||
Dirkjan Ochtman
|
r5601 | except ErrorResponse, err: | ||
Dirkjan Ochtman
|
r5603 | tmpl = self.templater(req) | ||
Dirkjan Ochtman
|
r5601 | req.respond(err.code, tmpl('error', error=err.message or '')) | ||
finally: | ||||
tmpl = None | ||||
def makeindex(self, req, tmpl, subdir=""): | ||||
Eric Hopper
|
r2356 | def archivelist(ui, nodeid, url): | ||
Alexis S. L. Carvalho
|
r3556 | allowed = ui.configlist("web", "allow_archive", untrusted=True) | ||
Brendan Cully
|
r3262 | for i in [('zip', '.zip'), ('gz', '.tar.gz'), ('bz2', '.tar.bz2')]: | ||
Alexis S. L. Carvalho
|
r3556 | if i[0] in allowed or ui.configbool("web", "allow" + i[0], | ||
untrusted=True): | ||||
Brendan Cully
|
r3262 | yield {"type" : i[0], "extension": i[1], | ||
"node": nodeid, "url": url} | ||||
Eric Hopper
|
r2356 | |||
Brendan Cully
|
r4841 | def entries(sortcolumn="", descending=False, subdir="", **map): | ||
Thomas Arendsen Hein
|
r3365 | def sessionvars(**map): | ||
fields = [] | ||||
if req.form.has_key('style'): | ||||
style = req.form['style'][0] | ||||
if style != get('web', 'style', ''): | ||||
fields.append(('style', style)) | ||||
separator = url[-1] == '?' and ';' or '?' | ||||
for name, value in fields: | ||||
yield dict(name=name, value=value, separator=separator) | ||||
separator = ';' | ||||
Eric Hopper
|
r2356 | rows = [] | ||
Thomas Arendsen Hein
|
r4462 | parity = paritygen(self.stripecount) | ||
Eric Hopper
|
r2356 | for name, path in self.repos: | ||
Brendan Cully
|
r4841 | if not name.startswith(subdir): | ||
continue | ||||
Brendan Cully
|
r4843 | name = name[len(subdir):] | ||
Brendan Cully
|
r4841 | |||
Dirkjan Ochtman
|
r5601 | u = ui.ui(parentui=self.parentui) | ||
Eric Hopper
|
r2356 | try: | ||
u.readconfig(os.path.join(path, '.hg', 'hgrc')) | ||||
Alexis S. L. Carvalho
|
r5332 | except Exception, e: | ||
u.warn(_('error reading %s/.hg/hgrc: %s\n' % (path, e))) | ||||
continue | ||||
Alexis S. L. Carvalho
|
r3556 | def get(section, name, default=None): | ||
return u.config(section, name, default, untrusted=True) | ||||
Eric Hopper
|
r2356 | |||
Markus F.X.J. Oberhumer
|
r4709 | if u.configbool("web", "hidden", untrusted=True): | ||
continue | ||||
Dirkjan Ochtman
|
r5579 | parts = [req.env['PATH_INFO'], name] | ||
if req.env['SCRIPT_NAME']: | ||||
Thomas Arendsen Hein
|
r5760 | parts.insert(0, req.env['SCRIPT_NAME']) | ||
Dirkjan Ochtman
|
r5579 | url = ('/'.join(parts).replace("//", "/")) + '/' | ||
Eric Hopper
|
r2356 | |||
# update time with local timezone | ||||
try: | ||||
d = (get_mtime(path), util.makedate()[1]) | ||||
except OSError: | ||||
continue | ||||
Thomas Arendsen Hein
|
r5779 | contact = get_contact(get) | ||
Eric Hopper
|
r2356 | description = get("web", "description", "") | ||
name = get("web", "name", name) | ||||
row = dict(contact=contact or "unknown", | ||||
contact_sort=contact.upper() or "unknown", | ||||
name=name, | ||||
name_sort=name, | ||||
url=url, | ||||
description=description or "unknown", | ||||
description_sort=description.upper() or "unknown", | ||||
lastchange=d, | ||||
lastchange_sort=d[1]-d[0], | ||||
Thomas Arendsen Hein
|
r3365 | sessionvars=sessionvars, | ||
Eric Hopper
|
r2356 | archives=archivelist(u, "tip", url)) | ||
if (not sortcolumn | ||||
or (sortcolumn, descending) == self.repos_sorted): | ||||
# fast path for unsorted output | ||||
Thomas Arendsen Hein
|
r4462 | row['parity'] = parity.next() | ||
Eric Hopper
|
r2356 | yield row | ||
else: | ||||
rows.append((row["%s_sort" % sortcolumn], row)) | ||||
if rows: | ||||
rows.sort() | ||||
if descending: | ||||
rows.reverse() | ||||
for key, row in rows: | ||||
Thomas Arendsen Hein
|
r4462 | row['parity'] = parity.next() | ||
Eric Hopper
|
r2356 | yield row | ||
Dirkjan Ochtman
|
r5601 | sortable = ["name", "description", "contact", "lastchange"] | ||
sortcolumn, descending = self.repos_sorted | ||||
if req.form.has_key('sort'): | ||||
sortcolumn = req.form['sort'][0] | ||||
descending = sortcolumn.startswith('-') | ||||
if descending: | ||||
sortcolumn = sortcolumn[1:] | ||||
if sortcolumn not in sortable: | ||||
sortcolumn = "" | ||||
Brendan Cully
|
r4841 | |||
Dirkjan Ochtman
|
r5601 | sort = [("sort_%s" % column, | ||
"%s%s" % ((not descending and column == sortcolumn) | ||||
and "-" or "", column)) | ||||
for column in sortable] | ||||
req.write(tmpl("index", entries=entries, subdir=subdir, | ||||
sortcolumn=sortcolumn, descending=descending, | ||||
**dict(sort))) | ||||
Dirkjan Ochtman
|
r5602 | |||
def templater(self, req): | ||||
def header(**map): | ||||
header_file = cStringIO.StringIO( | ||||
''.join(tmpl("header", encoding=util._encoding, **map))) | ||||
msg = mimetools.Message(header_file, 0) | ||||
req.header(msg.items()) | ||||
yield header_file.read() | ||||
def footer(**map): | ||||
yield tmpl("footer", **map) | ||||
def motd(**map): | ||||
if self.motd is not None: | ||||
yield self.motd | ||||
else: | ||||
yield config('web', 'motd', '') | ||||
def config(section, name, default=None, untrusted=True): | ||||
return self.parentui.config(section, name, default, untrusted) | ||||
url = req.env.get('SCRIPT_NAME', '') | ||||
if not url.endswith('/'): | ||||
url += '/' | ||||
staticurl = config('web', 'staticurl') or url + 'static/' | ||||
if not staticurl.endswith('/'): | ||||
staticurl += '/' | ||||
style = self.style | ||||
if style is None: | ||||
style = config('web', 'style', '') | ||||
if req.form.has_key('style'): | ||||
style = req.form['style'][0] | ||||
if self.stripecount is None: | ||||
self.stripecount = int(config('web', 'stripes', 1)) | ||||
mapfile = style_map(templater.templatepath(), style) | ||||
tmpl = templater.templater(mapfile, templater.common_filters, | ||||
defaults={"header": header, | ||||
"footer": footer, | ||||
"motd": motd, | ||||
"url": url, | ||||
"staticurl": staticurl}) | ||||
return tmpl | ||||