gendoc.py
240 lines
| 7.2 KiB
| text/x-python
|
PythonLexer
/ doc / gendoc.py
timeless
|
r27496 | #!/usr/bin/env python | ||
Takumi IINO
|
r19425 | """usage: %s DOC ... | ||
where DOC is the name of a document | ||||
""" | ||||
Pulkit Goyal
|
r28966 | from __future__ import absolute_import | ||
import os | ||||
import sys | ||||
import textwrap | ||||
Gregory Szorc
|
r27330 | |||
Matt Harbison
|
r41040 | try: | ||
import msvcrt | ||||
msvcrt.setmode(sys.stdout.fileno(), os.O_BINARY) | ||||
msvcrt.setmode(sys.stderr.fileno(), os.O_BINARY) | ||||
except ImportError: | ||||
pass | ||||
Gregory Szorc
|
r27330 | # This script is executed during installs and may not have C extensions | ||
# available. Relax C module requirements. | ||||
Matt Harbison
|
r41040 | os.environ[r'HGMODULEPOLICY'] = r'allow' | ||
Benoit Boissinot
|
r1814 | # import from the live mercurial repo | ||
Matt Harbison
|
r41040 | sys.path.insert(0, r"..") | ||
Matt Mackall
|
r5209 | from mercurial import demandimport; demandimport.enable() | ||
Yuya Nishihara
|
r32336 | # Load util so that the locale path is set by i18n.setdatapath() before | ||
# calling _(). | ||||
from mercurial import util | ||||
util.datapath | ||||
Pulkit Goyal
|
r28966 | from mercurial import ( | ||
commands, | ||||
Matt Harbison
|
r41059 | encoding, | ||
Pulkit Goyal
|
r28966 | extensions, | ||
help, | ||||
minirst, | ||||
Matt Harbison
|
r41040 | pycompat, | ||
Pulkit Goyal
|
r28966 | ui as uimod, | ||
) | ||||
from mercurial.i18n import ( | ||||
gettext, | ||||
_, | ||||
) | ||||
table = commands.table | ||||
globalopts = commands.globalopts | ||||
helptable = help.helptable | ||||
loaddoc = help.loaddoc | ||||
Benoit Boissinot
|
r1814 | |||
def get_desc(docstr): | ||||
if not docstr: | ||||
Matt Harbison
|
r41040 | return b"", b"" | ||
Benoit Boissinot
|
r1814 | # sanitize | ||
Matt Harbison
|
r41040 | docstr = docstr.strip(b"\n") | ||
Benoit Boissinot
|
r1814 | docstr = docstr.rstrip() | ||
shortdesc = docstr.splitlines()[0].strip() | ||||
Matt Harbison
|
r41040 | i = docstr.find(b"\n") | ||
Benoit Boissinot
|
r1814 | if i != -1: | ||
Matt Mackall
|
r10282 | desc = docstr[i + 2:] | ||
Benoit Boissinot
|
r1814 | else: | ||
Erik Zielke
|
r12780 | desc = shortdesc | ||
Matt Harbison
|
r41040 | desc = textwrap.dedent(desc.decode('latin1')).encode('latin1') | ||
Erik Zielke
|
r12780 | |||
Benoit Boissinot
|
r1814 | return (shortdesc, desc) | ||
def get_opts(opts): | ||||
FUJIWARA Katsunori
|
r11321 | for opt in opts: | ||
if len(opt) == 5: | ||||
shortopt, longopt, default, desc, optlabel = opt | ||||
else: | ||||
shortopt, longopt, default, desc = opt | ||||
Matt Harbison
|
r41040 | optlabel = _(b"VALUE") | ||
Benoit Boissinot
|
r1814 | allopts = [] | ||
if shortopt: | ||||
Matt Harbison
|
r41040 | allopts.append(b"-%s" % shortopt) | ||
Benoit Boissinot
|
r1814 | if longopt: | ||
Matt Harbison
|
r41040 | allopts.append(b"--%s" % longopt) | ||
FUJIWARA Katsunori
|
r20081 | if isinstance(default, list): | ||
Matt Harbison
|
r41040 | allopts[-1] += b" <%s[+]>" % optlabel | ||
FUJIWARA Katsunori
|
r20081 | elif (default is not None) and not isinstance(default, bool): | ||
Matt Harbison
|
r41040 | allopts[-1] += b" <%s>" % optlabel | ||
if b'\n' in desc: | ||||
Simon Heimberg
|
r20655 | # only remove line breaks and indentation | ||
Matt Harbison
|
r41040 | desc = b' '.join(l.lstrip() for l in desc.split(b'\n')) | ||
desc += default and _(b" (default: %s)") % bytes(default) or b"" | ||||
yield (b", ".join(allopts), desc) | ||||
Benoit Boissinot
|
r1814 | |||
Erik Zielke
|
r12756 | def get_cmd(cmd, cmdtable): | ||
Benoit Boissinot
|
r1814 | d = {} | ||
Erik Zielke
|
r12756 | attr = cmdtable[cmd] | ||
Matt Harbison
|
r41040 | cmds = cmd.lstrip(b"^").split(b"|") | ||
Benoit Boissinot
|
r1814 | |||
Matt Harbison
|
r41040 | d[b'cmd'] = cmds[0] | ||
d[b'aliases'] = cmd.split(b"|")[1:] | ||||
d[b'desc'] = get_desc(gettext(pycompat.getdoc(attr[0]))) | ||||
d[b'opts'] = list(get_opts(attr[1])) | ||||
Matt Mackall
|
r7376 | |||
Matt Harbison
|
r41040 | s = b'hg ' + cmds[0] | ||
Matt Mackall
|
r7376 | if len(attr) > 2: | ||
Matt Harbison
|
r41040 | if not attr[2].startswith(b'hg'): | ||
s += b' ' + attr[2] | ||||
Matt Mackall
|
r7376 | else: | ||
s = attr[2] | ||||
Matt Harbison
|
r41040 | d[b'synopsis'] = s.strip() | ||
Matt Mackall
|
r7376 | |||
Benoit Boissinot
|
r1814 | return d | ||
Takumi IINO
|
r19423 | def showdoc(ui): | ||
Benoit Boissinot
|
r1814 | # print options | ||
Matt Harbison
|
r41040 | ui.write(minirst.section(_(b"Options"))) | ||
FUJIWARA Katsunori
|
r20081 | multioccur = False | ||
Benoit Boissinot
|
r1814 | for optstr, desc in get_opts(globalopts): | ||
Matt Harbison
|
r41040 | ui.write(b"%s\n %s\n\n" % (optstr, desc)) | ||
if optstr.endswith(b"[+]>"): | ||||
FUJIWARA Katsunori
|
r20081 | multioccur = True | ||
if multioccur: | ||||
Matt Harbison
|
r41040 | ui.write(_(b"\n[+] marked option can be specified multiple times\n")) | ||
ui.write(b"\n") | ||||
Benoit Boissinot
|
r1814 | |||
# print cmds | ||||
Matt Harbison
|
r41040 | ui.write(minirst.section(_(b"Commands"))) | ||
Dan Villiom Podlaski Christiansen
|
r18748 | commandprinter(ui, table, minirst.subsection) | ||
Erik Zielke
|
r12756 | |||
Takumi IINO
|
r19233 | # print help topics | ||
# The config help topic is included in the hgrc.5 man page. | ||||
Matt Harbison
|
r41040 | helpprinter(ui, helptable, minirst.section, exclude=[b'config']) | ||
Erik Zielke
|
r12756 | |||
Matt Harbison
|
r41040 | ui.write(minirst.section(_(b"Extensions"))) | ||
ui.write(_(b"This section contains help for extensions that are " | ||||
b"distributed together with Mercurial. Help for other " | ||||
b"extensions is available in the help system.")) | ||||
ui.write((b"\n\n" | ||||
b".. contents::\n" | ||||
b" :class: htmlonly\n" | ||||
b" :local:\n" | ||||
b" :depth: 1\n\n")) | ||||
Erik Zielke
|
r12781 | |||
for extensionname in sorted(allextensionnames()): | ||||
Jun Wu
|
r27660 | mod = extensions.load(ui, extensionname, None) | ||
Dan Villiom Podlaski Christiansen
|
r18748 | ui.write(minirst.subsection(extensionname)) | ||
Matt Harbison
|
r41040 | ui.write(b"%s\n\n" % gettext(pycompat.getdoc(mod))) | ||
Erik Zielke
|
r12781 | cmdtable = getattr(mod, 'cmdtable', None) | ||
if cmdtable: | ||||
Matt Harbison
|
r41040 | ui.write(minirst.subsubsection(_(b'Commands'))) | ||
Dan Villiom Podlaski Christiansen
|
r18748 | commandprinter(ui, cmdtable, minirst.subsubsubsection) | ||
Erik Zielke
|
r12781 | |||
Takumi IINO
|
r19424 | def showtopic(ui, topic): | ||
extrahelptable = [ | ||||
Matt Harbison
|
r41040 | ([b"common"], b'', loaddoc(b'common'), help.TOPIC_CATEGORY_MISC), | ||
([b"hg.1"], b'', loaddoc(b'hg.1'), help.TOPIC_CATEGORY_CONFIG), | ||||
([b"hg-ssh.8"], b'', loaddoc(b'hg-ssh.8'), help.TOPIC_CATEGORY_CONFIG), | ||||
([b"hgignore.5"], b'', loaddoc(b'hgignore.5'), | ||||
Rodrigo Damazio
|
r40330 | help.TOPIC_CATEGORY_CONFIG), | ||
Matt Harbison
|
r41040 | ([b"hgrc.5"], b'', loaddoc(b'hgrc.5'), help.TOPIC_CATEGORY_CONFIG), | ||
([b"hgignore.5.gendoc"], b'', loaddoc(b'hgignore'), | ||||
help.TOPIC_CATEGORY_CONFIG), | ||||
([b"hgrc.5.gendoc"], b'', loaddoc(b'config'), | ||||
help.TOPIC_CATEGORY_CONFIG), | ||||
Takumi IINO
|
r19424 | ] | ||
helpprinter(ui, helptable + extrahelptable, None, include=[topic]) | ||||
Takumi IINO
|
r19233 | def helpprinter(ui, helptable, sectionfunc, include=[], exclude=[]): | ||
rdamazio@google.com
|
r40327 | for h in helptable: | ||
names, sec, doc = h[0:3] | ||||
Takumi IINO
|
r19233 | if exclude and names[0] in exclude: | ||
continue | ||||
if include and names[0] not in include: | ||||
continue | ||||
for name in names: | ||||
Matt Harbison
|
r41040 | ui.write(b".. _%s:\n" % name) | ||
ui.write(b"\n") | ||||
Takumi IINO
|
r19233 | if sectionfunc: | ||
ui.write(sectionfunc(sec)) | ||||
Augie Fackler
|
r21793 | if callable(doc): | ||
Yuya Nishihara
|
r26413 | doc = doc(ui) | ||
Takumi IINO
|
r19233 | ui.write(doc) | ||
Matt Harbison
|
r41040 | ui.write(b"\n") | ||
Takumi IINO
|
r19233 | |||
Erik Zielke
|
r12781 | def commandprinter(ui, cmdtable, sectionfunc): | ||
Benoit Boissinot
|
r1814 | h = {} | ||
Erik Zielke
|
r12756 | for c, attr in cmdtable.items(): | ||
Matt Harbison
|
r41040 | f = c.split(b"|")[0] | ||
f = f.lstrip(b"^") | ||||
Christian Ebert
|
r6488 | h[f] = c | ||
Benoit Boissinot
|
r1814 | cmds = h.keys() | ||
Matt Harbison
|
r41040 | for f in sorted(cmds): | ||
if f.startswith(b"debug"): | ||||
Matt Mackall
|
r10282 | continue | ||
Erik Zielke
|
r12756 | d = get_cmd(h[f], cmdtable) | ||
Matt Harbison
|
r41040 | ui.write(sectionfunc(d[b'cmd'])) | ||
FUJIWARA Katsunori
|
r20689 | # short description | ||
Matt Harbison
|
r41040 | ui.write(d[b'desc'][0]) | ||
Christian Ebert
|
r6488 | # synopsis | ||
Matt Harbison
|
r41040 | ui.write(b"::\n\n") | ||
synopsislines = d[b'synopsis'].splitlines() | ||||
Martin Geisler
|
r12813 | for line in synopsislines: | ||
# some commands (such as rebase) have a multi-line | ||||
# synopsis | ||||
Matt Harbison
|
r41040 | ui.write(b" %s\n" % line) | ||
ui.write(b'\n') | ||||
Christian Ebert
|
r6488 | # description | ||
Matt Harbison
|
r41040 | ui.write(b"%s\n\n" % d[b'desc'][1]) | ||
Christian Ebert
|
r6488 | # options | ||
Matt Harbison
|
r41040 | opt_output = list(d[b'opts']) | ||
Christian Ebert
|
r6488 | if opt_output: | ||
opts_len = max([len(line[0]) for line in opt_output]) | ||||
Matt Harbison
|
r41040 | ui.write(_(b"Options:\n\n")) | ||
FUJIWARA Katsunori
|
r20081 | multioccur = False | ||
Christian Ebert
|
r6488 | for optstr, desc in opt_output: | ||
if desc: | ||||
Matt Harbison
|
r41040 | s = b"%-*s %s" % (opts_len, optstr, desc) | ||
Christian Ebert
|
r6488 | else: | ||
s = optstr | ||||
Matt Harbison
|
r41040 | ui.write(b"%s\n" % s) | ||
if optstr.endswith(b"[+]>"): | ||||
FUJIWARA Katsunori
|
r20081 | multioccur = True | ||
if multioccur: | ||||
Matt Harbison
|
r41040 | ui.write(_(b"\n[+] marked option can be specified" | ||
b" multiple times\n")) | ||||
ui.write(b"\n") | ||||
Christian Ebert
|
r6488 | # aliases | ||
Matt Harbison
|
r41040 | if d[b'aliases']: | ||
ui.write(_(b" aliases: %s\n\n") % b" ".join(d[b'aliases'])) | ||||
Benoit Boissinot
|
r1814 | |||
Matt Mackall
|
r3797 | |||
Erik Zielke
|
r12781 | def allextensionnames(): | ||
Matt Harbison
|
r41040 | return set(extensions.enabled().keys()) | set(extensions.disabled().keys()) | ||
Erik Zielke
|
r12781 | |||
Benoit Boissinot
|
r1814 | if __name__ == "__main__": | ||
Matt Harbison
|
r41040 | doc = b'hg.1.gendoc' | ||
Takumi IINO
|
r19425 | if len(sys.argv) > 1: | ||
Matt Harbison
|
r41059 | doc = encoding.strtolocal(sys.argv[1]) | ||
Takumi IINO
|
r19425 | |||
Yuya Nishihara
|
r30559 | ui = uimod.ui.load() | ||
Matt Harbison
|
r41040 | if doc == b'hg.1.gendoc': | ||
Yuya Nishihara
|
r26412 | showdoc(ui) | ||
Takumi IINO
|
r19425 | else: | ||
Matt Harbison
|
r41066 | showtopic(ui, encoding.strtolocal(sys.argv[1])) | ||