help.py
828 lines
| 30.1 KiB
| text/x-python
|
PythonLexer
/ mercurial / help.py
Matt Mackall
|
r3795 | # help.py - help data for mercurial | ||
# | ||||
# Copyright 2006 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. | ||
Matt Mackall
|
r3795 | |||
Gregory Szorc
|
r27479 | from __future__ import absolute_import | ||
import itertools | ||||
import os | ||||
import textwrap | ||||
from .i18n import ( | ||||
_, | ||||
gettext, | ||||
) | ||||
from . import ( | ||||
cmdutil, | ||||
encoding, | ||||
error, | ||||
extensions, | ||||
Daniel Ploch
|
r37109 | fancyopts, | ||
Gregory Szorc
|
r27479 | filemerge, | ||
fileset, | ||||
minirst, | ||||
Pulkit Goyal
|
r32143 | pycompat, | ||
rdamazio@google.com
|
r40327 | registrar, | ||
Gregory Szorc
|
r27479 | revset, | ||
templatefilters, | ||||
Yuya Nishihara
|
r36940 | templatefuncs, | ||
Gregory Szorc
|
r27479 | templatekw, | ||
util, | ||||
) | ||||
from .hgweb import ( | ||||
webcommands, | ||||
) | ||||
Cédric Duval
|
r8863 | |||
Martin von Zweigbergk
|
r32291 | _exclkeywords = { | ||
Jun Wu
|
r31080 | "(ADVANCED)", | ||
Yuya Nishihara
|
r26370 | "(DEPRECATED)", | ||
"(EXPERIMENTAL)", | ||||
Jun Wu
|
r31080 | # i18n: "(ADVANCED)" is a keyword, must be translated consistently | ||
_("(ADVANCED)"), | ||||
Yuya Nishihara
|
r26370 | # i18n: "(DEPRECATED)" is a keyword, must be translated consistently | ||
_("(DEPRECATED)"), | ||||
# i18n: "(EXPERIMENTAL)" is a keyword, must be translated consistently | ||||
_("(EXPERIMENTAL)"), | ||||
Martin von Zweigbergk
|
r32291 | } | ||
Yuya Nishihara
|
r26369 | |||
rdamazio@google.com
|
r40327 | # The order in which command categories will be displayed. | ||
# Extensions with custom categories should insert them into this list | ||||
# after/before the appropriate item, rather than replacing the list or | ||||
# assuming absolute positions. | ||||
CATEGORY_ORDER = [ | ||||
rdamazio@google.com
|
r40329 | registrar.command.CATEGORY_REPO_CREATION, | ||
registrar.command.CATEGORY_REMOTE_REPO_MANAGEMENT, | ||||
registrar.command.CATEGORY_COMMITTING, | ||||
registrar.command.CATEGORY_CHANGE_MANAGEMENT, | ||||
registrar.command.CATEGORY_CHANGE_ORGANIZATION, | ||||
registrar.command.CATEGORY_FILE_CONTENTS, | ||||
registrar.command.CATEGORY_CHANGE_NAVIGATION , | ||||
registrar.command.CATEGORY_WORKING_DIRECTORY, | ||||
registrar.command.CATEGORY_IMPORT_EXPORT, | ||||
registrar.command.CATEGORY_MAINTENANCE, | ||||
registrar.command.CATEGORY_HELP, | ||||
registrar.command.CATEGORY_MISC, | ||||
rdamazio@google.com
|
r40327 | registrar.command.CATEGORY_NONE, | ||
] | ||||
# Human-readable category names. These are translated. | ||||
# Extensions with custom categories should add their names here. | ||||
CATEGORY_NAMES = { | ||||
rdamazio@google.com
|
r40329 | registrar.command.CATEGORY_REPO_CREATION: 'Repository creation', | ||
registrar.command.CATEGORY_REMOTE_REPO_MANAGEMENT: | ||||
'Remote repository management', | ||||
registrar.command.CATEGORY_COMMITTING: 'Change creation', | ||||
registrar.command.CATEGORY_CHANGE_NAVIGATION: 'Change navigation', | ||||
registrar.command.CATEGORY_CHANGE_MANAGEMENT: 'Change manipulation', | ||||
registrar.command.CATEGORY_CHANGE_ORGANIZATION: 'Change organization', | ||||
registrar.command.CATEGORY_WORKING_DIRECTORY: | ||||
'Working directory management', | ||||
registrar.command.CATEGORY_FILE_CONTENTS: 'File content management', | ||||
registrar.command.CATEGORY_IMPORT_EXPORT: 'Change import/export', | ||||
registrar.command.CATEGORY_MAINTENANCE: 'Repository maintenance', | ||||
registrar.command.CATEGORY_HELP: 'Help', | ||||
registrar.command.CATEGORY_MISC: 'Miscellaneous commands', | ||||
rdamazio@google.com
|
r40327 | registrar.command.CATEGORY_NONE: 'Uncategorized commands', | ||
} | ||||
Rodrigo Damazio
|
r40328 | # Topic categories. | ||
Rodrigo Damazio
|
r40330 | TOPIC_CATEGORY_IDS = 'ids' | ||
TOPIC_CATEGORY_OUTPUT = 'output' | ||||
TOPIC_CATEGORY_CONFIG = 'config' | ||||
TOPIC_CATEGORY_CONCEPTS = 'concepts' | ||||
TOPIC_CATEGORY_MISC = 'misc' | ||||
Rodrigo Damazio
|
r40328 | TOPIC_CATEGORY_NONE = 'none' | ||
# The order in which topic categories will be displayed. | ||||
# Extensions with custom categories should insert them into this list | ||||
# after/before the appropriate item, rather than replacing the list or | ||||
# assuming absolute positions. | ||||
TOPIC_CATEGORY_ORDER = [ | ||||
Rodrigo Damazio
|
r40330 | TOPIC_CATEGORY_IDS, | ||
TOPIC_CATEGORY_OUTPUT, | ||||
TOPIC_CATEGORY_CONFIG, | ||||
TOPIC_CATEGORY_CONCEPTS, | ||||
TOPIC_CATEGORY_MISC, | ||||
Rodrigo Damazio
|
r40328 | TOPIC_CATEGORY_NONE, | ||
] | ||||
# Human-readable topic category names. These are translated. | ||||
TOPIC_CATEGORY_NAMES = { | ||||
Rodrigo Damazio
|
r40330 | TOPIC_CATEGORY_IDS: 'Mercurial identifiers', | ||
TOPIC_CATEGORY_OUTPUT: 'Mercurial output', | ||||
TOPIC_CATEGORY_CONFIG: 'Mercurial configuration', | ||||
TOPIC_CATEGORY_CONCEPTS: 'Concepts', | ||||
TOPIC_CATEGORY_MISC: 'Miscellaneous', | ||||
TOPIC_CATEGORY_NONE: 'Uncategorized topics', | ||||
Rodrigo Damazio
|
r40328 | TOPIC_CATEGORY_NONE: 'Uncategorized topics', | ||
} | ||||
Augie Fackler
|
r20582 | def listexts(header, exts, indent=1, showdeprecated=False): | ||
Cédric Duval
|
r8879 | '''return a text listing of the given extensions''' | ||
Olav Reinert
|
r16852 | rst = [] | ||
if exts: | ||||
for name, desc in sorted(exts.iteritems()): | ||||
Yuya Nishihara
|
r26371 | if not showdeprecated and any(w in desc for w in _exclkeywords): | ||
Augie Fackler
|
r20582 | continue | ||
Olav Reinert
|
r16852 | rst.append('%s:%s: %s\n' % (' ' * indent, name, desc)) | ||
timeless
|
r27151 | if rst: | ||
rst.insert(0, '\n%s\n\n' % header) | ||||
Olav Reinert
|
r16852 | return rst | ||
Cédric Duval
|
r8864 | |||
Yuya Nishihara
|
r26413 | def extshelp(ui): | ||
rst = loaddoc('extensions')(ui).splitlines(True) | ||||
Augie Fackler
|
r20582 | rst.extend(listexts( | ||
_('enabled extensions:'), extensions.enabled(), showdeprecated=True)) | ||||
Yuya Nishihara
|
r36263 | rst.extend(listexts(_('disabled extensions:'), extensions.disabled(), | ||
showdeprecated=ui.verbose)) | ||||
Olav Reinert
|
r16852 | doc = ''.join(rst) | ||
Cédric Duval
|
r8863 | return doc | ||
Martin Geisler
|
r7013 | |||
Matt Mackall
|
r22116 | def optrst(header, options, verbose): | ||
Olav Reinert
|
r16781 | data = [] | ||
multioccur = False | ||||
for option in options: | ||||
if len(option) == 5: | ||||
shortopt, longopt, default, desc, optlabel = option | ||||
else: | ||||
shortopt, longopt, default, desc = option | ||||
optlabel = _("VALUE") # default label | ||||
Yuya Nishihara
|
r26369 | if not verbose and any(w in desc for w in _exclkeywords): | ||
Olav Reinert
|
r16781 | continue | ||
so = '' | ||||
if shortopt: | ||||
so = '-' + shortopt | ||||
lo = '--' + longopt | ||||
Daniel Ploch
|
r37109 | |||
if isinstance(default, fancyopts.customopt): | ||||
Daniel Ploch
|
r37110 | default = default.getdefaultvalue() | ||
Daniel Ploch
|
r37109 | if default and not callable(default): | ||
Augie Fackler
|
r32619 | # default is of unknown type, and in Python 2 we abused | ||
# the %s-shows-repr property to handle integers etc. To | ||||
# match that behavior on Python 3, we do str(default) and | ||||
# then convert it to bytes. | ||||
desc += _(" (default: %s)") % pycompat.bytestr(default) | ||||
Olav Reinert
|
r16781 | |||
if isinstance(default, list): | ||||
lo += " %s [+]" % optlabel | ||||
multioccur = True | ||||
elif (default is not None) and not isinstance(default, bool): | ||||
lo += " %s" % optlabel | ||||
data.append((so, lo, desc)) | ||||
Matt Mackall
|
r22117 | if multioccur: | ||
header += (_(" ([+] can be repeated)")) | ||||
Olav Reinert
|
r16781 | |||
Matt Mackall
|
r22116 | rst = ['\n%s:\n\n' % header] | ||
rst.extend(minirst.maketable(data, 1)) | ||||
Olav Reinert
|
r16781 | |||
Olav Reinert
|
r16815 | return ''.join(rst) | ||
Olav Reinert
|
r16781 | |||
FUJIWARA Katsunori
|
r17837 | def indicateomitted(rst, omitted, notomitted=None): | ||
rst.append('\n\n.. container:: omitted\n\n %s\n\n' % omitted) | ||||
if notomitted: | ||||
rst.append('\n\n.. container:: notomitted\n\n %s\n\n' % notomitted) | ||||
timeless
|
r27323 | def filtercmd(ui, cmd, kw, doc): | ||
if not ui.debugflag and cmd.startswith("debug") and kw != "debug": | ||||
return True | ||||
if not ui.verbose and doc and any(w in doc for w in _exclkeywords): | ||||
return True | ||||
return False | ||||
Yuya Nishihara
|
r32567 | def topicmatch(ui, commands, kw): | ||
Augie Fackler
|
r16710 | """Return help topics matching kw. | ||
Returns {'section': [(name, summary), ...], ...} where section is | ||||
one of topics, commands, extensions, or extensioncommands. | ||||
""" | ||||
kw = encoding.lower(kw) | ||||
def lowercontains(container): | ||||
Nikolaj Sjujskij
|
r16845 | return kw in encoding.lower(container) # translated in helptable | ||
Augie Fackler
|
r16710 | results = {'topics': [], | ||
'commands': [], | ||||
'extensions': [], | ||||
'extensioncommands': [], | ||||
} | ||||
Rodrigo Damazio
|
r40328 | for topic in helptable: | ||
names, header, doc = topic[0:3] | ||||
Gregory Szorc
|
r22322 | # Old extensions may use a str as doc. | ||
Augie Fackler
|
r16710 | if (sum(map(lowercontains, names)) | ||
or lowercontains(header) | ||||
Yuya Nishihara
|
r26413 | or (callable(doc) and lowercontains(doc(ui)))): | ||
Augie Fackler
|
r16710 | results['topics'].append((names[0], header)) | ||
for cmd, entry in commands.table.iteritems(): | ||||
if len(entry) == 3: | ||||
summary = entry[2] | ||||
else: | ||||
summary = '' | ||||
Nikolaj Sjujskij
|
r16845 | # translate docs *before* searching there | ||
Yuya Nishihara
|
r32615 | docs = _(pycompat.getdoc(entry[0])) or '' | ||
Augie Fackler
|
r16710 | if kw in cmd or lowercontains(summary) or lowercontains(docs): | ||
Nikolaj Sjujskij
|
r16845 | doclines = docs.splitlines() | ||
Augie Fackler
|
r16710 | if doclines: | ||
summary = doclines[0] | ||||
Yuya Nishihara
|
r36264 | cmdname = cmdutil.parsealiases(cmd)[0] | ||
timeless
|
r27324 | if filtercmd(ui, cmdname, kw, docs): | ||
continue | ||||
Augie Fackler
|
r16710 | results['commands'].append((cmdname, summary)) | ||
for name, docs in itertools.chain( | ||||
FUJIWARA Katsunori
|
r19769 | extensions.enabled(False).iteritems(), | ||
Augie Fackler
|
r16710 | extensions.disabled().iteritems()): | ||
Simon Farnsworth
|
r28058 | if not docs: | ||
continue | ||||
r26845 | name = name.rpartition('.')[-1] | |||
Augie Fackler
|
r16710 | if lowercontains(name) or lowercontains(docs): | ||
Nikolaj Sjujskij
|
r16845 | # extension docs are already translated | ||
results['extensions'].append((name, docs.splitlines()[0])) | ||||
Yuya Nishihara
|
r34913 | try: | ||
mod = extensions.load(ui, name, '') | ||||
except ImportError: | ||||
# debug message would be printed in extensions.load() | ||||
continue | ||||
Augie Fackler
|
r16710 | for cmd, entry in getattr(mod, 'cmdtable', {}).iteritems(): | ||
Augie Fackler
|
r16711 | if kw in cmd or (len(entry) > 2 and lowercontains(entry[2])): | ||
Yuya Nishihara
|
r36264 | cmdname = cmdutil.parsealiases(cmd)[0] | ||
Yuya Nishihara
|
r32615 | cmddoc = pycompat.getdoc(entry[0]) | ||
if cmddoc: | ||||
cmddoc = gettext(cmddoc).splitlines()[0] | ||||
Thomas Arendsen Hein
|
r16884 | else: | ||
cmddoc = _('(no help text available)') | ||||
timeless
|
r27387 | if filtercmd(ui, cmdname, kw, cmddoc): | ||
continue | ||||
Thomas Arendsen Hein
|
r16884 | results['extensioncommands'].append((cmdname, cmddoc)) | ||
Augie Fackler
|
r16710 | return results | ||
Gregory Szorc
|
r27375 | def loaddoc(topic, subdir=None): | ||
Martin Geisler
|
r9539 | """Return a delayed loader for help/topic.txt.""" | ||
Matt Mackall
|
r3798 | |||
Yuya Nishihara
|
r26413 | def loader(ui): | ||
Mads Kiilerich
|
r22637 | docdir = os.path.join(util.datapath, 'help') | ||
Gregory Szorc
|
r27375 | if subdir: | ||
docdir = os.path.join(docdir, subdir) | ||||
Martin Geisler
|
r9539 | path = os.path.join(docdir, topic + ".txt") | ||
Dan Villiom Podlaski Christiansen
|
r14168 | doc = gettext(util.readfile(path)) | ||
Patrick Mezard
|
r12820 | for rewriter in helphooks.get(topic, []): | ||
Yuya Nishihara
|
r26414 | doc = rewriter(ui, topic, doc) | ||
Patrick Mezard
|
r12820 | return doc | ||
Martin Geisler
|
r9539 | return loader | ||
Alexander Solovyov
|
r7677 | |||
Gregory Szorc
|
r27376 | internalstable = sorted([ | ||
Gregory Szorc
|
r36469 | (['bundle2'], _('Bundle2'), | ||
loaddoc('bundle2', subdir='internals')), | ||||
Gregory Szorc
|
r29747 | (['bundles'], _('Bundles'), | ||
Gregory Szorc
|
r27376 | loaddoc('bundles', subdir='internals')), | ||
Gregory Szorc
|
r39446 | (['cbor'], _('CBOR'), | ||
loaddoc('cbor', subdir='internals')), | ||||
Augie Fackler
|
r31293 | (['censor'], _('Censor'), | ||
loaddoc('censor', subdir='internals')), | ||||
Gregory Szorc
|
r29747 | (['changegroups'], _('Changegroups'), | ||
Gregory Szorc
|
r27376 | loaddoc('changegroups', subdir='internals')), | ||
Kevin Bullock
|
r34952 | (['config'], _('Config Registrar'), | ||
Boris Feld
|
r34933 | loaddoc('config', subdir='internals')), | ||
Gregory Szorc
|
r29747 | (['requirements'], _('Repository Requirements'), | ||
Gregory Szorc
|
r28523 | loaddoc('requirements', subdir='internals')), | ||
Gregory Szorc
|
r29747 | (['revlogs'], _('Revision Logs'), | ||
Gregory Szorc
|
r27631 | loaddoc('revlogs', subdir='internals')), | ||
Gregory Szorc
|
r29859 | (['wireprotocol'], _('Wire Protocol'), | ||
loaddoc('wireprotocol', subdir='internals')), | ||||
Yuya Nishihara
|
r39650 | (['wireprotocolrpc'], _('Wire Protocol RPC'), | ||
loaddoc('wireprotocolrpc', subdir='internals')), | ||||
Yuya Nishihara
|
r39497 | (['wireprotocolv2'], _('Wire Protocol Version 2'), | ||
loaddoc('wireprotocolv2', subdir='internals')), | ||||
Gregory Szorc
|
r27376 | ]) | ||
def internalshelp(ui): | ||||
"""Generate the index for the "internals" topic.""" | ||||
Matt DeVore
|
r32076 | lines = ['To access a subtopic, use "hg help internals.{subtopic-name}"\n', | ||
'\n'] | ||||
Gregory Szorc
|
r27376 | for names, header, doc in internalstable: | ||
Yuya Nishihara
|
r27400 | lines.append(' :%s: %s\n' % (names[0], header)) | ||
Gregory Szorc
|
r27376 | |||
Yuya Nishihara
|
r27400 | return ''.join(lines) | ||
Gregory Szorc
|
r27376 | |||
Yun Lee
|
r13888 | helptable = sorted([ | ||
Rodrigo Damazio
|
r40330 | (['bundlespec'], _("Bundle File Formats"), loaddoc('bundlespec'), | ||
TOPIC_CATEGORY_CONCEPTS), | ||||
(['color'], _("Colorizing Outputs"), loaddoc('color'), | ||||
TOPIC_CATEGORY_OUTPUT), | ||||
(["config", "hgrc"], _("Configuration Files"), loaddoc('config'), | ||||
TOPIC_CATEGORY_CONFIG), | ||||
(['deprecated'], _("Deprecated Features"), loaddoc('deprecated'), | ||||
TOPIC_CATEGORY_MISC), | ||||
(["dates"], _("Date Formats"), loaddoc('dates'), TOPIC_CATEGORY_OUTPUT), | ||||
(["flags"], _("Command-line flags"), loaddoc('flags'), | ||||
TOPIC_CATEGORY_CONFIG), | ||||
(["patterns"], _("File Name Patterns"), loaddoc('patterns'), | ||||
TOPIC_CATEGORY_IDS), | ||||
Matt Mackall
|
r10282 | (['environment', 'env'], _('Environment Variables'), | ||
Rodrigo Damazio
|
r40330 | loaddoc('environment'), TOPIC_CATEGORY_CONFIG), | ||
Martin von Zweigbergk
|
r30769 | (['revisions', 'revs', 'revsets', 'revset', 'multirevs', 'mrevs'], | ||
Rodrigo Damazio
|
r40330 | _('Specifying Revisions'), loaddoc('revisions'), TOPIC_CATEGORY_IDS), | ||
(['filesets', 'fileset'], _("Specifying File Sets"), loaddoc('filesets'), | ||||
TOPIC_CATEGORY_IDS), | ||||
(['diffs'], _('Diff Formats'), loaddoc('diffs'), TOPIC_CATEGORY_OUTPUT), | ||||
Gregory Szorc
|
r30812 | (['merge-tools', 'mergetools', 'mergetool'], _('Merge Tools'), | ||
Rodrigo Damazio
|
r40330 | loaddoc('merge-tools'), TOPIC_CATEGORY_CONFIG), | ||
A. S. Budden
|
r16568 | (['templating', 'templates', 'template', 'style'], _('Template Usage'), | ||
Rodrigo Damazio
|
r40330 | loaddoc('templates'), TOPIC_CATEGORY_OUTPUT), | ||
(['urls'], _('URL Paths'), loaddoc('urls'), TOPIC_CATEGORY_IDS), | ||||
(["extensions"], _("Using Additional Features"), extshelp, | ||||
TOPIC_CATEGORY_CONFIG), | ||||
(["subrepos", "subrepo"], _("Subrepositories"), loaddoc('subrepos'), | ||||
TOPIC_CATEGORY_CONCEPTS), | ||||
(["hgweb"], _("Configuring hgweb"), loaddoc('hgweb'), | ||||
TOPIC_CATEGORY_CONFIG), | ||||
(["glossary"], _("Glossary"), loaddoc('glossary'), TOPIC_CATEGORY_CONCEPTS), | ||||
Mads Kiilerich
|
r17321 | (["hgignore", "ignore"], _("Syntax for Mercurial Ignore Files"), | ||
Rodrigo Damazio
|
r40330 | loaddoc('hgignore'), TOPIC_CATEGORY_IDS), | ||
(["phases"], _("Working with Phases"), loaddoc('phases'), | ||||
TOPIC_CATEGORY_CONCEPTS), | ||||
Gregory Szorc
|
r25881 | (['scripting'], _('Using Mercurial from scripts and automation'), | ||
Rodrigo Damazio
|
r40330 | loaddoc('scripting'), TOPIC_CATEGORY_MISC), | ||
(['internals'], _("Technical implementation topics"), internalshelp, | ||||
TOPIC_CATEGORY_MISC), | ||||
(['pager'], _("Pager Support"), loaddoc('pager'), TOPIC_CATEGORY_CONFIG), | ||||
Yun Lee
|
r13888 | ]) | ||
Patrick Mezard
|
r12820 | |||
Gregory Szorc
|
r27379 | # Maps topics with sub-topics to a list of their sub-topics. | ||
subtopics = { | ||||
'internals': internalstable, | ||||
} | ||||
Patrick Mezard
|
r12820 | # Map topics to lists of callable taking the current topic help and | ||
# returning the updated version | ||||
Matt Mackall
|
r14318 | helphooks = {} | ||
Patrick Mezard
|
r12820 | |||
def addtopichook(topic, rewriter): | ||||
helphooks.setdefault(topic, []).append(rewriter) | ||||
Patrick Mezard
|
r13593 | |||
Yuya Nishihara
|
r26414 | def makeitemsdoc(ui, topic, doc, marker, items, dedent=False): | ||
Patrick Mezard
|
r13593 | """Extract docstring from the items key to function mapping, build a | ||
timeless@mozdev.org
|
r26196 | single documentation block and use it to overwrite the marker in doc. | ||
Patrick Mezard
|
r13593 | """ | ||
entries = [] | ||||
for name in sorted(items): | ||||
Yuya Nishihara
|
r32615 | text = (pycompat.getdoc(items[name]) or '').rstrip() | ||
Yuya Nishihara
|
r26415 | if (not text | ||
or not ui.verbose and any(w in text for w in _exclkeywords)): | ||||
Patrick Mezard
|
r13593 | continue | ||
text = gettext(text) | ||||
Gregory Szorc
|
r24098 | if dedent: | ||
Augie Fackler
|
r32549 | # Abuse latin1 to use textwrap.dedent() on bytes. | ||
text = textwrap.dedent(text.decode('latin1')).encode('latin1') | ||||
Patrick Mezard
|
r13593 | lines = text.splitlines() | ||
"Yann E. MORIN"
|
r16250 | doclines = [(lines[0])] | ||
for l in lines[1:]: | ||||
# Stop once we find some Python doctest | ||||
if l.strip().startswith('>>>'): | ||||
break | ||||
Gregory Szorc
|
r24098 | if dedent: | ||
doclines.append(l.rstrip()) | ||||
else: | ||||
doclines.append(' ' + l.strip()) | ||||
"Yann E. MORIN"
|
r16250 | entries.append('\n'.join(doclines)) | ||
Patrick Mezard
|
r13593 | entries = '\n\n'.join(entries) | ||
return doc.replace(marker, entries) | ||||
Matt Mackall
|
r14318 | |||
Gregory Szorc
|
r24098 | def addtopicsymbols(topic, marker, symbols, dedent=False): | ||
Yuya Nishihara
|
r26414 | def add(ui, topic, doc): | ||
return makeitemsdoc(ui, topic, doc, marker, symbols, dedent=dedent) | ||||
Matt Mackall
|
r14318 | addtopichook(topic, add) | ||
Gregory Szorc
|
r31793 | addtopicsymbols('bundlespec', '.. bundlecompressionmarker', | ||
util.bundlecompressiontopics()) | ||||
Matt Mackall
|
r14686 | addtopicsymbols('filesets', '.. predicatesmarker', fileset.symbols) | ||
Gregory Szorc
|
r24099 | addtopicsymbols('merge-tools', '.. internaltoolsmarker', | ||
filemerge.internalsdoc) | ||||
Martin von Zweigbergk
|
r30769 | addtopicsymbols('revisions', '.. predicatesmarker', revset.symbols) | ||
Yuya Nishihara
|
r26436 | addtopicsymbols('templates', '.. keywordsmarker', templatekw.keywords) | ||
Matt Mackall
|
r14318 | addtopicsymbols('templates', '.. filtersmarker', templatefilters.filters) | ||
Yuya Nishihara
|
r36940 | addtopicsymbols('templates', '.. functionsmarker', templatefuncs.funcs) | ||
Gregory Szorc
|
r24098 | addtopicsymbols('hgweb', '.. webcommandsmarker', webcommands.commands, | ||
dedent=True) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
Yuya Nishihara
|
r32567 | def help_(ui, commands, name, unknowncmd=False, full=True, subtopic=None, | ||
**opts): | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | ''' | ||
Generate the help for 'name' as unformatted restructured text. If | ||||
'name' is None, describe the commands available. | ||||
''' | ||||
Pulkit Goyal
|
r32143 | opts = pycompat.byteskwargs(opts) | ||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
Gregory Szorc
|
r27378 | def helpcmd(name, subtopic=None): | ||
Dan Villiom Podlaski Christiansen
|
r18746 | try: | ||
aliases, entry = cmdutil.findcmd(name, commands.table, | ||||
strict=unknowncmd) | ||||
Gregory Szorc
|
r25660 | except error.AmbiguousCommand as inst: | ||
Martijn Pieters
|
r40299 | # py3 fix: except vars can't be used outside the scope of the | ||
Dan Villiom Podlaski Christiansen
|
r18746 | # except block, nor can be used inside a lambda. python issue4617 | ||
prefix = inst.args[0] | ||||
Yuya Nishihara
|
r36264 | select = lambda c: cmdutil.parsealiases(c)[0].startswith(prefix) | ||
Dan Villiom Podlaski Christiansen
|
r18746 | rst = helplist(select) | ||
return rst | ||||
rst = [] | ||||
# check if it's an invalid alias and display its error if it is | ||||
Yuya Nishihara
|
r22160 | if getattr(entry[0], 'badalias', None): | ||
Yuya Nishihara
|
r22162 | rst.append(entry[0].badalias + '\n') | ||
if entry[0].unknowncmd: | ||||
try: | ||||
rst.extend(helpextcmd(entry[0].cmdname)) | ||||
except error.UnknownCommand: | ||||
pass | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | return rst | ||
# synopsis | ||||
if len(entry) > 2: | ||||
if entry[2].startswith('hg'): | ||||
rst.append("%s\n" % entry[2]) | ||||
else: | ||||
rst.append('hg %s %s\n' % (aliases[0], entry[2])) | ||||
else: | ||||
rst.append('hg %s\n' % aliases[0]) | ||||
# aliases | ||||
if full and not ui.quiet and len(aliases) > 1: | ||||
rst.append(_("\naliases: %s\n") % ', '.join(aliases[1:])) | ||||
rst.append('\n') | ||||
# description | ||||
Yuya Nishihara
|
r32615 | doc = gettext(pycompat.getdoc(entry[0])) | ||
Dan Villiom Podlaski Christiansen
|
r18746 | if not doc: | ||
doc = _("(no help text available)") | ||||
if util.safehasattr(entry[0], 'definition'): # aliased command | ||||
timeless
|
r28828 | source = entry[0].source | ||
Dan Villiom Podlaski Christiansen
|
r18746 | if entry[0].definition.startswith('!'): # shell alias | ||
Rodrigo Damazio
|
r37152 | doc = (_('shell alias for: %s\n\n%s\n\ndefined by: %s\n') % | ||
(entry[0].definition[1:], doc, source)) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | else: | ||
timeless
|
r28828 | doc = (_('alias for: hg %s\n\n%s\n\ndefined by: %s\n') % | ||
(entry[0].definition, doc, source)) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | doc = doc.splitlines(True) | ||
if ui.quiet or not full: | ||||
rst.append(doc[0]) | ||||
else: | ||||
rst.extend(doc) | ||||
rst.append('\n') | ||||
# check if this command shadows a non-trivial (multi-line) | ||||
# extension help text | ||||
try: | ||||
mod = extensions.find(name) | ||||
Yuya Nishihara
|
r32615 | doc = gettext(pycompat.getdoc(mod)) or '' | ||
Dan Villiom Podlaski Christiansen
|
r18746 | if '\n' in doc.strip(): | ||
timeless
|
r29974 | msg = _("(use 'hg help -e %s' to show help for " | ||
"the %s extension)") % (name, name) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | rst.append('\n%s\n' % msg) | ||
except KeyError: | ||||
pass | ||||
# options | ||||
if not ui.quiet and entry[1]: | ||||
Matt Mackall
|
r22116 | rst.append(optrst(_("options"), entry[1], ui.verbose)) | ||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
if ui.verbose: | ||||
Matt Mackall
|
r22116 | rst.append(optrst(_("global options"), | ||
commands.globalopts, ui.verbose)) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
if not ui.verbose: | ||||
if not full: | ||||
timeless
|
r29974 | rst.append(_("\n(use 'hg %s -h' to show more help)\n") | ||
Dan Villiom Podlaski Christiansen
|
r18746 | % name) | ||
elif not ui.quiet: | ||||
Matt Mackall
|
r22110 | rst.append(_('\n(some details hidden, use --verbose ' | ||
'to show complete help)')) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
return rst | ||||
timeless
|
r27325 | def helplist(select=None, **opts): | ||
rdamazio@google.com
|
r40327 | # Category -> list of commands | ||
cats = {} | ||||
# Command -> short description | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | h = {} | ||
rdamazio@google.com
|
r40327 | # Command -> string showing synonyms | ||
syns = {} | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | for c, e in commands.table.iteritems(): | ||
Yuya Nishihara
|
r36264 | fs = cmdutil.parsealiases(c) | ||
f = fs[0] | ||||
rdamazio@google.com
|
r40327 | syns[f] = ', '.join(fs) | ||
func = e[0] | ||||
Rodrigo Damazio
|
r40331 | if select and not select(f): | ||
Dan Villiom Podlaski Christiansen
|
r18746 | continue | ||
if (not select and name != 'shortlist' and | ||||
rdamazio@google.com
|
r40327 | func.__module__ != commands.__name__): | ||
Dan Villiom Podlaski Christiansen
|
r18746 | continue | ||
Rodrigo Damazio
|
r40331 | if name == "shortlist": | ||
if not getattr(func, 'helpbasic', False): | ||||
continue | ||||
rdamazio@google.com
|
r40327 | doc = pycompat.getdoc(func) | ||
timeless
|
r27323 | if filtercmd(ui, f, name, doc): | ||
Dan Villiom Podlaski Christiansen
|
r18746 | continue | ||
doc = gettext(doc) | ||||
if not doc: | ||||
doc = _("(no help text available)") | ||||
h[f] = doc.splitlines()[0].rstrip() | ||||
rdamazio@google.com
|
r40327 | |||
cat = getattr(func, 'helpcategory', None) or ( | ||||
registrar.command.CATEGORY_NONE) | ||||
cats.setdefault(cat, []).append(f) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
rst = [] | ||||
if not h: | ||||
if not ui.quiet: | ||||
rst.append(_('no commands defined\n')) | ||||
return rst | ||||
rdamazio@google.com
|
r40327 | # Output top header. | ||
Dan Villiom Podlaski Christiansen
|
r18746 | if not ui.quiet: | ||
rdamazio@google.com
|
r40327 | if name == "shortlist": | ||
rst.append(_('basic commands:\n\n')) | ||||
elif name == "debug": | ||||
rst.append(_('debug commands (internal and unsupported):\n\n')) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | else: | ||
rdamazio@google.com
|
r40327 | rst.append(_('list of commands:\n')) | ||
def appendcmds(cmds): | ||||
cmds = sorted(cmds) | ||||
for c in cmds: | ||||
if ui.verbose: | ||||
rst.append(" :%s: %s\n" % (syns[c], h[c])) | ||||
else: | ||||
rst.append(' :%s: %s\n' % (c, h[c])) | ||||
if name in ('shortlist', 'debug'): | ||||
# List without categories. | ||||
appendcmds(h) | ||||
else: | ||||
# Check that all categories have an order. | ||||
missing_order = set(cats.keys()) - set(CATEGORY_ORDER) | ||||
if missing_order: | ||||
ui.develwarn('help categories missing from CATEGORY_ORDER: %s' % | ||||
missing_order) | ||||
# List per category. | ||||
for cat in CATEGORY_ORDER: | ||||
catfns = cats.get(cat, []) | ||||
if catfns: | ||||
if len(cats) > 1: | ||||
catname = gettext(CATEGORY_NAMES[cat]) | ||||
rst.append("\n%s:\n" % catname) | ||||
rst.append("\n") | ||||
appendcmds(catfns) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
timeless
|
r27325 | ex = opts.get | ||
Pulkit Goyal
|
r35357 | anyopts = (ex(r'keyword') or not (ex(r'command') or ex(r'extension'))) | ||
timeless
|
r27325 | if not name and anyopts: | ||
Dan Villiom Podlaski Christiansen
|
r18746 | exts = listexts(_('enabled extensions:'), extensions.enabled()) | ||
if exts: | ||||
rst.append('\n') | ||||
rst.extend(exts) | ||||
Rodrigo Damazio
|
r40328 | rst.append(_("\nadditional help topics:\n")) | ||
# Group commands by category. | ||||
topiccats = {} | ||||
for topic in helptable: | ||||
names, header, doc = topic[0:3] | ||||
if len(topic) > 3 and topic[3]: | ||||
category = topic[3] | ||||
else: | ||||
category = TOPIC_CATEGORY_NONE | ||||
topiccats.setdefault(category, []).append((names[0], header)) | ||||
# Check that all categories have an order. | ||||
missing_order = set(topiccats.keys()) - set(TOPIC_CATEGORY_ORDER) | ||||
if missing_order: | ||||
ui.develwarn( | ||||
'help categories missing from TOPIC_CATEGORY_ORDER: %s' % | ||||
missing_order) | ||||
# Output topics per category. | ||||
for cat in TOPIC_CATEGORY_ORDER: | ||||
topics = topiccats.get(cat, []) | ||||
if topics: | ||||
if len(topiccats) > 1: | ||||
catname = gettext(TOPIC_CATEGORY_NAMES[cat]) | ||||
rst.append("\n%s:\n" % catname) | ||||
rst.append("\n") | ||||
for t, desc in topics: | ||||
rst.append(" :%s: %s\n" % (t, desc)) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
Matt Mackall
|
r22115 | if ui.quiet: | ||
pass | ||||
elif ui.verbose: | ||||
Matt Mackall
|
r22116 | rst.append('\n%s\n' % optrst(_("global options"), | ||
commands.globalopts, ui.verbose)) | ||||
Matt Mackall
|
r22115 | if name == 'shortlist': | ||
timeless
|
r29974 | rst.append(_("\n(use 'hg help' for the full list " | ||
"of commands)\n")) | ||||
Matt Mackall
|
r22115 | else: | ||
if name == 'shortlist': | ||||
timeless
|
r29974 | rst.append(_("\n(use 'hg help' for the full list of commands " | ||
"or 'hg -v' for details)\n")) | ||||
Matt Mackall
|
r22115 | elif name and not full: | ||
timeless
|
r29974 | rst.append(_("\n(use 'hg help %s' to show the full help " | ||
"text)\n") % name) | ||||
rdamazio@google.com
|
r40327 | elif name and syns and name in syns.keys(): | ||
timeless
|
r29974 | rst.append(_("\n(use 'hg help -v -e %s' to show built-in " | ||
"aliases and global options)\n") % name) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | else: | ||
timeless
|
r29974 | rst.append(_("\n(use 'hg help -v%s' to show built-in aliases " | ||
"and global options)\n") | ||||
Matt Mackall
|
r22118 | % (name and " " + name or "")) | ||
Dan Villiom Podlaski Christiansen
|
r18746 | return rst | ||
Gregory Szorc
|
r27378 | def helptopic(name, subtopic=None): | ||
Gregory Szorc
|
r27379 | # Look for sub-topic entry first. | ||
header, doc = None, None | ||||
if subtopic and name in subtopics: | ||||
for names, header, doc in subtopics[name]: | ||||
if subtopic in names: | ||||
break | ||||
if not header: | ||||
Rodrigo Damazio
|
r40328 | for topic in helptable: | ||
names, header, doc = topic[0:3] | ||||
Gregory Szorc
|
r27379 | if name in names: | ||
break | ||||
else: | ||||
raise error.UnknownCommand(name) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
Dan Villiom Podlaski Christiansen
|
r18748 | rst = [minirst.section(header)] | ||
Dan Villiom Podlaski Christiansen
|
r18746 | # description | ||
if not doc: | ||||
rst.append(" %s\n" % _("(no help text available)")) | ||||
Augie Fackler
|
r21796 | if callable(doc): | ||
Yuya Nishihara
|
r26413 | rst += [" %s\n" % l for l in doc(ui).splitlines()] | ||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
if not ui.verbose: | ||||
Matt Mackall
|
r22114 | omitted = _('(some details hidden, use --verbose' | ||
' to show complete help)') | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | indicateomitted(rst, omitted) | ||
try: | ||||
cmdutil.findcmd(name, commands.table) | ||||
timeless
|
r29974 | rst.append(_("\nuse 'hg help -c %s' to see help for " | ||
"the %s command\n") % (name, name)) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | except error.UnknownCommand: | ||
pass | ||||
return rst | ||||
Gregory Szorc
|
r27378 | def helpext(name, subtopic=None): | ||
Dan Villiom Podlaski Christiansen
|
r18746 | try: | ||
mod = extensions.find(name) | ||||
Yuya Nishihara
|
r32615 | doc = gettext(pycompat.getdoc(mod)) or _('no help text available') | ||
Dan Villiom Podlaski Christiansen
|
r18746 | except KeyError: | ||
mod = None | ||||
doc = extensions.disabledext(name) | ||||
if not doc: | ||||
raise error.UnknownCommand(name) | ||||
if '\n' not in doc: | ||||
head, tail = doc, "" | ||||
else: | ||||
head, tail = doc.split('\n', 1) | ||||
r26845 | rst = [_('%s extension - %s\n\n') % (name.rpartition('.')[-1], head)] | |||
Dan Villiom Podlaski Christiansen
|
r18746 | if tail: | ||
rst.extend(tail.splitlines(True)) | ||||
rst.append('\n') | ||||
if not ui.verbose: | ||||
Matt Mackall
|
r22114 | omitted = _('(some details hidden, use --verbose' | ||
' to show complete help)') | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | indicateomitted(rst, omitted) | ||
if mod: | ||||
try: | ||||
ct = mod.cmdtable | ||||
except AttributeError: | ||||
ct = {} | ||||
r26845 | modcmds = set([c.partition('|')[0] for c in ct]) | |||
Dan Villiom Podlaski Christiansen
|
r18746 | rst.extend(helplist(modcmds.__contains__)) | ||
else: | ||||
timeless
|
r29974 | rst.append(_("(use 'hg help extensions' for information on enabling" | ||
" extensions)\n")) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | return rst | ||
Gregory Szorc
|
r27378 | def helpextcmd(name, subtopic=None): | ||
Yuya Nishihara
|
r37996 | cmd, ext, doc = extensions.disabledcmd(ui, name, | ||
Dan Villiom Podlaski Christiansen
|
r18746 | ui.configbool('ui', 'strict')) | ||
Yuya Nishihara
|
r37996 | doc = doc.splitlines()[0] | ||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
rst = listexts(_("'%s' is provided by the following " | ||||
timeless
|
r27152 | "extension:") % cmd, {ext: doc}, indent=4, | ||
showdeprecated=True) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | rst.append('\n') | ||
timeless
|
r29974 | rst.append(_("(use 'hg help extensions' for information on enabling " | ||
"extensions)\n")) | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | return rst | ||
rst = [] | ||||
kw = opts.get('keyword') | ||||
timeless
|
r27325 | if kw or name is None and any(opts[o] for o in opts): | ||
Yuya Nishihara
|
r32567 | matches = topicmatch(ui, commands, name or '') | ||
timeless@mozdev.org
|
r26238 | helpareas = [] | ||
if opts.get('extension'): | ||||
helpareas += [('extensions', _('Extensions'))] | ||||
if opts.get('command'): | ||||
helpareas += [('commands', _('Commands'))] | ||||
if not helpareas: | ||||
helpareas = [('topics', _('Topics')), | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | ('commands', _('Commands')), | ||
('extensions', _('Extensions')), | ||||
timeless@mozdev.org
|
r26238 | ('extensioncommands', _('Extension Commands'))] | ||
for t, title in helpareas: | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | if matches[t]: | ||
rst.append('%s:\n\n' % title) | ||||
rst.extend(minirst.maketable(sorted(matches[t]), 1)) | ||||
rst.append('\n') | ||||
Pierre-Yves David
|
r21288 | if not rst: | ||
msg = _('no matches') | ||||
timeless
|
r29974 | hint = _("try 'hg help' for a list of topics") | ||
Pierre-Yves David
|
r26587 | raise error.Abort(msg, hint=hint) | ||
Dan Villiom Podlaski Christiansen
|
r18746 | elif name and name != 'shortlist': | ||
timeless@mozdev.org
|
r26238 | queries = [] | ||
Dan Villiom Podlaski Christiansen
|
r18746 | if unknowncmd: | ||
timeless@mozdev.org
|
r26238 | queries += [helpextcmd] | ||
if opts.get('extension'): | ||||
queries += [helpext] | ||||
if opts.get('command'): | ||||
queries += [helpcmd] | ||||
if not queries: | ||||
Dan Villiom Podlaski Christiansen
|
r18746 | queries = (helptopic, helpcmd, helpext, helpextcmd) | ||
for f in queries: | ||||
try: | ||||
Gregory Szorc
|
r27378 | rst = f(name, subtopic) | ||
Dan Villiom Podlaski Christiansen
|
r18746 | break | ||
Pierre-Yves David
|
r21289 | except error.UnknownCommand: | ||
pass | ||||
else: | ||||
if unknowncmd: | ||||
raise error.UnknownCommand(name) | ||||
else: | ||||
msg = _('no such help topic: %s') % name | ||||
timeless
|
r29974 | hint = _("try 'hg help --keyword %s'") % name | ||
Pierre-Yves David
|
r26587 | raise error.Abort(msg, hint=hint) | ||
Dan Villiom Podlaski Christiansen
|
r18746 | else: | ||
# program name | ||||
if not ui.quiet: | ||||
rst = [_("Mercurial Distributed SCM\n"), '\n'] | ||||
Augie Fackler
|
r32547 | rst.extend(helplist(None, **pycompat.strkwargs(opts))) | ||
Dan Villiom Podlaski Christiansen
|
r18746 | |||
return ''.join(rst) | ||||
Augie Fackler
|
r31059 | |||
Yuya Nishihara
|
r39375 | def formattedhelp(ui, commands, fullname, keep=None, unknowncmd=False, | ||
full=True, **opts): | ||||
Augie Fackler
|
r31059 | """get help for a given topic (as a dotted name) as rendered rst | ||
Either returns the rendered help text or raises an exception. | ||||
""" | ||||
if keep is None: | ||||
keep = [] | ||||
Augie Fackler
|
r31265 | else: | ||
keep = list(keep) # make a copy so we can mutate this later | ||||
Yuya Nishihara
|
r39375 | |||
# <fullname> := <name>[.<subtopic][.<section>] | ||||
name = subtopic = section = None | ||||
if fullname is not None: | ||||
nameparts = fullname.split('.') | ||||
name = nameparts.pop(0) | ||||
if nameparts and name in subtopics: | ||||
subtopic = nameparts.pop(0) | ||||
if nameparts: | ||||
section = encoding.lower('.'.join(nameparts)) | ||||
Jun Wu
|
r33499 | textwidth = ui.configint('ui', 'textwidth') | ||
Augie Fackler
|
r31059 | termwidth = ui.termwidth() - 2 | ||
if textwidth <= 0 or termwidth < textwidth: | ||||
textwidth = termwidth | ||||
Yuya Nishihara
|
r32567 | text = help_(ui, commands, name, | ||
Augie Fackler
|
r31059 | subtopic=subtopic, unknowncmd=unknowncmd, full=full, **opts) | ||
Yuya Nishihara
|
r39344 | blocks, pruned = minirst.parse(text, keep=keep) | ||
Augie Fackler
|
r31059 | if 'verbose' in pruned: | ||
keep.append('omitted') | ||||
else: | ||||
keep.append('notomitted') | ||||
Yuya Nishihara
|
r39344 | blocks, pruned = minirst.parse(text, keep=keep) | ||
if section: | ||||
blocks = minirst.filtersections(blocks, section) | ||||
Yuya Nishihara
|
r39345 | |||
# We could have been given a weird ".foo" section without a name | ||||
# to look for, or we could have simply failed to found "foo.bar" | ||||
# because bar isn't a section of foo | ||||
if section and not (blocks and name): | ||||
raise error.Abort(_("help section not found: %s") % fullname) | ||||
Yuya Nishihara
|
r39344 | return minirst.formatplain(blocks, textwidth) | ||