webcommands.py
1411 lines
| 43.5 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> | ||||
# | ||||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Dirkjan Ochtman
|
r5591 | |||
Yuya Nishihara
|
r27046 | from __future__ import absolute_import | ||
import copy | ||||
import mimetypes | ||||
import os | ||||
import re | ||||
from ..i18n import _ | ||||
r35582 | from ..node import hex, nullid, short | |||
Yuya Nishihara
|
r27046 | |||
from .common import ( | ||||
ErrorResponse, | ||||
HTTP_FORBIDDEN, | ||||
HTTP_NOT_FOUND, | ||||
HTTP_OK, | ||||
get_contact, | ||||
paritygen, | ||||
staticfile, | ||||
) | ||||
from .. import ( | ||||
archival, | ||||
Yuya Nishihara
|
r32904 | dagop, | ||
Yuya Nishihara
|
r27046 | encoding, | ||
error, | ||||
graphmod, | ||||
Augie Fackler
|
r34810 | pycompat, | ||
Yuya Nishihara
|
r27046 | revset, | ||
Yuya Nishihara
|
r31024 | revsetlang, | ||
Yuya Nishihara
|
r27046 | scmutil, | ||
Yuya Nishihara
|
r31023 | smartset, | ||
Yuya Nishihara
|
r27046 | templater, | ||
util, | ||||
) | ||||
from . import ( | ||||
webutil, | ||||
) | ||||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | __all__ = [] | ||
Gregory Szorc
|
r24077 | commands = {} | ||
Gregory Szorc
|
r24076 | |||
class webcommand(object): | ||||
"""Decorator used to register a web command handler. | ||||
The decorator takes as its positional arguments the name/path the | ||||
command should be accessible under. | ||||
Usage: | ||||
Dirkjan Ochtman
|
r5963 | |||
Gregory Szorc
|
r24076 | @webcommand('mycommand') | ||
def mycommand(web, req, tmpl): | ||||
pass | ||||
""" | ||||
Dirkjan Ochtman
|
r5963 | |||
Gregory Szorc
|
r24076 | def __init__(self, name): | ||
self.name = name | ||||
def __call__(self, func): | ||||
__all__.append(self.name) | ||||
Gregory Szorc
|
r24077 | commands[self.name] = func | ||
Gregory Szorc
|
r24076 | return func | ||
@webcommand('log') | ||||
Dirkjan Ochtman
|
r5600 | def log(web, req, tmpl): | ||
Gregory Szorc
|
r24087 | """ | ||
/log[/{revision}[/{path}]] | ||||
-------------------------- | ||||
Show repository or file history. | ||||
For URLs of the form ``/log/{revision}``, a list of changesets starting at | ||||
the specified changeset identifier is shown. If ``{revision}`` is not | ||||
defined, the default is ``tip``. This form is equivalent to the | ||||
``changelog`` handler. | ||||
For URLs of the form ``/log/{revision}/{file}``, the history for a specific | ||||
file will be shown. This form is equivalent to the ``filelog`` handler. | ||||
""" | ||||
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 | |||
Gregory Szorc
|
r24076 | @webcommand('rawfile') | ||
Dirkjan Ochtman
|
r5890 | def rawfile(web, req, tmpl): | ||
Boris Feld
|
r34609 | guessmime = web.configbool('web', 'guessmime') | ||
Matt Mackall
|
r15004 | |||
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) | ||
Gregory Szorc
|
r25660 | except error.LookupError as inst: | ||
Dirkjan Ochtman
|
r6368 | 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() | ||||
Matt Mackall
|
r15004 | mt = 'application/binary' | ||
if guessmime: | ||||
mt = mimetypes.guess_type(path)[0] | ||||
if mt is None: | ||||
Jordi Gutiérrez Hermoso
|
r24306 | if util.binary(text): | ||
mt = 'application/binary' | ||||
else: | ||||
mt = 'text/plain' | ||||
Julian Cowley
|
r11332 | if mt.startswith('text/'): | ||
mt += '; charset="%s"' % encoding.encoding | ||||
Dirkjan Ochtman
|
r5890 | |||
Mads Kiilerich
|
r18352 | req.respond(HTTP_OK, mt, path, body=text) | ||
return [] | ||||
Dirkjan Ochtman
|
r5890 | |||
r25602 | def _filerevision(web, req, tmpl, fctx): | |||
Dirkjan Ochtman
|
r6393 | f = fctx.path() | ||
text = fctx.data() | ||||
parity = paritygen(web.stripecount) | ||||
Denis Laxalde
|
r32070 | ishead = fctx.filerev() in fctx.filelog().headrevs() | ||
Dirkjan Ochtman
|
r6393 | |||
Alexander Plavin
|
r19657 | if util.binary(text): | ||
Dirkjan Ochtman
|
r6393 | mt = mimetypes.guess_type(f)[0] or 'application/octet-stream' | ||
text = '(binary:%s)' % mt | ||||
def lines(): | ||||
Nicolas Dumazet
|
r9136 | for lineno, t in enumerate(text.splitlines(True)): | ||
Dirkjan Ochtman
|
r6393 | yield {"line": t, | ||
"lineid": "l%d" % (lineno + 1), | ||||
"linenumber": "% 6d" % (lineno + 1), | ||||
timeless
|
r29216 | "parity": next(parity)} | ||
Dirkjan Ochtman
|
r6393 | |||
return tmpl("filerevision", | ||||
file=f, | ||||
path=webutil.up(f), | ||||
text=lines(), | ||||
r25602 | symrev=webutil.symrevorshortnode(req, fctx), | |||
Matt Mackall
|
r6434 | rename=webutil.renamelink(fctx), | ||
r27294 | permissions=fctx.manifest().flags(f), | |||
Denis Laxalde
|
r32070 | ishead=int(ishead), | ||
Pulkit Goyal
|
r36452 | **pycompat.strkwargs(webutil.commonentry(web.repo, fctx))) | ||
Dirkjan Ochtman
|
r6393 | |||
Gregory Szorc
|
r24076 | @webcommand('file') | ||
Dirkjan Ochtman
|
r5600 | def file(web, req, tmpl): | ||
Gregory Szorc
|
r24088 | """ | ||
/file/{revision}[/{path}] | ||||
------------------------- | ||||
Show information about a directory or file in the repository. | ||||
Info about the ``path`` given as a URL parameter will be rendered. | ||||
If ``path`` is a directory, information about the entries in that | ||||
directory will be rendered. This form is equivalent to the ``manifest`` | ||||
handler. | ||||
If ``path`` is a file, information about that file will be shown via | ||||
the ``filerevision`` template. | ||||
If ``path`` is not defined, information about the root directory will | ||||
be rendered. | ||||
""" | ||||
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: | ||
r25602 | return _filerevision(web, req, tmpl, webutil.filectx(web.repo, req)) | |||
Gregory Szorc
|
r25660 | except error.LookupError as 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
|
r10247 | def _search(web, req, tmpl): | ||
Alexander Plavin
|
r19656 | MODE_REVISION = 'rev' | ||
MODE_KEYWORD = 'keyword' | ||||
Alexander Plavin
|
r19722 | MODE_REVSET = 'revset' | ||
Dirkjan Ochtman
|
r10247 | |||
Alexander Plavin
|
r19633 | def revsearch(ctx): | ||
yield ctx | ||||
Alexander Plavin
|
r19632 | def keywordsearch(query): | ||
FUJIWARA Katsunori
|
r15727 | lower = encoding.lower | ||
qw = lower(query).split() | ||||
Dirkjan Ochtman
|
r6393 | |||
def revgen(): | ||||
Pierre-Yves David
|
r18497 | cl = web.repo.changelog | ||
Patrick Mezard
|
r12059 | for i in xrange(len(web.repo) - 1, 0, -100): | ||
Dirkjan Ochtman
|
r6393 | l = [] | ||
Alexander Plavin
|
r19491 | for j in cl.revs(max(0, i - 99), i): | ||
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: | ||||
FUJIWARA Katsunori
|
r15727 | if not (q in lower(ctx.user()) or | ||
q in lower(ctx.description()) or | ||||
q in lower(" ".join(ctx.files()))): | ||||
Dirkjan Ochtman
|
r6393 | miss = 1 | ||
break | ||||
if miss: | ||||
continue | ||||
Alexander Plavin
|
r19533 | yield ctx | ||
Alexander Plavin
|
r19722 | def revsetsearch(revs): | ||
for r in revs: | ||||
yield web.repo[r] | ||||
Alexander Plavin
|
r19631 | searchfuncs = { | ||
Wagner Bruna
|
r20004 | MODE_REVISION: (revsearch, 'exact revision search'), | ||
MODE_KEYWORD: (keywordsearch, 'literal keyword search'), | ||||
MODE_REVSET: (revsetsearch, 'revset expression search'), | ||||
Alexander Plavin
|
r19631 | } | ||
Alexander Plavin
|
r19632 | def getsearchmode(query): | ||
Alexander Plavin
|
r19633 | try: | ||
ctx = web.repo[query] | ||||
except (error.RepoError, error.LookupError): | ||||
Alexander Plavin
|
r19722 | # query is not an exact revision pointer, need to | ||
Mads Kiilerich
|
r19951 | # decide if it's a revset expression or keywords | ||
Alexander Plavin
|
r19722 | pass | ||
Alexander Plavin
|
r19633 | else: | ||
Alexander Plavin
|
r19656 | return MODE_REVISION, ctx | ||
Alexander Plavin
|
r19631 | |||
Alexander Plavin
|
r19722 | revdef = 'reverse(%s)' % query | ||
try: | ||||
Yuya Nishihara
|
r31024 | tree = revsetlang.parse(revdef) | ||
Yuya Nishihara
|
r27009 | except error.ParseError: | ||
Alexander Plavin
|
r19722 | # can't parse to a revset tree | ||
return MODE_KEYWORD, query | ||||
Yuya Nishihara
|
r31024 | if revsetlang.depth(tree) <= 2: | ||
Alexander Plavin
|
r19722 | # no revset syntax used | ||
return MODE_KEYWORD, query | ||||
Augie Fackler
|
r25149 | if any((token, (value or '')[:3]) == ('string', 're:') | ||
Yuya Nishihara
|
r31024 | for token, value, pos in revsetlang.tokenize(revdef)): | ||
Alexander Plavin
|
r19722 | return MODE_KEYWORD, query | ||
Yuya Nishihara
|
r31024 | funcsused = revsetlang.funcsused(tree) | ||
Alexander Plavin
|
r19722 | if not funcsused.issubset(revset.safesymbols): | ||
return MODE_KEYWORD, query | ||||
Gregory Szorc
|
r33554 | mfunc = revset.match(web.repo.ui, revdef, repo=web.repo) | ||
Alexander Plavin
|
r19722 | try: | ||
Yuya Nishihara
|
r24114 | revs = mfunc(web.repo) | ||
Alexander Plavin
|
r19722 | return MODE_REVSET, revs | ||
# ParseError: wrongly placed tokens, wrongs arguments, etc | ||||
# RepoLookupError: no such revision, e.g. in 'revision:' | ||||
# Abort: bookmark/tag not exists | ||||
# LookupError: ambiguous identifier, e.g. in '(bc)' on a large repo | ||||
Yuya Nishihara
|
r27009 | except (error.ParseError, error.RepoLookupError, error.Abort, | ||
LookupError): | ||||
Alexander Plavin
|
r19722 | return MODE_KEYWORD, query | ||
Alexander Plavin
|
r19533 | def changelist(**map): | ||
count = 0 | ||||
Alexander Plavin
|
r19765 | for ctx in searchfunc[0](funcarg): | ||
Andrew Beekhof
|
r6659 | count += 1 | ||
Dirkjan Ochtman
|
r6393 | n = ctx.node() | ||
showtags = webutil.showtag(web.repo, tmpl, 'changelogtag', n) | ||||
Dirkjan Ochtman
|
r7311 | files = webutil.listfilediffs(tmpl, ctx.files(), n, web.maxfiles) | ||
Dirkjan Ochtman
|
r6393 | |||
yield tmpl('searchentry', | ||||
timeless
|
r29216 | parity=next(parity), | ||
Dirkjan Ochtman
|
r6393 | changelogtag=showtags, | ||
Dirkjan Ochtman
|
r7311 | files=files, | ||
Pulkit Goyal
|
r36452 | **pycompat.strkwargs(webutil.commonentry(web.repo, ctx))) | ||
Dirkjan Ochtman
|
r6393 | |||
Dirkjan Ochtman
|
r10247 | if count >= revcount: | ||
Dirkjan Ochtman
|
r6393 | break | ||
Alexander Plavin
|
r19418 | query = req.form['rev'][0] | ||
revcount = web.maxchanges | ||||
if 'revcount' in req.form: | ||||
Isaac Jurado
|
r20092 | try: | ||
revcount = int(req.form.get('revcount', [revcount])[0]) | ||||
revcount = max(revcount, 1) | ||||
tmpl.defaults['sessionvars']['revcount'] = revcount | ||||
except ValueError: | ||||
pass | ||||
Alexander Plavin
|
r19418 | |||
lessvars = copy.copy(tmpl.defaults['sessionvars']) | ||||
Augie Fackler
|
r36571 | lessvars['revcount'] = max(revcount // 2, 1) | ||
Alexander Plavin
|
r19418 | lessvars['rev'] = query | ||
morevars = copy.copy(tmpl.defaults['sessionvars']) | ||||
morevars['revcount'] = revcount * 2 | ||||
morevars['rev'] = query | ||||
Alexander Plavin
|
r19632 | mode, funcarg = getsearchmode(query) | ||
Alexander Plavin
|
r19768 | |||
if 'forcekw' in req.form: | ||||
showforcekw = '' | ||||
showunforcekw = searchfuncs[mode][1] | ||||
mode = MODE_KEYWORD | ||||
funcarg = query | ||||
else: | ||||
if mode != MODE_KEYWORD: | ||||
showforcekw = searchfuncs[MODE_KEYWORD][1] | ||||
else: | ||||
showforcekw = '' | ||||
showunforcekw = '' | ||||
Alexander Plavin
|
r19631 | searchfunc = searchfuncs[mode] | ||
Patrick Mezard
|
r12059 | tip = web.repo['tip'] | ||
Dirkjan Ochtman
|
r6393 | parity = paritygen(web.stripecount) | ||
r25602 | return tmpl('search', query=query, node=tip.hex(), symrev='tip', | |||
Dirkjan Ochtman
|
r10247 | entries=changelist, archives=web.archivelist("tip"), | ||
Alexander Plavin
|
r19765 | morevars=morevars, lessvars=lessvars, | ||
Alexander Plavin
|
r19768 | modedesc=searchfunc[1], | ||
showforcekw=showforcekw, showunforcekw=showunforcekw) | ||||
Dirkjan Ochtman
|
r6393 | |||
Gregory Szorc
|
r24076 | @webcommand('changelog') | ||
Dirkjan Ochtman
|
r10247 | def changelog(web, req, tmpl, shortlog=False): | ||
Gregory Szorc
|
r24089 | """ | ||
/changelog[/{revision}] | ||||
----------------------- | ||||
Show information about multiple changesets. | ||||
If the optional ``revision`` URL argument is absent, information about | ||||
all changesets starting at ``tip`` will be rendered. If the ``revision`` | ||||
argument is present, changesets will be shown starting from the specified | ||||
revision. | ||||
If ``revision`` is absent, the ``rev`` query string argument may be | ||||
defined. This will perform a search for changesets. | ||||
The argument for ``rev`` can be a single revision, a revision set, | ||||
or a literal keyword to search for in changeset data (equivalent to | ||||
Wagner Bruna
|
r24867 | :hg:`log -k`). | ||
Gregory Szorc
|
r24089 | |||
The ``revcount`` query string argument defines the maximum numbers of | ||||
changesets to render. | ||||
For non-searches, the ``changelog`` template will be rendered. | ||||
""" | ||||
Dirkjan Ochtman
|
r10247 | |||
Alexander Plavin
|
r19396 | query = '' | ||
Christian Ebert
|
r5915 | if 'node' in req.form: | ||
Dirkjan Ochtman
|
r6392 | ctx = webutil.changectx(web.repo, req) | ||
r25602 | symrev = webutil.symrevorshortnode(req, ctx) | |||
Alexander Plavin
|
r19534 | elif 'rev' in req.form: | ||
Alexander Plavin
|
r19634 | return _search(web, req, tmpl) | ||
Dirkjan Ochtman
|
r5591 | else: | ||
Alexander Plavin
|
r19534 | ctx = web.repo['tip'] | ||
r25602 | symrev = 'tip' | |||
Dirkjan Ochtman
|
r6393 | |||
Alexander Plavin
|
r19737 | def changelist(): | ||
Pierre-Yves David
|
r18427 | revs = [] | ||
Alexander Plavin
|
r19486 | if pos != -1: | ||
revs = web.repo.changelog.revs(pos, 0) | ||||
curcount = 0 | ||||
Gregory Szorc
|
r23745 | for rev in revs: | ||
Alexander Plavin
|
r19486 | curcount += 1 | ||
Alexander Plavin
|
r19738 | if curcount > revcount + 1: | ||
Alexander Plavin
|
r19486 | break | ||
Gregory Szorc
|
r23745 | |||
entry = webutil.changelistentry(web, web.repo[rev], tmpl) | ||||
timeless
|
r29216 | entry['parity'] = next(parity) | ||
Gregory Szorc
|
r23745 | yield entry | ||
Dirkjan Ochtman
|
r6393 | |||
Jordi Gutiérrez Hermoso
|
r24306 | if shortlog: | ||
revcount = web.maxshortchanges | ||||
else: | ||||
revcount = web.maxchanges | ||||
Dirkjan Ochtman
|
r10246 | if 'revcount' in req.form: | ||
Isaac Jurado
|
r20092 | try: | ||
revcount = int(req.form.get('revcount', [revcount])[0]) | ||||
revcount = max(revcount, 1) | ||||
tmpl.defaults['sessionvars']['revcount'] = revcount | ||||
except ValueError: | ||||
pass | ||||
Dirkjan Ochtman
|
r10246 | |||
lessvars = copy.copy(tmpl.defaults['sessionvars']) | ||||
Augie Fackler
|
r36571 | lessvars['revcount'] = max(revcount // 2, 1) | ||
Dirkjan Ochtman
|
r10246 | morevars = copy.copy(tmpl.defaults['sessionvars']) | ||
morevars['revcount'] = revcount * 2 | ||||
Patrick Mezard
|
r12059 | count = len(web.repo) | ||
Dirkjan Ochtman
|
r6393 | pos = ctx.rev() | ||
Alexander Plavin
|
r19486 | parity = paritygen(web.stripecount) | ||
Dirkjan Ochtman
|
r6393 | |||
Pierre-Yves David
|
r18409 | changenav = webutil.revnav(web.repo).gen(pos, revcount, count) | ||
Dirkjan Ochtman
|
r6393 | |||
Alexander Plavin
|
r19737 | entries = list(changelist()) | ||
latestentry = entries[:1] | ||||
Alexander Plavin
|
r19738 | if len(entries) > revcount: | ||
nextentry = entries[-1:] | ||||
entries = entries[:-1] | ||||
else: | ||||
nextentry = [] | ||||
Alexander Plavin
|
r19737 | |||
r35315 | return tmpl('shortlog' if shortlog else 'changelog', changenav=changenav, | |||
r25602 | node=ctx.hex(), rev=pos, symrev=symrev, changesets=count, | |||
Alexander Plavin
|
r19737 | entries=entries, | ||
Alexander Plavin
|
r19738 | latestentry=latestentry, nextentry=nextentry, | ||
Dirkjan Ochtman
|
r10246 | archives=web.archivelist("tip"), revcount=revcount, | ||
Alexander Plavin
|
r19396 | morevars=morevars, lessvars=lessvars, query=query) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('shortlog') | ||
Dirkjan Ochtman
|
r5600 | def shortlog(web, req, tmpl): | ||
Gregory Szorc
|
r24086 | """ | ||
/shortlog | ||||
--------- | ||||
Show basic information about a set of changesets. | ||||
This accepts the same parameters as the ``changelog`` handler. The only | ||||
difference is the ``shortlog`` template will be rendered instead of the | ||||
``changelog`` template. | ||||
""" | ||||
Mads Kiilerich
|
r19872 | return changelog(web, req, tmpl, shortlog=True) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('changeset') | ||
Dirkjan Ochtman
|
r5600 | def changeset(web, req, tmpl): | ||
Gregory Szorc
|
r24085 | """ | ||
/changeset[/{revision}] | ||||
----------------------- | ||||
Show information about a single changeset. | ||||
A URL path argument is the changeset identifier to show. See ``hg help | ||||
revisions`` for possible values. If not defined, the ``tip`` changeset | ||||
will be shown. | ||||
The ``changeset`` template is rendered. Contents of the ``changesettag``, | ||||
``changesetbookmark``, ``filenodelink``, ``filenolink``, and the many | ||||
templates related to diffs may all be used to produce the output. | ||||
""" | ||||
Dirkjan Ochtman
|
r6393 | ctx = webutil.changectx(web.repo, req) | ||
Gregory Szorc
|
r24177 | return tmpl('changeset', **webutil.changesetentry(web, req, tmpl, ctx)) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | rev = webcommand('rev')(changeset) | ||
Dirkjan Ochtman
|
r5591 | |||
Martin Geisler
|
r16448 | def decodepath(path): | ||
"""Hook for mapping a path in the repository to a path in the | ||||
working copy. | ||||
Extensions (e.g., largefiles) can override this to remap files in | ||||
the virtual file system presented by the manifest command below.""" | ||||
return path | ||||
Gregory Szorc
|
r24076 | @webcommand('manifest') | ||
Dirkjan Ochtman
|
r5600 | def manifest(web, req, tmpl): | ||
Gregory Szorc
|
r24090 | """ | ||
/manifest[/{revision}[/{path}]] | ||||
------------------------------- | ||||
Show information about a directory. | ||||
Wagner Bruna
|
r24868 | If the URL path arguments are omitted, information about the root | ||
Gregory Szorc
|
r24090 | directory for the ``tip`` changeset will be shown. | ||
Because this handler can only show information for directories, it | ||||
is recommended to use the ``file`` handler instead, as it can handle both | ||||
directories and files. | ||||
The ``manifest`` template will be rendered for this handler. | ||||
""" | ||||
r25602 | if 'node' in req.form: | |||
ctx = webutil.changectx(web.repo, req) | ||||
symrev = webutil.symrevorshortnode(req, ctx) | ||||
else: | ||||
ctx = web.repo['tip'] | ||||
symrev = 'tip' | ||||
Dirkjan Ochtman
|
r6393 | 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) | ||
Augie Fackler
|
r36731 | if path and path[-1:] != "/": | ||
Dirkjan Ochtman
|
r6393 | path += "/" | ||
l = len(path) | ||||
abspath = "/" + path | ||||
Martin Geisler
|
r16448 | for full, n in mf.iteritems(): | ||
# the virtual path (working copy path) used for the full | ||||
# (repository) path | ||||
f = decodepath(full) | ||||
Dirkjan Ochtman
|
r6393 | if f[:l] != path: | ||
continue | ||||
remain = f[l:] | ||||
Ry4an Brase
|
r7305 | elements = remain.split('/') | ||
if len(elements) == 1: | ||||
Martin Geisler
|
r16448 | files[remain] = full | ||
Ry4an Brase
|
r7305 | 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 | |||
Dirkjan Ochtman
|
r7565 | if mf and not files and not dirs: | ||
Dirkjan Ochtman
|
r6393 | raise ErrorResponse(HTTP_NOT_FOUND, 'path not found: ' + path) | ||
def filelist(**map): | ||||
Matt Mackall
|
r8209 | for f in sorted(files): | ||
Ry4an Brase
|
r7305 | full = files[f] | ||
Dirkjan Ochtman
|
r6393 | |||
fctx = ctx.filectx(full) | ||||
yield {"file": full, | ||||
timeless
|
r29216 | "parity": next(parity), | ||
Dirkjan Ochtman
|
r6393 | "basename": f, | ||
Matt Mackall
|
r6747 | "date": fctx.date(), | ||
Dirkjan Ochtman
|
r6393 | "size": fctx.size(), | ||
"permissions": mf.flags(full)} | ||||
def dirlist(**map): | ||||
Matt Mackall
|
r8209 | for d in sorted(dirs): | ||
Dirkjan Ochtman
|
r6393 | |||
Ry4an Brase
|
r7305 | emptydirs = [] | ||
h = dirs[d] | ||||
while isinstance(h, dict) and len(h) == 1: | ||||
Augie Fackler
|
r36288 | k, v = next(iter(h.items())) | ||
Ry4an Brase
|
r7305 | if v: | ||
emptydirs.append(k) | ||||
h = v | ||||
path = "%s%s" % (abspath, d) | ||||
timeless
|
r29216 | yield {"parity": next(parity), | ||
Ry4an Brase
|
r7305 | "path": path, | ||
"emptydirs": "/".join(emptydirs), | ||||
"basename": d} | ||||
Dirkjan Ochtman
|
r6393 | |||
return tmpl("manifest", | ||||
r25602 | symrev=symrev, | |||
Dirkjan Ochtman
|
r6393 | path=abspath, | ||
up=webutil.up(abspath), | ||||
timeless
|
r29216 | upparity=next(parity), | ||
Dirkjan Ochtman
|
r6393 | fentries=filelist, | ||
dentries=dirlist, | ||||
archives=web.archivelist(hex(node)), | ||||
Augie Fackler
|
r36289 | **pycompat.strkwargs(webutil.commonentry(web.repo, ctx))) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('tags') | ||
Dirkjan Ochtman
|
r5600 | def tags(web, req, tmpl): | ||
Gregory Szorc
|
r24084 | """ | ||
/tags | ||||
----- | ||||
Show information about tags. | ||||
No arguments are accepted. | ||||
The ``tags`` template is rendered. | ||||
""" | ||||
Matt Mackall
|
r18029 | i = list(reversed(web.repo.tagslist())) | ||
Dirkjan Ochtman
|
r6393 | parity = paritygen(web.stripecount) | ||
Pierre-Yves David
|
r18402 | def entries(notip, latestonly, **map): | ||
t = i | ||||
if notip: | ||||
t = [(k, n) for k, n in i if k != "tip"] | ||||
if latestonly: | ||||
t = t[:1] | ||||
for k, n in t: | ||||
timeless
|
r29216 | yield {"parity": next(parity), | ||
Dirkjan Ochtman
|
r6393 | "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()), | ||||
Pierre-Yves David
|
r18402 | entries=lambda **x: entries(False, False, **x), | ||
entriesnotip=lambda **x: entries(True, False, **x), | ||||
latestentry=lambda **x: entries(True, True, **x)) | ||||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('bookmarks') | ||
Alexander Solovyov
|
r13597 | def bookmarks(web, req, tmpl): | ||
Gregory Szorc
|
r24083 | """ | ||
/bookmarks | ||||
---------- | ||||
Show information about bookmarks. | ||||
No arguments are accepted. | ||||
The ``bookmarks`` template is rendered. | ||||
""" | ||||
Kevin Bullock
|
r18478 | i = [b for b in web.repo._bookmarks.items() if b[1] in web.repo] | ||
r28711 | sortkey = lambda b: (web.repo[b[1]].rev(), b[0]) | |||
i = sorted(i, key=sortkey, reverse=True) | ||||
Alexander Solovyov
|
r13597 | parity = paritygen(web.stripecount) | ||
Pierre-Yves David
|
r18402 | def entries(latestonly, **map): | ||
r28710 | t = i | |||
Pierre-Yves David
|
r18402 | if latestonly: | ||
r28710 | t = i[:1] | |||
Pierre-Yves David
|
r18402 | for k, n in t: | ||
timeless
|
r29216 | yield {"parity": next(parity), | ||
Alexander Solovyov
|
r13597 | "bookmark": k, | ||
"date": web.repo[n].date(), | ||||
"node": hex(n)} | ||||
r28712 | if i: | |||
latestrev = i[0][1] | ||||
else: | ||||
latestrev = -1 | ||||
Alexander Solovyov
|
r13597 | return tmpl("bookmarks", | ||
node=hex(web.repo.changelog.tip()), | ||||
r28712 | lastchange=[{"date": web.repo[latestrev].date()}], | |||
Pierre-Yves David
|
r18402 | entries=lambda **x: entries(latestonly=False, **x), | ||
latestentry=lambda **x: entries(latestonly=True, **x)) | ||||
Alexander Solovyov
|
r13597 | |||
Gregory Szorc
|
r24076 | @webcommand('branches') | ||
Sune Foldager
|
r8352 | def branches(web, req, tmpl): | ||
Gregory Szorc
|
r24082 | """ | ||
/branches | ||||
--------- | ||||
Show information about branches. | ||||
All known branches are contained in the output, even closed branches. | ||||
No arguments are accepted. | ||||
The ``branches`` template is rendered. | ||||
""" | ||||
r26129 | entries = webutil.branchentries(web.repo, web.stripecount) | |||
latestentry = webutil.branchentries(web.repo, web.stripecount, 1) | ||||
Sune Foldager
|
r8352 | return tmpl('branches', node=hex(web.repo.changelog.tip()), | ||
r26129 | entries=entries, latestentry=latestentry) | |||
Sune Foldager
|
r8352 | |||
Gregory Szorc
|
r24076 | @webcommand('summary') | ||
Dirkjan Ochtman
|
r5600 | def summary(web, req, tmpl): | ||
Gregory Szorc
|
r24091 | """ | ||
/summary | ||||
-------- | ||||
Show a summary of repository state. | ||||
Information about the latest changesets, bookmarks, tags, and branches | ||||
is captured by this handler. | ||||
The ``summary`` template is rendered. | ||||
""" | ||||
Patrick Mezard
|
r17261 | i = reversed(web.repo.tagslist()) | ||
Dirkjan Ochtman
|
r6393 | |||
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", | ||||
timeless
|
r29216 | parity=next(parity), | ||
Dirkjan Ochtman
|
r6393 | tag=k, | ||
node=hex(n), | ||||
Matt Mackall
|
r6747 | date=web.repo[n].date()) | ||
Dirkjan Ochtman
|
r6393 | |||
Yuya Nishihara
|
r13924 | def bookmarks(**map): | ||
parity = paritygen(web.stripecount) | ||||
Kevin Bullock
|
r18563 | marks = [b for b in web.repo._bookmarks.items() if b[1] in web.repo] | ||
r28711 | sortkey = lambda b: (web.repo[b[1]].rev(), b[0]) | |||
marks = sorted(marks, key=sortkey, reverse=True) | ||||
for k, n in marks[:10]: # limit to 10 bookmarks | ||||
timeless
|
r29216 | yield {'parity': next(parity), | ||
Yuya Nishihara
|
r13924 | 'bookmark': k, | ||
'date': web.repo[n].date(), | ||||
'node': hex(n)} | ||||
Dirkjan Ochtman
|
r6393 | def changelist(**map): | ||
Matt Mackall
|
r10282 | parity = paritygen(web.stripecount, offset=start - end) | ||
Dirkjan Ochtman
|
r6393 | l = [] # build a list in forward order for efficiency | ||
Kevin Bullock
|
r18563 | revs = [] | ||
if start < end: | ||||
revs = web.repo.changelog.revs(start, end - 1) | ||||
for i in revs: | ||||
Matt Mackall
|
r6747 | ctx = web.repo[i] | ||
Dirkjan Ochtman
|
r6393 | |||
Pierre-Yves David
|
r18319 | l.append(tmpl( | ||
r27294 | 'shortlogentry', | |||
timeless
|
r29216 | parity=next(parity), | ||
Pulkit Goyal
|
r36452 | **pycompat.strkwargs(webutil.commonentry(web.repo, ctx)))) | ||
Dirkjan Ochtman
|
r6393 | |||
Laura Médioni
|
r29382 | for entry in reversed(l): | ||
yield entry | ||||
Dirkjan Ochtman
|
r6393 | |||
Patrick Mezard
|
r12059 | tip = web.repo['tip'] | ||
count = len(web.repo) | ||||
Dirkjan Ochtman
|
r6393 | start = max(0, count - web.maxchanges) | ||
end = min(count, start + web.maxchanges) | ||||
Boris Feld
|
r34235 | desc = web.config("web", "description") | ||
if not desc: | ||||
desc = 'unknown' | ||||
Dirkjan Ochtman
|
r6393 | return tmpl("summary", | ||
Boris Feld
|
r34235 | desc=desc, | ||
Dirkjan Ochtman
|
r6393 | owner=get_contact(web.config) or "unknown", | ||
Patrick Mezard
|
r12059 | lastchange=tip.date(), | ||
Dirkjan Ochtman
|
r6393 | tags=tagentries, | ||
Yuya Nishihara
|
r13924 | bookmarks=bookmarks, | ||
r26131 | branches=webutil.branchentries(web.repo, web.stripecount, 10), | |||
Dirkjan Ochtman
|
r6393 | shortlog=changelist, | ||
Patrick Mezard
|
r12059 | node=tip.hex(), | ||
r25602 | symrev='tip', | |||
Gregory Szorc
|
r29471 | archives=web.archivelist("tip"), | ||
labels=web.configlist('web', 'labels')) | ||||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('filediff') | ||
Dirkjan Ochtman
|
r5600 | def filediff(web, req, tmpl): | ||
Gregory Szorc
|
r24092 | """ | ||
/diff/{revision}/{path} | ||||
----------------------- | ||||
Show how a file changed in a particular commit. | ||||
The ``filediff`` template is rendered. | ||||
Mads Kiilerich
|
r26781 | This handler is registered under both the ``/diff`` and ``/filediff`` | ||
Gregory Szorc
|
r24092 | paths. ``/diff`` is used in modern code. | ||
""" | ||||
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: | ||||
path = fctx.path() | ||||
Matt Mackall
|
r16722 | ctx = fctx.changectx() | ||
Denis Laxalde
|
r31082 | basectx = ctx.p1() | ||
Dirkjan Ochtman
|
r6393 | |||
Boris Feld
|
r34243 | style = web.config('web', 'style') | ||
Dirkjan Ochtman
|
r9402 | if 'style' in req.form: | ||
style = req.form['style'][0] | ||||
Denis Laxalde
|
r31660 | diffs = webutil.diffs(web, tmpl, ctx, basectx, [path], style) | ||
r27160 | if fctx is not None: | |||
Jordi Gutiérrez Hermoso
|
r24306 | rename = webutil.renamelink(fctx) | ||
ctx = fctx | ||||
else: | ||||
rename = [] | ||||
ctx = ctx | ||||
Dirkjan Ochtman
|
r6393 | return tmpl("filediff", | ||
file=path, | ||||
r25602 | symrev=webutil.symrevorshortnode(req, ctx), | |||
Dirkjan Ochtman
|
r7183 | rename=rename, | ||
r27294 | diff=diffs, | |||
Pulkit Goyal
|
r36452 | **pycompat.strkwargs(webutil.commonentry(web.repo, ctx))) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | diff = webcommand('diff')(filediff) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('comparison') | ||
wujek srujek
|
r17202 | def comparison(web, req, tmpl): | ||
Gregory Szorc
|
r24093 | """ | ||
/comparison/{revision}/{path} | ||||
----------------------------- | ||||
Show a comparison between the old and new versions of a file from changes | ||||
made on a particular revision. | ||||
This is similar to the ``diff`` handler. However, this form features | ||||
a split or side-by-side diff rather than a unified diff. | ||||
The ``context`` query string argument can be used to control the lines of | ||||
context in the diff. | ||||
The ``filecomparison`` template is rendered. | ||||
""" | ||||
wujek srujek
|
r17202 | ctx = webutil.changectx(web.repo, req) | ||
Ross Lagerwall
|
r17289 | if 'file' not in req.form: | ||
raise ErrorResponse(HTTP_NOT_FOUND, 'file not given') | ||||
wujek srujek
|
r17202 | path = webutil.cleanpath(web.repo, req.form['file'][0]) | ||
parsecontext = lambda v: v == 'full' and -1 or int(v) | ||||
if 'context' in req.form: | ||||
context = parsecontext(req.form['context'][0]) | ||||
else: | ||||
context = parsecontext(web.config('web', 'comparisoncontext', '5')) | ||||
wujek srujek
|
r17302 | def filelines(f): | ||
Jun Wu
|
r32136 | if f.isbinary(): | ||
wujek srujek
|
r17302 | mt = mimetypes.guess_type(f.path())[0] | ||
if not mt: | ||||
mt = 'application/octet-stream' | ||||
return [_('(binary file %s, hash: %s)') % (mt, hex(f.filenode()))] | ||||
return f.data().splitlines() | ||||
r27158 | fctx = None | |||
FUJIWARA Katsunori
|
r21123 | parent = ctx.p1() | ||
leftrev = parent.rev() | ||||
leftnode = parent.node() | ||||
rightrev = ctx.rev() | ||||
rightnode = ctx.node() | ||||
wujek srujek
|
r17302 | if path in ctx: | ||
fctx = ctx[path] | ||||
rightlines = filelines(fctx) | ||||
FUJIWARA Katsunori
|
r21121 | if path not in parent: | ||
wujek srujek
|
r17302 | leftlines = () | ||
else: | ||||
FUJIWARA Katsunori
|
r21121 | pfctx = parent[path] | ||
wujek srujek
|
r17302 | leftlines = filelines(pfctx) | ||
else: | ||||
rightlines = () | ||||
r27158 | pfctx = ctx.parents()[0][path] | |||
leftlines = filelines(pfctx) | ||||
wujek srujek
|
r17302 | |||
comparison = webutil.compare(tmpl, context, leftlines, rightlines) | ||||
r27158 | if fctx is not None: | |||
r27159 | rename = webutil.renamelink(fctx) | |||
r27158 | ctx = fctx | |||
else: | ||||
r27159 | rename = [] | |||
r27158 | ctx = ctx | |||
wujek srujek
|
r17202 | return tmpl('filecomparison', | ||
file=path, | ||||
r25602 | symrev=webutil.symrevorshortnode(req, ctx), | |||
wujek srujek
|
r17202 | rename=rename, | ||
wujek srujek
|
r17302 | leftrev=leftrev, | ||
leftnode=hex(leftnode), | ||||
rightrev=rightrev, | ||||
rightnode=hex(rightnode), | ||||
r27294 | comparison=comparison, | |||
Pulkit Goyal
|
r36452 | **pycompat.strkwargs(webutil.commonentry(web.repo, ctx))) | ||
wujek srujek
|
r17202 | |||
Gregory Szorc
|
r24076 | @webcommand('annotate') | ||
Dirkjan Ochtman
|
r5600 | def annotate(web, req, tmpl): | ||
Gregory Szorc
|
r24094 | """ | ||
/annotate/{revision}/{path} | ||||
--------------------------- | ||||
Show changeset information for each line in a file. | ||||
Gregory Szorc
|
r34391 | The ``ignorews``, ``ignorewsamount``, ``ignorewseol``, and | ||
``ignoreblanklines`` query string arguments have the same meaning as | ||||
Gregory Szorc
|
r34404 | their ``[annotate]`` config equivalents. It uses the hgrc boolean | ||
parsing logic to interpret the value. e.g. ``0`` and ``false`` are | ||||
false and ``1`` and ``true`` are true. If not defined, the server | ||||
default settings are used. | ||||
Gregory Szorc
|
r34391 | |||
Gregory Szorc
|
r24094 | The ``fileannotate`` template is rendered. | ||
""" | ||||
Dirkjan Ochtman
|
r6393 | fctx = webutil.filectx(web.repo, req) | ||
f = fctx.path() | ||||
parity = paritygen(web.stripecount) | ||||
Denis Laxalde
|
r32994 | ishead = fctx.filerev() in fctx.filelog().headrevs() | ||
Dirkjan Ochtman
|
r6393 | |||
Gregory Szorc
|
r30298 | # parents() is called once per line and several lines likely belong to | ||
# same revision. So it is worth caching. | ||||
# TODO there are still redundant operations within basefilectx.parents() | ||||
# and from the fctx.annotate() call itself that could be cached. | ||||
parentscache = {} | ||||
Denis Laxalde
|
r29522 | def parents(f): | ||
Gregory Szorc
|
r30298 | rev = f.rev() | ||
if rev not in parentscache: | ||||
parentscache[rev] = [] | ||||
for p in f.parents(): | ||||
entry = { | ||||
'node': p.hex(), | ||||
'rev': p.rev(), | ||||
} | ||||
parentscache[rev].append(entry) | ||||
for p in parentscache[rev]: | ||||
yield p | ||||
Denis Laxalde
|
r29522 | |||
Dirkjan Ochtman
|
r6393 | def annotate(**map): | ||
Jun Wu
|
r32136 | if fctx.isbinary(): | ||
Dirkjan Ochtman
|
r6393 | mt = (mimetypes.guess_type(fctx.path())[0] | ||
or 'application/octet-stream') | ||||
r29538 | lines = [((fctx.filectx(fctx.filerev()), 1), '(binary:%s)' % mt)] | |||
Dirkjan Ochtman
|
r6393 | else: | ||
Gregory Szorc
|
r34391 | lines = webutil.annotate(req, fctx, web.repo.ui) | ||
Jun Wu
|
r30081 | |||
Denis Laxalde
|
r29388 | previousrev = None | ||
r29572 | blockparitygen = paritygen(1) | |||
Siddharth Agarwal
|
r34433 | for lineno, (aline, l) in enumerate(lines): | ||
f = aline.fctx | ||||
Denis Laxalde
|
r29388 | rev = f.rev() | ||
r29572 | if rev != previousrev: | |||
blockhead = True | ||||
blockparity = next(blockparitygen) | ||||
else: | ||||
blockhead = None | ||||
Denis Laxalde
|
r29388 | previousrev = rev | ||
timeless
|
r29216 | yield {"parity": next(parity), | ||
Alexander Solovyov
|
r14055 | "node": f.hex(), | ||
Denis Laxalde
|
r29388 | "rev": rev, | ||
Patrick Mezard
|
r6564 | "author": f.user(), | ||
Denis Laxalde
|
r29522 | "parents": parents(f), | ||
Dirkjan Ochtman
|
r6657 | "desc": f.description(), | ||
Benoit Boissinot
|
r18581 | "extra": f.extra(), | ||
Dirkjan Ochtman
|
r6393 | "file": f.path(), | ||
Denis Laxalde
|
r29388 | "blockhead": blockhead, | ||
r29572 | "blockparity": blockparity, | |||
Siddharth Agarwal
|
r34433 | "targetline": aline.lineno, | ||
Dirkjan Ochtman
|
r6393 | "line": l, | ||
Gregory Szorc
|
r24712 | "lineno": lineno + 1, | ||
Dirkjan Ochtman
|
r6393 | "lineid": "l%d" % (lineno + 1), | ||
Oli Thissen
|
r13199 | "linenumber": "% 6d" % (lineno + 1), | ||
"revdate": f.date()} | ||||
Dirkjan Ochtman
|
r6393 | |||
Gregory Szorc
|
r34392 | diffopts = webutil.difffeatureopts(req, web.repo.ui, 'annotate') | ||
diffopts = {k: getattr(diffopts, k) for k in diffopts.defaults} | ||||
Dirkjan Ochtman
|
r6393 | return tmpl("fileannotate", | ||
file=f, | ||||
annotate=annotate, | ||||
path=webutil.up(f), | ||||
r25602 | symrev=webutil.symrevorshortnode(req, fctx), | |||
Matt Mackall
|
r6434 | rename=webutil.renamelink(fctx), | ||
r27294 | permissions=fctx.manifest().flags(f), | |||
Denis Laxalde
|
r32994 | ishead=int(ishead), | ||
Gregory Szorc
|
r34392 | diffopts=diffopts, | ||
Pulkit Goyal
|
r36452 | **pycompat.strkwargs(webutil.commonentry(web.repo, fctx))) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('filelog') | ||
Dirkjan Ochtman
|
r5600 | def filelog(web, req, tmpl): | ||
Gregory Szorc
|
r24095 | """ | ||
/filelog/{revision}/{path} | ||||
-------------------------- | ||||
Show information about the history of a file in the repository. | ||||
The ``revcount`` query string argument can be defined to control the | ||||
maximum number of entries to show. | ||||
The ``filelog`` template will be rendered. | ||||
""" | ||||
Dirkjan Ochtman
|
r7300 | |||
try: | ||||
fctx = webutil.filectx(web.repo, req) | ||||
f = fctx.path() | ||||
fl = fctx.filelog() | ||||
Matt Mackall
|
r7633 | except error.LookupError: | ||
Dirkjan Ochtman
|
r7300 | 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() | ||||
Matt Mackall
|
r7361 | first = fl.linkrev(0) | ||
Dirkjan Ochtman
|
r7300 | if rev < first: # current rev is from before file existed | ||
raise | ||||
frev = numrevs - 1 | ||||
Matt Mackall
|
r7361 | while fl.linkrev(frev) > rev: | ||
Dirkjan Ochtman
|
r7300 | frev -= 1 | ||
Matt Mackall
|
r7361 | fctx = web.repo.filectx(f, fl.linkrev(frev)) | ||
Dirkjan Ochtman
|
r7300 | |||
Dirkjan Ochtman
|
r10246 | revcount = web.maxshortchanges | ||
if 'revcount' in req.form: | ||||
Isaac Jurado
|
r20092 | try: | ||
revcount = int(req.form.get('revcount', [revcount])[0]) | ||||
revcount = max(revcount, 1) | ||||
tmpl.defaults['sessionvars']['revcount'] = revcount | ||||
except ValueError: | ||||
pass | ||||
Dirkjan Ochtman
|
r10246 | |||
Denis Laxalde
|
r31665 | lrange = webutil.linerange(req) | ||
Dirkjan Ochtman
|
r10246 | lessvars = copy.copy(tmpl.defaults['sessionvars']) | ||
Augie Fackler
|
r36571 | lessvars['revcount'] = max(revcount // 2, 1) | ||
Dirkjan Ochtman
|
r10246 | morevars = copy.copy(tmpl.defaults['sessionvars']) | ||
morevars['revcount'] = revcount * 2 | ||||
Denis Laxalde
|
r31661 | patch = 'patch' in req.form | ||
if patch: | ||||
lessvars['patch'] = morevars['patch'] = req.form['patch'][0] | ||||
Denis Laxalde
|
r31939 | descend = 'descend' in req.form | ||
if descend: | ||||
lessvars['descend'] = morevars['descend'] = req.form['descend'][0] | ||||
Denis Laxalde
|
r31661 | |||
Dirkjan Ochtman
|
r7300 | count = fctx.filerev() + 1 | ||
Denis Laxalde
|
r30826 | start = max(0, count - revcount) # first rev on this page | ||
Dirkjan Ochtman
|
r10246 | end = min(count, start + revcount) # last rev on this page | ||
Denis Laxalde
|
r30825 | parity = paritygen(web.stripecount, offset=start - end) | ||
Dirkjan Ochtman
|
r6393 | |||
Denis Laxalde
|
r30816 | repo = web.repo | ||
revs = fctx.filelog().revs(start, end - 1) | ||||
entries = [] | ||||
Denis Laxalde
|
r31661 | |||
Boris Feld
|
r34243 | diffstyle = web.config('web', 'style') | ||
Denis Laxalde
|
r31661 | if 'style' in req.form: | ||
diffstyle = req.form['style'][0] | ||||
Denis Laxalde
|
r31667 | def diff(fctx, linerange=None): | ||
Denis Laxalde
|
r31661 | ctx = fctx.changectx() | ||
basectx = ctx.p1() | ||||
path = fctx.path() | ||||
Denis Laxalde
|
r31667 | return webutil.diffs(web, tmpl, ctx, basectx, [path], diffstyle, | ||
Denis Laxalde
|
r31727 | linerange=linerange, | ||
lineidprefix='%s-' % ctx.hex()[:12]) | ||||
Denis Laxalde
|
r31661 | |||
Denis Laxalde
|
r31665 | linerange = None | ||
if lrange is not None: | ||||
linerange = webutil.formatlinerange(*lrange) | ||||
# deactivate numeric nav links when linerange is specified as this | ||||
# would required a dedicated "revnav" class | ||||
nav = None | ||||
Denis Laxalde
|
r31939 | if descend: | ||
Yuya Nishihara
|
r32904 | it = dagop.blockdescendants(fctx, *lrange) | ||
Denis Laxalde
|
r31939 | else: | ||
Yuya Nishihara
|
r32904 | it = dagop.blockancestors(fctx, *lrange) | ||
Denis Laxalde
|
r31939 | for i, (c, lr) in enumerate(it, 1): | ||
Denis Laxalde
|
r31665 | diffs = None | ||
if patch: | ||||
Denis Laxalde
|
r31667 | diffs = diff(c, linerange=lr) | ||
Denis Laxalde
|
r31665 | # follow renames accross filtered (not in range) revisions | ||
path = c.path() | ||||
entries.append(dict( | ||||
parity=next(parity), | ||||
filerev=c.rev(), | ||||
file=path, | ||||
diff=diffs, | ||||
linerange=webutil.formatlinerange(*lr), | ||||
Pulkit Goyal
|
r36452 | **pycompat.strkwargs(webutil.commonentry(repo, c)))) | ||
Denis Laxalde
|
r31665 | if i == revcount: | ||
break | ||||
lessvars['linerange'] = webutil.formatlinerange(*lrange) | ||||
morevars['linerange'] = lessvars['linerange'] | ||||
else: | ||||
for i in revs: | ||||
iterfctx = fctx.filectx(i) | ||||
diffs = None | ||||
if patch: | ||||
diffs = diff(iterfctx) | ||||
entries.append(dict( | ||||
parity=next(parity), | ||||
filerev=i, | ||||
file=f, | ||||
diff=diffs, | ||||
rename=webutil.renamelink(iterfctx), | ||||
Pulkit Goyal
|
r36452 | **pycompat.strkwargs(webutil.commonentry(repo, iterfctx)))) | ||
Denis Laxalde
|
r31665 | entries.reverse() | ||
revnav = webutil.filerevnav(web.repo, fctx.path()) | ||||
nav = revnav.gen(end - 1, revcount, count) | ||||
Dirkjan Ochtman
|
r6393 | |||
Alexander Plavin
|
r20022 | latestentry = entries[:1] | ||
r27294 | return tmpl("filelog", | |||
file=f, | ||||
nav=nav, | ||||
r25602 | symrev=webutil.symrevorshortnode(req, fctx), | |||
Alexander Plavin
|
r20022 | entries=entries, | ||
Denis Laxalde
|
r31939 | descend=descend, | ||
Denis Laxalde
|
r31661 | patch=patch, | ||
Alexander Plavin
|
r20022 | latestentry=latestentry, | ||
Denis Laxalde
|
r31665 | linerange=linerange, | ||
r27294 | revcount=revcount, | |||
morevars=morevars, | ||||
lessvars=lessvars, | ||||
Pulkit Goyal
|
r36452 | **pycompat.strkwargs(webutil.commonentry(web.repo, fctx))) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('archive') | ||
Dirkjan Ochtman
|
r5600 | def archive(web, req, tmpl): | ||
Gregory Szorc
|
r24096 | """ | ||
/archive/{revision}.{format}[/{path}] | ||||
------------------------------------- | ||||
Obtain an archive of repository content. | ||||
The content and type of the archive is defined by a URL path parameter. | ||||
``format`` is the file extension of the archive type to be generated. e.g. | ||||
``zip`` or ``tar.bz2``. Not all archive types may be allowed by your | ||||
server configuration. | ||||
The optional ``path`` URL parameter controls content to include in the | ||||
archive. If omitted, every file in the specified revision is present in the | ||||
archive. If included, only the specified file or contents of the specified | ||||
directory will be included in the archive. | ||||
No template is used for this handler. Raw, binary content is generated. | ||||
""" | ||||
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] | ||
r30734 | if type_ not in web.archivespecs: | |||
Dirkjan Ochtman
|
r6393 | msg = 'Unsupported archive type: %s' % type_ | ||
raise ErrorResponse(HTTP_NOT_FOUND, msg) | ||||
Rocco Rutte
|
r7029 | if not ((type_ in allowed or | ||
Yuya Nishihara
|
r34655 | web.configbool("web", "allow" + type_))): | ||
Rocco Rutte
|
r7029 | msg = 'Archive type not allowed: %s' % type_ | ||
raise ErrorResponse(HTTP_FORBIDDEN, msg) | ||||
Pulkit Goyal
|
r36402 | reponame = re.sub(br"\W+", "-", os.path.basename(web.reponame)) | ||
Dirkjan Ochtman
|
r6393 | cnode = web.repo.lookup(key) | ||
arch_version = key | ||||
if cnode == key or key == 'tip': | ||||
arch_version = short(cnode) | ||||
name = "%s-%s" % (reponame, arch_version) | ||||
Angel Ezquerra
|
r18771 | |||
ctx = webutil.changectx(web.repo, req) | ||||
pats = [] | ||||
Martin von Zweigbergk
|
r34085 | match = scmutil.match(ctx, []) | ||
Angel Ezquerra
|
r18771 | file = req.form.get('file', None) | ||
if file: | ||||
Angel Ezquerra
|
r18968 | pats = ['path:' + file[0]] | ||
Martin von Zweigbergk
|
r34085 | match = scmutil.match(ctx, pats, default='path') | ||
Angel Ezquerra
|
r18968 | if pats: | ||
Martin von Zweigbergk
|
r34085 | files = [f for f in ctx.manifest().keys() if match(f)] | ||
Angel Ezquerra
|
r18968 | if not files: | ||
raise ErrorResponse(HTTP_NOT_FOUND, | ||||
'file(s) not found: %s' % file[0]) | ||||
Angel Ezquerra
|
r18771 | |||
Gregory Szorc
|
r26136 | mimetype, artype, extension, encoding = web.archivespecs[type_] | ||
Dirkjan Ochtman
|
r6393 | headers = [ | ||
('Content-Disposition', 'attachment; filename=%s%s' % (name, extension)) | ||||
Mads Kiilerich
|
r18347 | ] | ||
Dirkjan Ochtman
|
r6393 | if encoding: | ||
headers.append(('Content-Encoding', encoding)) | ||||
Mads Kiilerich
|
r18348 | req.headers.extend(headers) | ||
Mads Kiilerich
|
r18347 | req.respond(HTTP_OK, mimetype) | ||
Jordi Gutiérrez Hermoso
|
r17933 | |||
archival.archive(web.repo, req, cnode, artype, prefix=name, | ||||
Martin von Zweigbergk
|
r34085 | matchfn=match, | ||
Jordi Gutiérrez Hermoso
|
r17933 | subrepos=web.configbool("web", "archivesubrepos")) | ||
Dirkjan Ochtman
|
r6393 | return [] | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('static') | ||
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: | ||||
Mads Kiilerich
|
r22634 | tp = web.templatepath or templater.templatepaths() | ||
Brendan Cully
|
r7107 | if isinstance(tp, str): | ||
tp = [tp] | ||||
Brendan Cully
|
r7288 | static = [os.path.join(p, 'static') for p in tp] | ||
Mads Kiilerich
|
r18645 | staticfile(static, fname, req) | ||
return [] | ||||
Dirkjan Ochtman
|
r6691 | |||
Gregory Szorc
|
r24076 | @webcommand('graph') | ||
Dirkjan Ochtman
|
r6691 | def graph(web, req, tmpl): | ||
Gregory Szorc
|
r24097 | """ | ||
/graph[/{revision}] | ||||
------------------- | ||||
Show information about the graphical topology of the repository. | ||||
Information rendered by this handler can be used to create visual | ||||
representations of repository topology. | ||||
r35412 | The ``revision`` URL parameter controls the starting changeset. If it's | |||
absent, the default is ``tip``. | ||||
Gregory Szorc
|
r24097 | |||
The ``revcount`` query string argument can define the number of changesets | ||||
to show information for. | ||||
r35412 | The ``graphtop`` query string argument can specify the starting changeset | |||
for producing ``jsdata`` variable that is used for rendering graph in | ||||
JavaScript. By default it has the same value as ``revision``. | ||||
Gregory Szorc
|
r24097 | This handler will render the ``graph`` template. | ||
""" | ||||
Dirkjan Ochtman
|
r10245 | |||
r25602 | if 'node' in req.form: | |||
ctx = webutil.changectx(web.repo, req) | ||||
symrev = webutil.symrevorshortnode(req, ctx) | ||||
else: | ||||
ctx = web.repo['tip'] | ||||
symrev = 'tip' | ||||
Patrick Mezard
|
r17318 | rev = ctx.rev() | ||
Dirkjan Ochtman
|
r6691 | bg_height = 39 | ||
Dirkjan Ochtman
|
r10245 | revcount = web.maxshortchanges | ||
Dirkjan Ochtman
|
r7345 | if 'revcount' in req.form: | ||
Isaac Jurado
|
r20092 | try: | ||
revcount = int(req.form.get('revcount', [revcount])[0]) | ||||
revcount = max(revcount, 1) | ||||
tmpl.defaults['sessionvars']['revcount'] = revcount | ||||
except ValueError: | ||||
pass | ||||
Dirkjan Ochtman
|
r7345 | |||
lessvars = copy.copy(tmpl.defaults['sessionvars']) | ||||
Augie Fackler
|
r36571 | lessvars['revcount'] = max(revcount // 2, 1) | ||
Dirkjan Ochtman
|
r7345 | morevars = copy.copy(tmpl.defaults['sessionvars']) | ||
morevars['revcount'] = revcount * 2 | ||||
r35410 | graphtop = req.form.get('graphtop', [ctx.hex()])[0] | |||
graphvars = copy.copy(tmpl.defaults['sessionvars']) | ||||
graphvars['graphtop'] = graphtop | ||||
Patrick Mezard
|
r17318 | count = len(web.repo) | ||
pos = rev | ||||
uprev = min(max(0, count - 1), rev + revcount) | ||||
Dirkjan Ochtman
|
r6691 | downrev = max(0, rev - revcount) | ||
Pierre-Yves David
|
r18409 | changenav = webutil.revnav(web.repo).gen(pos, revcount, count) | ||
Dirkjan Ochtman
|
r6691 | |||
Pierre-Yves David
|
r18428 | tree = [] | ||
r35410 | nextentry = [] | |||
lastrev = 0 | ||||
Alexander Plavin
|
r19487 | if pos != -1: | ||
allrevs = web.repo.changelog.revs(pos, 0) | ||||
revs = [] | ||||
for i in allrevs: | ||||
revs.append(i) | ||||
r35410 | if len(revs) >= revcount + 1: | |||
Alexander Plavin
|
r19487 | break | ||
r35410 | if len(revs) > revcount: | |||
nextentry = [webutil.commonentry(web.repo, web.repo[revs[-1]])] | ||||
revs = revs[:-1] | ||||
lastrev = revs[-1] | ||||
Lucas Moscovicz
|
r20761 | # We have to feed a baseset to dagwalker as it is expecting smartset | ||
# object. This does not have a big impact on hgweb performance itself | ||||
# since hgweb graphing code is not itself lazy yet. | ||||
Yuya Nishihara
|
r31023 | dag = graphmod.dagwalker(web.repo, smartset.baseset(revs)) | ||
Lucas Moscovicz
|
r20761 | # As we said one line above... not lazy. | ||
r35407 | tree = list(item for item in graphmod.colored(dag, web.repo) | |||
if item[1] == graphmod.CHANGESET) | ||||
Paul Boddie
|
r16773 | |||
r35582 | def nodecurrent(ctx): | |||
wpnodes = web.repo.dirstate.parents() | ||||
if wpnodes[1] == nullid: | ||||
wpnodes = wpnodes[:1] | ||||
if ctx.node() in wpnodes: | ||||
return '@' | ||||
return '' | ||||
def nodesymbol(ctx): | ||||
if ctx.obsolete(): | ||||
return 'x' | ||||
elif ctx.isunstable(): | ||||
return '*' | ||||
elif ctx.closesbranch(): | ||||
return '_' | ||||
else: | ||||
return 'o' | ||||
r35410 | def fulltree(): | |||
pos = web.repo[graphtop].rev() | ||||
tree = [] | ||||
if pos != -1: | ||||
revs = web.repo.changelog.revs(pos, lastrev) | ||||
dag = graphmod.dagwalker(web.repo, smartset.baseset(revs)) | ||||
tree = list(item for item in graphmod.colored(dag, web.repo) | ||||
if item[1] == graphmod.CHANGESET) | ||||
return tree | ||||
r35409 | def jsdata(): | |||
return [{'node': pycompat.bytestr(ctx), | ||||
r35582 | 'graphnode': nodecurrent(ctx) + nodesymbol(ctx), | |||
r35409 | 'vertex': vtx, | |||
'edges': edges} | ||||
r35410 | for (id, type, ctx, vtx, edges) in fulltree()] | |||
Paul Boddie
|
r16773 | |||
r35409 | def nodes(): | |||
r35548 | parity = paritygen(web.stripecount) | |||
r35409 | for row, (id, type, ctx, vtx, edges) in enumerate(tree): | |||
entry = webutil.commonentry(web.repo, ctx) | ||||
edgedata = [{'col': edge[0], | ||||
'nextcol': edge[1], | ||||
'color': (edge[2] - 1) % 6 + 1, | ||||
'width': edge[3], | ||||
'bcolor': edge[4]} | ||||
for edge in edges] | ||||
Paul Boddie
|
r16773 | |||
r35409 | entry.update({'col': vtx[0], | |||
'color': (vtx[1] - 1) % 6 + 1, | ||||
r35548 | 'parity': next(parity), | |||
r35409 | 'edges': edgedata, | |||
'row': row, | ||||
'nextrow': row + 1}) | ||||
r35095 | ||||
r35409 | yield entry | |||
Paul Boddie
|
r16773 | |||
rows = len(tree) | ||||
Dirkjan Ochtman
|
r6691 | |||
r25602 | return tmpl('graph', rev=rev, symrev=symrev, revcount=revcount, | |||
uprev=uprev, | ||||
Dirkjan Ochtman
|
r7345 | lessvars=lessvars, morevars=morevars, downrev=downrev, | ||
r35410 | graphvars=graphvars, | |||
r35408 | rows=rows, | |||
bg_height=bg_height, | ||||
changesets=count, | ||||
r35410 | nextentry=nextentry, | |||
r35409 | jsdata=lambda **x: jsdata(), | |||
nodes=lambda **x: nodes(), | ||||
Patrick Mezard
|
r17318 | node=ctx.hex(), changenav=changenav) | ||
Augie Fackler
|
r12666 | |||
def _getdoc(e): | ||||
doc = e[0].__doc__ | ||||
if doc: | ||||
r26846 | doc = _(doc).partition('\n')[0] | |||
Augie Fackler
|
r12666 | else: | ||
doc = _('(no help text available)') | ||||
return doc | ||||
Gregory Szorc
|
r24076 | @webcommand('help') | ||
Augie Fackler
|
r12666 | def help(web, req, tmpl): | ||
Gregory Szorc
|
r24081 | """ | ||
/help[/{topic}] | ||||
--------------- | ||||
Render help documentation. | ||||
This web command is roughly equivalent to :hg:`help`. If a ``topic`` | ||||
is defined, that help topic will be rendered. If not, an index of | ||||
available help topics will be rendered. | ||||
The ``help`` template will be rendered when requesting help for a topic. | ||||
``helptopics`` will be rendered for the index of help topics. | ||||
""" | ||||
Yuya Nishihara
|
r27046 | from .. import commands, help as helpmod # avoid cycle | ||
Augie Fackler
|
r12666 | |||
topicname = req.form.get('node', [None])[0] | ||||
if not topicname: | ||||
def topics(**map): | ||||
Mads Kiilerich
|
r22199 | for entries, summary, _doc in helpmod.helptable: | ||
Mads Kiilerich
|
r17322 | yield {'topic': entries[0], 'summary': summary} | ||
Augie Fackler
|
r12666 | |||
early, other = [], [] | ||||
r26846 | primary = lambda s: s.partition('|')[0] | |||
Augie Fackler
|
r12666 | for c, e in commands.table.iteritems(): | ||
doc = _getdoc(e) | ||||
if 'DEPRECATED' in doc or c.startswith('debug'): | ||||
continue | ||||
cmd = primary(c) | ||||
if cmd.startswith('^'): | ||||
early.append((cmd[1:], doc)) | ||||
else: | ||||
other.append((cmd, doc)) | ||||
early.sort() | ||||
other.sort() | ||||
def earlycommands(**map): | ||||
for c, doc in early: | ||||
yield {'topic': c, 'summary': doc} | ||||
def othercommands(**map): | ||||
for c, doc in other: | ||||
yield {'topic': c, 'summary': doc} | ||||
return tmpl('helptopics', topics=topics, earlycommands=earlycommands, | ||||
othercommands=othercommands, title='Index') | ||||
Gregory Szorc
|
r27581 | # Render an index of sub-topics. | ||
if topicname in helpmod.subtopics: | ||||
topics = [] | ||||
for entries, summary, _doc in helpmod.subtopics[topicname]: | ||||
topics.append({ | ||||
'topic': '%s.%s' % (topicname, entries[0]), | ||||
'basename': entries[0], | ||||
'summary': summary, | ||||
}) | ||||
return tmpl('helptopics', topics=topics, title=topicname, | ||||
subindex=True) | ||||
Yuya Nishihara
|
r30559 | u = webutil.wsgiui.load() | ||
Adrian Buehlmann
|
r17146 | u.verbose = True | ||
Gregory Szorc
|
r27582 | |||
# Render a page from a sub-topic. | ||||
if '.' in topicname: | ||||
# TODO implement support for rendering sections, like | ||||
# `hg help` works. | ||||
topic, subtopic = topicname.split('.', 1) | ||||
if topic not in helpmod.subtopics: | ||||
raise ErrorResponse(HTTP_NOT_FOUND) | ||||
else: | ||||
topic = topicname | ||||
subtopic = None | ||||
Augie Fackler
|
r12666 | try: | ||
Yuya Nishihara
|
r32567 | doc = helpmod.help_(u, commands, topic, subtopic=subtopic) | ||
Yuya Nishihara
|
r36262 | except error.Abort: | ||
Augie Fackler
|
r12666 | raise ErrorResponse(HTTP_NOT_FOUND) | ||
return tmpl('help', topic=topicname, doc=doc) | ||||
FUJIWARA Katsunori
|
r24859 | |||
# tell hggettext to extract docstrings from these functions: | ||||
i18nfunctions = commands.values() | ||||