webcommands.py
1476 lines
| 44.8 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 _ | ||||
r37928 | from ..node import hex, short | |||
Yuya Nishihara
|
r27046 | |||
from .common import ( | ||||
ErrorResponse, | ||||
HTTP_FORBIDDEN, | ||||
HTTP_NOT_FOUND, | ||||
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, | ||
Yuya Nishihara
|
r37418 | templateutil, | ||
Yuya Nishihara
|
r37102 | ) | ||
from ..utils import ( | ||||
stringutil, | ||||
Yuya Nishihara
|
r27046 | ) | ||
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. | ||||
Gregory Szorc
|
r36886 | When called, functions receive as arguments a ``requestcontext``, | ||
``wsgirequest``, and a templater instance for generatoring output. | ||||
The functions should populate the ``rctx.res`` object with details | ||||
about the HTTP response. | ||||
Gregory Szorc
|
r36896 | The function returns a generator to be consumed by the WSGI application. | ||
For most commands, this should be the result from | ||||
Gregory Szorc
|
r36899 | ``web.res.sendresponse()``. Many commands will call ``web.sendtemplate()`` | ||
to render a template. | ||||
Gregory Szorc
|
r36886 | |||
Gregory Szorc
|
r24076 | Usage: | ||
Dirkjan Ochtman
|
r5963 | |||
Gregory Szorc
|
r24076 | @webcommand('mycommand') | ||
Gregory Szorc
|
r36903 | def mycommand(web): | ||
Gregory Szorc
|
r24076 | 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') | ||||
Gregory Szorc
|
r36903 | def log(web): | ||
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. | ||||
""" | ||||
Gregory Szorc
|
r36898 | if web.req.qsparams.get('file'): | ||
Gregory Szorc
|
r36903 | return filelog(web) | ||
Dirkjan Ochtman
|
r5591 | else: | ||
Gregory Szorc
|
r36903 | return changelog(web) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('rawfile') | ||
Gregory Szorc
|
r36903 | def rawfile(web): | ||
Boris Feld
|
r34609 | guessmime = web.configbool('web', 'guessmime') | ||
Matt Mackall
|
r15004 | |||
Gregory Szorc
|
r36898 | path = webutil.cleanpath(web.repo, web.req.qsparams.get('file', '')) | ||
Dirkjan Ochtman
|
r5890 | if not path: | ||
Gregory Szorc
|
r36903 | return manifest(web) | ||
Dirkjan Ochtman
|
r5890 | |||
try: | ||||
Gregory Szorc
|
r36902 | fctx = webutil.filectx(web.repo, web.req) | ||
Gregory Szorc
|
r25660 | except error.LookupError as inst: | ||
Dirkjan Ochtman
|
r6368 | try: | ||
Gregory Szorc
|
r36903 | return manifest(web) | ||
Dirkjan Ochtman
|
r6368 | 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: | ||||
Yuya Nishihara
|
r37102 | if stringutil.binary(text): | ||
Jordi Gutiérrez Hermoso
|
r24306 | mt = 'application/binary' | ||
else: | ||||
mt = 'text/plain' | ||||
Julian Cowley
|
r11332 | if mt.startswith('text/'): | ||
mt += '; charset="%s"' % encoding.encoding | ||||
Dirkjan Ochtman
|
r5890 | |||
Gregory Szorc
|
r36887 | web.res.headers['Content-Type'] = mt | ||
filename = (path.rpartition('/')[-1] | ||||
.replace('\\', '\\\\').replace('"', '\\"')) | ||||
web.res.headers['Content-Disposition'] = 'inline; filename="%s"' % filename | ||||
web.res.setbodybytes(text) | ||||
Gregory Szorc
|
r36896 | return web.res.sendresponse() | ||
Dirkjan Ochtman
|
r5890 | |||
Gregory Szorc
|
r36903 | def _filerevision(web, 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 | |||
Yuya Nishihara
|
r37102 | if stringutil.binary(text): | ||
Dirkjan Ochtman
|
r6393 | mt = mimetypes.guess_type(f)[0] or 'application/octet-stream' | ||
text = '(binary:%s)' % mt | ||||
Yuya Nishihara
|
r38071 | def lines(context): | ||
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 | |||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'filerevision', | ||
file=f, | ||||
path=webutil.up(f), | ||||
Yuya Nishihara
|
r38071 | text=templateutil.mappinggenerator(lines), | ||
Gregory Szorc
|
r36902 | symrev=webutil.symrevorshortnode(web.req, fctx), | ||
Gregory Szorc
|
r36887 | rename=webutil.renamelink(fctx), | ||
permissions=fctx.manifest().flags(f), | ||||
ishead=int(ishead), | ||||
Gregory Szorc
|
r36899 | **pycompat.strkwargs(webutil.commonentry(web.repo, fctx))) | ||
Dirkjan Ochtman
|
r6393 | |||
Gregory Szorc
|
r24076 | @webcommand('file') | ||
Gregory Szorc
|
r36903 | def file(web): | ||
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. | ||||
""" | ||||
Gregory Szorc
|
r36888 | if web.req.qsparams.get('style') == 'raw': | ||
Gregory Szorc
|
r36903 | return rawfile(web) | ||
Gregory Szorc
|
r36888 | |||
Gregory Szorc
|
r36898 | path = webutil.cleanpath(web.repo, web.req.qsparams.get('file', '')) | ||
Benoit Boissinot
|
r6853 | if not path: | ||
Gregory Szorc
|
r36903 | return manifest(web) | ||
Benoit Boissinot
|
r6853 | try: | ||
Gregory Szorc
|
r36903 | return _filerevision(web, webutil.filectx(web.repo, web.req)) | ||
Gregory Szorc
|
r25660 | except error.LookupError as inst: | ||
Dirkjan Ochtman
|
r5591 | try: | ||
Gregory Szorc
|
r36903 | return manifest(web) | ||
Benoit Boissinot
|
r6853 | except ErrorResponse: | ||
raise inst | ||||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r36900 | def _search(web): | ||
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: | ||
Martin von Zweigbergk
|
r37405 | ctx = scmutil.revsymbol(web.repo, query) | ||
Alexander Plavin
|
r19633 | 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 | ||||
Yuya Nishihara
|
r37692 | mfunc = revset.match(web.repo.ui, revdef, | ||
lookup=revset.lookupfn(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 | ||
Yuya Nishihara
|
r37418 | def changelist(context): | ||
Alexander Plavin
|
r19533 | count = 0 | ||
Alexander Plavin
|
r19765 | for ctx in searchfunc[0](funcarg): | ||
Andrew Beekhof
|
r6659 | count += 1 | ||
Dirkjan Ochtman
|
r6393 | n = ctx.node() | ||
Yuya Nishihara
|
r37931 | showtags = webutil.showtag(web.repo, 'changelogtag', n) | ||
Yuya Nishihara
|
r37973 | files = webutil.listfilediffs(ctx.files(), n, web.maxfiles) | ||
Dirkjan Ochtman
|
r6393 | |||
Yuya Nishihara
|
r37037 | lm = webutil.commonentry(web.repo, ctx) | ||
lm.update({ | ||||
'parity': next(parity), | ||||
'changelogtag': showtags, | ||||
'files': files, | ||||
}) | ||||
Yuya Nishihara
|
r37418 | yield lm | ||
Dirkjan Ochtman
|
r6393 | |||
Dirkjan Ochtman
|
r10247 | if count >= revcount: | ||
Dirkjan Ochtman
|
r6393 | break | ||
Gregory Szorc
|
r36898 | query = web.req.qsparams['rev'] | ||
Alexander Plavin
|
r19418 | revcount = web.maxchanges | ||
Gregory Szorc
|
r36898 | if 'revcount' in web.req.qsparams: | ||
Isaac Jurado
|
r20092 | try: | ||
Gregory Szorc
|
r36898 | revcount = int(web.req.qsparams.get('revcount', revcount)) | ||
Isaac Jurado
|
r20092 | revcount = max(revcount, 1) | ||
Gregory Szorc
|
r36900 | web.tmpl.defaults['sessionvars']['revcount'] = revcount | ||
Isaac Jurado
|
r20092 | except ValueError: | ||
pass | ||||
Alexander Plavin
|
r19418 | |||
Gregory Szorc
|
r36900 | lessvars = copy.copy(web.tmpl.defaults['sessionvars']) | ||
Augie Fackler
|
r36571 | lessvars['revcount'] = max(revcount // 2, 1) | ||
Alexander Plavin
|
r19418 | lessvars['rev'] = query | ||
Gregory Szorc
|
r36900 | morevars = copy.copy(web.tmpl.defaults['sessionvars']) | ||
Alexander Plavin
|
r19418 | morevars['revcount'] = revcount * 2 | ||
morevars['rev'] = query | ||||
Alexander Plavin
|
r19632 | mode, funcarg = getsearchmode(query) | ||
Alexander Plavin
|
r19768 | |||
Gregory Szorc
|
r36898 | if 'forcekw' in web.req.qsparams: | ||
Alexander Plavin
|
r19768 | 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) | ||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'search', | ||
query=query, | ||||
node=tip.hex(), | ||||
symrev='tip', | ||||
Yuya Nishihara
|
r37418 | entries=templateutil.mappinggenerator(changelist, name='searchentry'), | ||
Gregory Szorc
|
r36887 | archives=web.archivelist('tip'), | ||
morevars=morevars, | ||||
lessvars=lessvars, | ||||
modedesc=searchfunc[1], | ||||
showforcekw=showforcekw, | ||||
Gregory Szorc
|
r36899 | showunforcekw=showunforcekw) | ||
Dirkjan Ochtman
|
r6393 | |||
Gregory Szorc
|
r24076 | @webcommand('changelog') | ||
Gregory Szorc
|
r36903 | def changelog(web, 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 = '' | ||
Gregory Szorc
|
r36898 | if 'node' in web.req.qsparams: | ||
Gregory Szorc
|
r36902 | ctx = webutil.changectx(web.repo, web.req) | ||
symrev = webutil.symrevorshortnode(web.req, ctx) | ||||
Gregory Szorc
|
r36898 | elif 'rev' in web.req.qsparams: | ||
Gregory Szorc
|
r36900 | return _search(web) | ||
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) | ||||
Gregory Szorc
|
r23745 | |||
Gregory Szorc
|
r38054 | for entry in webutil.changelistentries(web, revs, revcount, parity): | ||
Gregory Szorc
|
r23745 | yield entry | ||
Dirkjan Ochtman
|
r6393 | |||
Jordi Gutiérrez Hermoso
|
r24306 | if shortlog: | ||
revcount = web.maxshortchanges | ||||
else: | ||||
revcount = web.maxchanges | ||||
Gregory Szorc
|
r36898 | if 'revcount' in web.req.qsparams: | ||
Isaac Jurado
|
r20092 | try: | ||
Gregory Szorc
|
r36898 | revcount = int(web.req.qsparams.get('revcount', revcount)) | ||
Isaac Jurado
|
r20092 | revcount = max(revcount, 1) | ||
Gregory Szorc
|
r36900 | web.tmpl.defaults['sessionvars']['revcount'] = revcount | ||
Isaac Jurado
|
r20092 | except ValueError: | ||
pass | ||||
Dirkjan Ochtman
|
r10246 | |||
Gregory Szorc
|
r36900 | lessvars = copy.copy(web.tmpl.defaults['sessionvars']) | ||
Augie Fackler
|
r36571 | lessvars['revcount'] = max(revcount // 2, 1) | ||
Gregory Szorc
|
r36900 | morevars = copy.copy(web.tmpl.defaults['sessionvars']) | ||
Dirkjan Ochtman
|
r10246 | 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 | |||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'shortlog' if shortlog else 'changelog', | ||
changenav=changenav, | ||||
node=ctx.hex(), | ||||
rev=pos, | ||||
symrev=symrev, | ||||
changesets=count, | ||||
Yuya Nishihara
|
r38072 | entries=templateutil.mappinglist(entries), | ||
latestentry=templateutil.mappinglist(latestentry), | ||||
nextentry=templateutil.mappinglist(nextentry), | ||||
Gregory Szorc
|
r36887 | archives=web.archivelist('tip'), | ||
revcount=revcount, | ||||
morevars=morevars, | ||||
lessvars=lessvars, | ||||
Gregory Szorc
|
r36899 | query=query) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('shortlog') | ||
Gregory Szorc
|
r36903 | def shortlog(web): | ||
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. | ||||
""" | ||||
Gregory Szorc
|
r36903 | return changelog(web, shortlog=True) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('changeset') | ||
Gregory Szorc
|
r36903 | def changeset(web): | ||
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. | ||||
""" | ||||
Gregory Szorc
|
r36902 | ctx = webutil.changectx(web.repo, web.req) | ||
Gregory Szorc
|
r36899 | |||
return web.sendtemplate( | ||||
'changeset', | ||||
Gregory Szorc
|
r36902 | **webutil.changesetentry(web, 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') | ||
Gregory Szorc
|
r36903 | def manifest(web): | ||
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. | ||||
""" | ||||
Gregory Szorc
|
r36898 | if 'node' in web.req.qsparams: | ||
Gregory Szorc
|
r36902 | ctx = webutil.changectx(web.repo, web.req) | ||
symrev = webutil.symrevorshortnode(web.req, ctx) | ||||
r25602 | else: | |||
ctx = web.repo['tip'] | ||||
symrev = 'tip' | ||||
Gregory Szorc
|
r36898 | path = webutil.cleanpath(web.repo, web.req.qsparams.get('file', '')) | ||
Dirkjan Ochtman
|
r6393 | 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) | ||
Yuya Nishihara
|
r38073 | def filelist(context): | ||
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)} | ||||
Yuya Nishihara
|
r38073 | def dirlist(context): | ||
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 | |||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'manifest', | ||
symrev=symrev, | ||||
path=abspath, | ||||
up=webutil.up(abspath), | ||||
upparity=next(parity), | ||||
Yuya Nishihara
|
r38073 | fentries=templateutil.mappinggenerator(filelist), | ||
dentries=templateutil.mappinggenerator(dirlist), | ||||
Gregory Szorc
|
r36887 | archives=web.archivelist(hex(node)), | ||
Gregory Szorc
|
r36899 | **pycompat.strkwargs(webutil.commonentry(web.repo, ctx))) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('tags') | ||
Gregory Szorc
|
r36903 | def tags(web): | ||
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) | ||
Yuya Nishihara
|
r38074 | def entries(context, notip, latestonly): | ||
Pierre-Yves David
|
r18402 | 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)} | ||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'tags', | ||
node=hex(web.repo.changelog.tip()), | ||||
Yuya Nishihara
|
r38074 | entries=templateutil.mappinggenerator(entries, args=(False, False)), | ||
entriesnotip=templateutil.mappinggenerator(entries, | ||||
args=(True, False)), | ||||
latestentry=templateutil.mappinggenerator(entries, args=(True, True))) | ||||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('bookmarks') | ||
Gregory Szorc
|
r36903 | def bookmarks(web): | ||
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) | ||
Yuya Nishihara
|
r38149 | def entries(context, latestonly): | ||
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 | ||||
Yuya Nishihara
|
r38150 | lastdate = web.repo[latestrev].date() | ||
r28712 | ||||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'bookmarks', | ||
node=hex(web.repo.changelog.tip()), | ||||
Yuya Nishihara
|
r38150 | lastchange=templateutil.mappinglist([{'date': lastdate}]), | ||
Yuya Nishihara
|
r38149 | entries=templateutil.mappinggenerator(entries, args=(False,)), | ||
latestentry=templateutil.mappinggenerator(entries, args=(True,))) | ||||
Alexander Solovyov
|
r13597 | |||
Gregory Szorc
|
r24076 | @webcommand('branches') | ||
Gregory Szorc
|
r36903 | def branches(web): | ||
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) | ||||
Gregory Szorc
|
r36887 | |||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'branches', | ||
node=hex(web.repo.changelog.tip()), | ||||
entries=entries, | ||||
Gregory Szorc
|
r36899 | latestentry=latestentry) | ||
Sune Foldager
|
r8352 | |||
Gregory Szorc
|
r24076 | @webcommand('summary') | ||
Gregory Szorc
|
r36903 | def summary(web): | ||
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 | |||
Yuya Nishihara
|
r37419 | def tagentries(context): | ||
Dirkjan Ochtman
|
r6393 | 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 | ||||
Yuya Nishihara
|
r37419 | yield { | ||
Yuya Nishihara
|
r37037 | 'parity': next(parity), | ||
'tag': k, | ||||
'node': hex(n), | ||||
'date': web.repo[n].date(), | ||||
Yuya Nishihara
|
r37419 | } | ||
Dirkjan Ochtman
|
r6393 | |||
Yuya Nishihara
|
r38151 | def bookmarks(context): | ||
Yuya Nishihara
|
r13924 | 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)} | ||||
Yuya Nishihara
|
r37419 | def changelist(context): | ||
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] | ||
Yuya Nishihara
|
r37037 | lm = webutil.commonentry(web.repo, ctx) | ||
lm['parity'] = next(parity) | ||||
Yuya Nishihara
|
r37419 | l.append(lm) | ||
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' | ||||
Yuya Nishihara
|
r37528 | labels = web.configlist('web', 'labels') | ||
Gregory Szorc
|
r36887 | |||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'summary', | ||
desc=desc, | ||||
owner=get_contact(web.config) or 'unknown', | ||||
lastchange=tip.date(), | ||||
Yuya Nishihara
|
r37419 | tags=templateutil.mappinggenerator(tagentries, name='tagentry'), | ||
Yuya Nishihara
|
r38151 | bookmarks=templateutil.mappinggenerator(bookmarks), | ||
Gregory Szorc
|
r36887 | branches=webutil.branchentries(web.repo, web.stripecount, 10), | ||
Yuya Nishihara
|
r37419 | shortlog=templateutil.mappinggenerator(changelist, | ||
name='shortlogentry'), | ||||
Gregory Szorc
|
r36887 | node=tip.hex(), | ||
symrev='tip', | ||||
archives=web.archivelist('tip'), | ||||
Yuya Nishihara
|
r37528 | labels=templateutil.hybridlist(labels, name='label')) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('filediff') | ||
Gregory Szorc
|
r36903 | def filediff(web): | ||
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: | ||||
Gregory Szorc
|
r36902 | fctx = webutil.filectx(web.repo, web.req) | ||
Benoit Boissinot
|
r7280 | except LookupError: | ||
Gregory Szorc
|
r36902 | ctx = webutil.changectx(web.repo, web.req) | ||
Gregory Szorc
|
r36898 | path = webutil.cleanpath(web.repo, web.req.qsparams['file']) | ||
Dirkjan Ochtman
|
r7183 | 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') | ||
Gregory Szorc
|
r36898 | if 'style' in web.req.qsparams: | ||
style = web.req.qsparams['style'] | ||||
Dirkjan Ochtman
|
r9402 | |||
Gregory Szorc
|
r36901 | diffs = webutil.diffs(web, ctx, basectx, [path], style) | ||
r27160 | if fctx is not None: | |||
Jordi Gutiérrez Hermoso
|
r24306 | rename = webutil.renamelink(fctx) | ||
ctx = fctx | ||||
else: | ||||
Yuya Nishihara
|
r37921 | rename = templateutil.mappinglist([]) | ||
Jordi Gutiérrez Hermoso
|
r24306 | ctx = ctx | ||
Gregory Szorc
|
r36887 | |||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'filediff', | ||
file=path, | ||||
Gregory Szorc
|
r36902 | symrev=webutil.symrevorshortnode(web.req, ctx), | ||
Gregory Szorc
|
r36887 | rename=rename, | ||
diff=diffs, | ||||
Gregory Szorc
|
r36899 | **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') | ||
Gregory Szorc
|
r36903 | def comparison(web): | ||
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. | ||||
""" | ||||
Gregory Szorc
|
r36902 | ctx = webutil.changectx(web.repo, web.req) | ||
Gregory Szorc
|
r36898 | if 'file' not in web.req.qsparams: | ||
Ross Lagerwall
|
r17289 | raise ErrorResponse(HTTP_NOT_FOUND, 'file not given') | ||
Gregory Szorc
|
r36898 | path = webutil.cleanpath(web.repo, web.req.qsparams['file']) | ||
wujek srujek
|
r17202 | |||
parsecontext = lambda v: v == 'full' and -1 or int(v) | ||||
Gregory Szorc
|
r36898 | if 'context' in web.req.qsparams: | ||
context = parsecontext(web.req.qsparams['context']) | ||||
wujek srujek
|
r17202 | 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 | |||
Yuya Nishihara
|
r38013 | comparison = webutil.compare(context, leftlines, rightlines) | ||
r27158 | if fctx is not None: | |||
r27159 | rename = webutil.renamelink(fctx) | |||
r27158 | ctx = fctx | |||
else: | ||||
Yuya Nishihara
|
r37921 | rename = templateutil.mappinglist([]) | ||
r27158 | ctx = ctx | |||
Gregory Szorc
|
r36887 | |||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'filecomparison', | ||
file=path, | ||||
Gregory Szorc
|
r36902 | symrev=webutil.symrevorshortnode(web.req, ctx), | ||
Gregory Szorc
|
r36887 | rename=rename, | ||
leftrev=leftrev, | ||||
leftnode=hex(leftnode), | ||||
rightrev=rightrev, | ||||
rightnode=hex(rightnode), | ||||
comparison=comparison, | ||||
Gregory Szorc
|
r36899 | **pycompat.strkwargs(webutil.commonentry(web.repo, ctx))) | ||
wujek srujek
|
r17202 | |||
Gregory Szorc
|
r24076 | @webcommand('annotate') | ||
Gregory Szorc
|
r36903 | def annotate(web): | ||
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. | ||
""" | ||||
Gregory Szorc
|
r36902 | fctx = webutil.filectx(web.repo, web.req) | ||
Dirkjan Ochtman
|
r6393 | 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 = {} | ||||
Yuya Nishihara
|
r38153 | def parents(context, 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 | |||
Yuya Nishihara
|
r38152 | def annotate(context): | ||
Jun Wu
|
r32136 | if fctx.isbinary(): | ||
Dirkjan Ochtman
|
r6393 | mt = (mimetypes.guess_type(fctx.path())[0] | ||
or 'application/octet-stream') | ||||
Yuya Nishihara
|
r37084 | lines = [dagop.annotateline(fctx=fctx.filectx(fctx.filerev()), | ||
lineno=1, text='(binary:%s)' % mt)] | ||||
Dirkjan Ochtman
|
r6393 | else: | ||
Gregory Szorc
|
r36902 | lines = webutil.annotate(web.req, fctx, web.repo.ui) | ||
Jun Wu
|
r30081 | |||
Denis Laxalde
|
r29388 | previousrev = None | ||
r29572 | blockparitygen = paritygen(1) | |||
Yuya Nishihara
|
r37084 | for lineno, aline in enumerate(lines): | ||
Siddharth Agarwal
|
r34433 | 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(), | ||
Yuya Nishihara
|
r38153 | "parents": templateutil.mappinggenerator(parents, args=(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, | ||
Yuya Nishihara
|
r37084 | "line": aline.text, | ||
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
|
r36902 | diffopts = webutil.difffeatureopts(web.req, web.repo.ui, 'annotate') | ||
Gregory Szorc
|
r34392 | diffopts = {k: getattr(diffopts, k) for k in diffopts.defaults} | ||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'fileannotate', | ||
file=f, | ||||
Yuya Nishihara
|
r38152 | annotate=templateutil.mappinggenerator(annotate), | ||
Gregory Szorc
|
r36887 | path=webutil.up(f), | ||
Gregory Szorc
|
r36902 | symrev=webutil.symrevorshortnode(web.req, fctx), | ||
Gregory Szorc
|
r36887 | rename=webutil.renamelink(fctx), | ||
permissions=fctx.manifest().flags(f), | ||||
ishead=int(ishead), | ||||
Yuya Nishihara
|
r38154 | diffopts=templateutil.hybriddict(diffopts), | ||
Gregory Szorc
|
r36899 | **pycompat.strkwargs(webutil.commonentry(web.repo, fctx))) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('filelog') | ||
Gregory Szorc
|
r36903 | def filelog(web): | ||
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: | ||||
Gregory Szorc
|
r36902 | fctx = webutil.filectx(web.repo, web.req) | ||
Dirkjan Ochtman
|
r7300 | f = fctx.path() | ||
fl = fctx.filelog() | ||||
Matt Mackall
|
r7633 | except error.LookupError: | ||
Gregory Szorc
|
r36898 | f = webutil.cleanpath(web.repo, web.req.qsparams['file']) | ||
Dirkjan Ochtman
|
r7300 | fl = web.repo.file(f) | ||
numrevs = len(fl) | ||||
if not numrevs: # file doesn't exist at all | ||||
raise | ||||
Gregory Szorc
|
r36902 | rev = webutil.changectx(web.repo, web.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 | ||
Gregory Szorc
|
r36898 | if 'revcount' in web.req.qsparams: | ||
Isaac Jurado
|
r20092 | try: | ||
Gregory Szorc
|
r36898 | revcount = int(web.req.qsparams.get('revcount', revcount)) | ||
Isaac Jurado
|
r20092 | revcount = max(revcount, 1) | ||
Gregory Szorc
|
r36900 | web.tmpl.defaults['sessionvars']['revcount'] = revcount | ||
Isaac Jurado
|
r20092 | except ValueError: | ||
pass | ||||
Dirkjan Ochtman
|
r10246 | |||
Gregory Szorc
|
r36902 | lrange = webutil.linerange(web.req) | ||
Denis Laxalde
|
r31665 | |||
Gregory Szorc
|
r36900 | lessvars = copy.copy(web.tmpl.defaults['sessionvars']) | ||
Augie Fackler
|
r36571 | lessvars['revcount'] = max(revcount // 2, 1) | ||
Gregory Szorc
|
r36900 | morevars = copy.copy(web.tmpl.defaults['sessionvars']) | ||
Dirkjan Ochtman
|
r10246 | morevars['revcount'] = revcount * 2 | ||
Gregory Szorc
|
r36898 | patch = 'patch' in web.req.qsparams | ||
Denis Laxalde
|
r31661 | if patch: | ||
Gregory Szorc
|
r36898 | lessvars['patch'] = morevars['patch'] = web.req.qsparams['patch'] | ||
descend = 'descend' in web.req.qsparams | ||||
Denis Laxalde
|
r31939 | if descend: | ||
Gregory Szorc
|
r36898 | lessvars['descend'] = morevars['descend'] = web.req.qsparams['descend'] | ||
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 | ||
Martin von Zweigbergk
|
r37381 | filelog = fctx.filelog() | ||
revs = [filerev for filerev in filelog.revs(start, end - 1) | ||||
if filelog.linkrev(filerev) in repo] | ||||
Denis Laxalde
|
r30816 | entries = [] | ||
Denis Laxalde
|
r31661 | |||
Boris Feld
|
r34243 | diffstyle = web.config('web', 'style') | ||
Gregory Szorc
|
r36898 | if 'style' in web.req.qsparams: | ||
diffstyle = web.req.qsparams['style'] | ||||
Denis Laxalde
|
r31661 | |||
Denis Laxalde
|
r31667 | def diff(fctx, linerange=None): | ||
Denis Laxalde
|
r31661 | ctx = fctx.changectx() | ||
basectx = ctx.p1() | ||||
path = fctx.path() | ||||
Gregory Szorc
|
r36901 | return webutil.diffs(web, 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 | ||||
Yuya Nishihara
|
r37716 | nav = templateutil.mappinglist([]) | ||
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() | ||||
Yuya Nishihara
|
r38155 | lm = webutil.commonentry(repo, c) | ||
lm.update({ | ||||
'parity': next(parity), | ||||
'filerev': c.rev(), | ||||
'file': path, | ||||
'diff': diffs, | ||||
'linerange': webutil.formatlinerange(*lr), | ||||
}) | ||||
entries.append(lm) | ||||
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) | ||||
Yuya Nishihara
|
r38155 | lm = webutil.commonentry(repo, iterfctx) | ||
lm.update({ | ||||
'parity': next(parity), | ||||
'filerev': i, | ||||
'file': f, | ||||
'diff': diffs, | ||||
'rename': webutil.renamelink(iterfctx), | ||||
}) | ||||
entries.append(lm) | ||||
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] | ||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36886 | 'filelog', | ||
file=f, | ||||
nav=nav, | ||||
Gregory Szorc
|
r36902 | symrev=webutil.symrevorshortnode(web.req, fctx), | ||
Yuya Nishihara
|
r38156 | entries=templateutil.mappinglist(entries), | ||
Gregory Szorc
|
r36886 | descend=descend, | ||
patch=patch, | ||||
Yuya Nishihara
|
r38156 | latestentry=templateutil.mappinglist(latestentry), | ||
Gregory Szorc
|
r36886 | linerange=linerange, | ||
revcount=revcount, | ||||
morevars=morevars, | ||||
lessvars=lessvars, | ||||
Gregory Szorc
|
r36899 | **pycompat.strkwargs(webutil.commonentry(web.repo, fctx))) | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('archive') | ||
Gregory Szorc
|
r36903 | def archive(web): | ||
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. | ||||
""" | ||||
Gregory Szorc
|
r36898 | type_ = web.req.qsparams.get('type') | ||
Dirkjan Ochtman
|
r5591 | allowed = web.configlist("web", "allow_archive") | ||
Gregory Szorc
|
r36898 | key = web.req.qsparams['node'] | ||
Dirkjan Ochtman
|
r6393 | |||
Yuya Nishihara
|
r37530 | if type_ not in webutil.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 | |||
Gregory Szorc
|
r36902 | ctx = webutil.changectx(web.repo, web.req) | ||
Angel Ezquerra
|
r18771 | pats = [] | ||
Martin von Zweigbergk
|
r34085 | match = scmutil.match(ctx, []) | ||
Gregory Szorc
|
r36898 | file = web.req.qsparams.get('file') | ||
Angel Ezquerra
|
r18771 | if file: | ||
Gregory Szorc
|
r36881 | pats = ['path:' + file] | ||
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, | ||||
Gregory Szorc
|
r36881 | 'file(s) not found: %s' % file) | ||
Angel Ezquerra
|
r18771 | |||
Yuya Nishihara
|
r37530 | mimetype, artype, extension, encoding = webutil.archivespecs[type_] | ||
Gregory Szorc
|
r36892 | |||
web.res.headers['Content-Type'] = mimetype | ||||
web.res.headers['Content-Disposition'] = 'attachment; filename=%s%s' % ( | ||||
name, extension) | ||||
Dirkjan Ochtman
|
r6393 | if encoding: | ||
Gregory Szorc
|
r36892 | web.res.headers['Content-Encoding'] = encoding | ||
Jordi Gutiérrez Hermoso
|
r17933 | |||
Gregory Szorc
|
r36892 | web.res.setbodywillwrite() | ||
Gregory Szorc
|
r36995 | if list(web.res.sendresponse()): | ||
raise error.ProgrammingError('sendresponse() should not emit data ' | ||||
'if writing later') | ||||
Gregory Szorc
|
r36892 | |||
bodyfh = web.res.getbodyfile() | ||||
Gregory Szorc
|
r36891 | |||
archival.archive(web.repo, bodyfh, cnode, artype, prefix=name, | ||||
Martin von Zweigbergk
|
r34085 | matchfn=match, | ||
Jordi Gutiérrez Hermoso
|
r17933 | subrepos=web.configbool("web", "archivesubrepos")) | ||
Dirkjan Ochtman
|
r6393 | |||
Gregory Szorc
|
r36896 | return [] | ||
Dirkjan Ochtman
|
r5591 | |||
Gregory Szorc
|
r24076 | @webcommand('static') | ||
Gregory Szorc
|
r36903 | def static(web): | ||
Gregory Szorc
|
r36898 | fname = web.req.qsparams['file'] | ||
Dirkjan Ochtman
|
r5591 | # 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] | ||
Gregory Szorc
|
r36889 | |||
staticfile(static, fname, web.res) | ||||
Gregory Szorc
|
r36896 | return web.res.sendresponse() | ||
Dirkjan Ochtman
|
r6691 | |||
Gregory Szorc
|
r24076 | @webcommand('graph') | ||
Gregory Szorc
|
r36903 | def graph(web): | ||
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 | |||
Gregory Szorc
|
r36898 | if 'node' in web.req.qsparams: | ||
Gregory Szorc
|
r36902 | ctx = webutil.changectx(web.repo, web.req) | ||
symrev = webutil.symrevorshortnode(web.req, ctx) | ||||
r25602 | 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 | ||
Gregory Szorc
|
r36898 | if 'revcount' in web.req.qsparams: | ||
Isaac Jurado
|
r20092 | try: | ||
Gregory Szorc
|
r36898 | revcount = int(web.req.qsparams.get('revcount', revcount)) | ||
Isaac Jurado
|
r20092 | revcount = max(revcount, 1) | ||
Gregory Szorc
|
r36900 | web.tmpl.defaults['sessionvars']['revcount'] = revcount | ||
Isaac Jurado
|
r20092 | except ValueError: | ||
pass | ||||
Dirkjan Ochtman
|
r7345 | |||
Gregory Szorc
|
r36900 | lessvars = copy.copy(web.tmpl.defaults['sessionvars']) | ||
Augie Fackler
|
r36571 | lessvars['revcount'] = max(revcount // 2, 1) | ||
Gregory Szorc
|
r36900 | morevars = copy.copy(web.tmpl.defaults['sessionvars']) | ||
Dirkjan Ochtman
|
r7345 | morevars['revcount'] = revcount * 2 | ||
Gregory Szorc
|
r36898 | graphtop = web.req.qsparams.get('graphtop', ctx.hex()) | ||
Gregory Szorc
|
r36900 | graphvars = copy.copy(web.tmpl.defaults['sessionvars']) | ||
r35410 | 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 | |||
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), | ||||
r37928 | 'graphnode': webutil.getgraphnode(web.repo, 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 | |||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'graph', | ||
rev=rev, | ||||
symrev=symrev, | ||||
revcount=revcount, | ||||
uprev=uprev, | ||||
lessvars=lessvars, | ||||
morevars=morevars, | ||||
downrev=downrev, | ||||
graphvars=graphvars, | ||||
rows=rows, | ||||
bg_height=bg_height, | ||||
changesets=count, | ||||
nextentry=nextentry, | ||||
jsdata=lambda **x: jsdata(), | ||||
nodes=lambda **x: nodes(), | ||||
node=ctx.hex(), | ||||
Gregory Szorc
|
r36899 | 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') | ||
Gregory Szorc
|
r36903 | def help(web): | ||
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 | |||
Gregory Szorc
|
r36898 | topicname = web.req.qsparams.get('node') | ||
Augie Fackler
|
r12666 | 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} | ||||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'helptopics', | ||
topics=topics, | ||||
earlycommands=earlycommands, | ||||
othercommands=othercommands, | ||||
Gregory Szorc
|
r36899 | title='Index') | ||
Augie Fackler
|
r12666 | |||
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, | ||||
}) | ||||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'helptopics', | ||
topics=topics, | ||||
title=topicname, | ||||
Gregory Szorc
|
r36899 | subindex=True) | ||
Gregory Szorc
|
r27581 | |||
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) | ||
Gregory Szorc
|
r36887 | |||
Gregory Szorc
|
r36899 | return web.sendtemplate( | ||
Gregory Szorc
|
r36887 | 'help', | ||
topic=topicname, | ||||
Gregory Szorc
|
r36899 | doc=doc) | ||
FUJIWARA Katsunori
|
r24859 | |||
# tell hggettext to extract docstrings from these functions: | ||||
i18nfunctions = commands.values() | ||||