hgwebdir_mod.py
260 lines
| 10.7 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
|
r4462 | from common import get_mtime, staticfile, style_map, paritygen | ||
Matt Mackall
|
r3877 | from hgweb_mod import hgweb | ||
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): | ||
Alexis S. L. Carvalho
|
r5063 | return [(util.pconvert(name.strip(os.sep)), path) | ||
for name, path in items] | ||||
Eric Hopper
|
r2356 | |||
Alexis S. L. Carvalho
|
r4079 | self.parentui = parentui | ||
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 | ||||
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): | ||
Thomas Arendsen Hein
|
r3882 | header_file = cStringIO.StringIO( | ||
''.join(tmpl("header", encoding=util._encoding, **map))) | ||||
Eric Hopper
|
r2514 | msg = mimetools.Message(header_file, 0) | ||
req.header(msg.items()) | ||||
yield header_file.read() | ||||
Eric Hopper
|
r2356 | |||
def footer(**map): | ||||
Thomas Arendsen Hein
|
r3479 | yield tmpl("footer", **map) | ||
Eric Hopper
|
r2356 | |||
Alexis S. L. Carvalho
|
r3488 | def motd(**map): | ||
Alexis S. L. Carvalho
|
r4080 | if self.motd is not None: | ||
yield self.motd | ||||
else: | ||||
yield config('web', 'motd', '') | ||||
Alexis S. L. Carvalho
|
r3488 | |||
Dirkjan Ochtman
|
r5289 | parentui = self.parentui or ui.ui(report_untrusted=False, | ||
interactive=False) | ||||
Alexis S. L. Carvalho
|
r4079 | |||
Alexis S. L. Carvalho
|
r4080 | def config(section, name, default=None, untrusted=True): | ||
return parentui.config(section, name, default, untrusted) | ||||
Brendan Cully
|
r3328 | url = req.env['REQUEST_URI'].split('?')[0] | ||
if not url.endswith('/'): | ||||
url += '/' | ||||
Brendan Cully
|
r4841 | pathinfo = req.env.get('PATH_INFO', '').strip('/') + '/' | ||
base = url[:len(url) - len(pathinfo)] | ||||
if not base.endswith('/'): | ||||
base += '/' | ||||
Brendan Cully
|
r3328 | |||
Brendan Cully
|
r4841 | staticurl = config('web', 'staticurl') or base + 'static/' | ||
Alexis S. L. Carvalho
|
r4084 | if not staticurl.endswith('/'): | ||
staticurl += '/' | ||||
Edouard Gomez
|
r3221 | style = self.style | ||
Alexis S. L. Carvalho
|
r4080 | if style is None: | ||
style = config('web', 'style', '') | ||||
Edouard Gomez
|
r3221 | if req.form.has_key('style'): | ||
style = req.form['style'][0] | ||||
Thomas Arendsen Hein
|
r4462 | if self.stripecount is None: | ||
self.stripecount = int(config('web', 'stripes', 1)) | ||||
Thomas Arendsen Hein
|
r3276 | mapfile = style_map(templater.templatepath(), style) | ||
tmpl = templater.templater(mapfile, templater.common_filters, | ||||
Eric Hopper
|
r2356 | defaults={"header": header, | ||
Brendan Cully
|
r3328 | "footer": footer, | ||
Alexis S. L. Carvalho
|
r3488 | "motd": motd, | ||
Alexis S. L. Carvalho
|
r4084 | "url": url, | ||
"staticurl": staticurl}) | ||||
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 | |||
Alexis S. L. Carvalho
|
r4079 | u = ui.ui(parentui=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 | ||||
Eric Hopper
|
r2356 | url = ('/'.join([req.env["REQUEST_URI"].split('?')[0], name]) | ||
Brendan Cully
|
r3262 | .replace("//", "/")) + '/' | ||
Eric Hopper
|
r2356 | |||
# update time with local timezone | ||||
try: | ||||
d = (get_mtime(path), util.makedate()[1]) | ||||
except OSError: | ||||
continue | ||||
contact = (get("ui", "username") or # preferred | ||||
get("web", "contact") or # deprecated | ||||
get("web", "author", "")) # also | ||||
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 | ||
Brendan Cully
|
r4841 | def makeindex(req, subdir=""): | ||
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 = "" | ||||
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))) | ||||
Alexis S. L. Carvalho
|
r4244 | try: | ||
virtual = req.env.get("PATH_INFO", "").strip('/') | ||||
if virtual.startswith('static/'): | ||||
static = os.path.join(templater.templatepath(), 'static') | ||||
fname = virtual[7:] | ||||
req.write(staticfile(static, fname, req) or | ||||
tmpl('error', error='%r not found' % fname)) | ||||
elif virtual: | ||||
Brendan Cully
|
r4850 | repos = dict(self.repos) | ||
Alexis S. L. Carvalho
|
r4244 | while virtual: | ||
Brendan Cully
|
r4850 | real = repos.get(virtual) | ||
Alexis S. L. Carvalho
|
r4244 | if real: | ||
Brendan Cully
|
r4852 | req.env['REPO_NAME'] = virtual | ||
try: | ||||
repo = hg.repository(parentui, real) | ||||
hgweb(repo).run_wsgi(req) | ||||
except IOError, inst: | ||||
req.write(tmpl("error", error=inst.strerror)) | ||||
except hg.RepoError, inst: | ||||
req.write(tmpl("error", error=str(inst))) | ||||
return | ||||
# browse subdirectories | ||||
subdir = virtual + '/' | ||||
if [r for r in repos if r.startswith(subdir)]: | ||||
makeindex(req, subdir) | ||||
return | ||||
Alexis S. L. Carvalho
|
r4244 | up = virtual.rfind('/') | ||
if up < 0: | ||||
break | ||||
virtual = virtual[:up] | ||||
Thomas Arendsen Hein
|
r4957 | |||
Brendan Cully
|
r4852 | req.write(tmpl("notfound", repo=virtual)) | ||
Eric Hopper
|
r2356 | else: | ||
Alexis S. L. Carvalho
|
r4244 | if req.form.has_key('static'): | ||
static = os.path.join(templater.templatepath(), "static") | ||||
fname = req.form['static'][0] | ||||
req.write(staticfile(static, fname, req) | ||||
or tmpl("error", error="%r not found" % fname)) | ||||
else: | ||||
Brendan Cully
|
r4841 | makeindex(req) | ||
Alexis S. L. Carvalho
|
r4244 | finally: | ||
tmpl = None | ||||