check-seclevel.py
233 lines
| 5.9 KiB
| text/x-python
|
PythonLexer
/ doc / check-seclevel.py
Gregory Szorc
|
r46434 | #!/usr/bin/env python3 | ||
FUJIWARA Katsunori
|
r17648 | # | ||
timeless@mozdev.org
|
r26192 | # checkseclevel - checking section title levels in each online help document | ||
FUJIWARA Katsunori
|
r17648 | |||
Pulkit Goyal
|
r28965 | from __future__ import absolute_import | ||
FUJIWARA Katsunori
|
r17648 | import optparse | ||
Pulkit Goyal
|
r28965 | import os | ||
import sys | ||||
FUJIWARA Katsunori
|
r17648 | |||
# import from the live mercurial repo | ||||
Gregory Szorc
|
r27221 | os.environ['HGMODULEPOLICY'] = 'py' | ||
FUJIWARA Katsunori
|
r17648 | sys.path.insert(0, "..") | ||
Augie Fackler
|
r43346 | from mercurial import demandimport | ||
demandimport.enable() | ||||
Pulkit Goyal
|
r28965 | from mercurial import ( | ||
commands, | ||||
extensions, | ||||
help, | ||||
minirst, | ||||
ui as uimod, | ||||
) | ||||
table = commands.table | ||||
helptable = help.helptable | ||||
FUJIWARA Katsunori
|
r17648 | |||
Augie Fackler
|
r32544 | level2mark = [b'"', b'=', b'-', b'.', b'#'] | ||
reservedmarks = [b'"'] | ||||
FUJIWARA Katsunori
|
r17648 | |||
mark2level = {} | ||||
Augie Fackler
|
r32544 | for m, l in zip(level2mark, range(len(level2mark))): | ||
FUJIWARA Katsunori
|
r17648 | if m not in reservedmarks: | ||
mark2level[m] = l | ||||
initlevel_topic = 0 | ||||
initlevel_cmd = 1 | ||||
initlevel_ext = 1 | ||||
initlevel_ext_cmd = 3 | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r26411 | def showavailables(ui, initlevel): | ||
Augie Fackler
|
r43346 | avail = ' available marks and order of them in this help: %s\n' % ( | ||
', '.join(['%r' % (m * 4) for m in level2mark[initlevel + 1 :]]) | ||||
) | ||||
Augie Fackler
|
r32544 | ui.warn(avail.encode('utf-8')) | ||
FUJIWARA Katsunori
|
r17648 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r26411 | def checkseclevel(ui, doc, name, initlevel): | ||
Augie Fackler
|
r43350 | ui.notenoi18n('checking "%s"\n' % name) | ||
Augie Fackler
|
r32544 | if not isinstance(doc, bytes): | ||
doc = doc.encode('utf-8') | ||||
FUJIWARA Katsunori
|
r17648 | blocks, pruned = minirst.parse(doc, 0, ['verbose']) | ||
errorcnt = 0 | ||||
curlevel = initlevel | ||||
for block in blocks: | ||||
Augie Fackler
|
r32544 | if block[b'type'] != b'section': | ||
FUJIWARA Katsunori
|
r17648 | continue | ||
Augie Fackler
|
r32544 | mark = block[b'underline'] | ||
title = block[b'lines'][0] | ||||
FUJIWARA Katsunori
|
r17648 | if (mark not in mark2level) or (mark2level[mark] <= initlevel): | ||
Augie Fackler
|
r43346 | ui.warn( | ||
( | ||||
'invalid section mark %r for "%s" of %s\n' | ||||
% (mark * 4, title, name) | ||||
).encode('utf-8') | ||||
) | ||||
Yuya Nishihara
|
r26411 | showavailables(ui, initlevel) | ||
FUJIWARA Katsunori
|
r17648 | errorcnt += 1 | ||
continue | ||||
nextlevel = mark2level[mark] | ||||
if curlevel < nextlevel and curlevel + 1 != nextlevel: | ||||
r43364 | ui.warnnoi18n( | |||
'gap of section level at "%s" of %s\n' % (title, name) | ||||
) | ||||
Yuya Nishihara
|
r26411 | showavailables(ui, initlevel) | ||
FUJIWARA Katsunori
|
r17648 | errorcnt += 1 | ||
continue | ||||
Augie Fackler
|
r43350 | ui.notenoi18n( | ||
Augie Fackler
|
r43346 | 'appropriate section level for "%s %s"\n' | ||
% (mark * (nextlevel * 2), title) | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | curlevel = nextlevel | ||
return errorcnt | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r26411 | def checkcmdtable(ui, cmdtable, namefmt, initlevel): | ||
FUJIWARA Katsunori
|
r17648 | errorcnt = 0 | ||
for k, entry in cmdtable.items(): | ||||
Augie Fackler
|
r32544 | name = k.split(b"|")[0].lstrip(b"^") | ||
FUJIWARA Katsunori
|
r17648 | if not entry[0].__doc__: | ||
r43364 | ui.notenoi18n( | |||
'skip checking %s: no help document\n' % (namefmt % name) | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | continue | ||
Augie Fackler
|
r43346 | errorcnt += checkseclevel( | ||
ui, entry[0].__doc__, namefmt % name, initlevel | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | return errorcnt | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r26411 | def checkhghelps(ui): | ||
FUJIWARA Katsunori
|
r17648 | errorcnt = 0 | ||
Rodrigo Damazio
|
r40328 | for h in helptable: | ||
names, sec, doc = h[0:3] | ||||
Augie Fackler
|
r21792 | if callable(doc): | ||
Yuya Nishihara
|
r26413 | doc = doc(ui) | ||
Augie Fackler
|
r43346 | errorcnt += checkseclevel( | ||
ui, doc, '%s help topic' % names[0], initlevel_topic | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | |||
Yuya Nishihara
|
r26411 | errorcnt += checkcmdtable(ui, table, '%s command', initlevel_cmd) | ||
FUJIWARA Katsunori
|
r17648 | |||
Augie Fackler
|
r43346 | for name in sorted( | ||
list(extensions.enabled()) + list(extensions.disabled()) | ||||
): | ||||
Bryan O'Sullivan
|
r27511 | mod = extensions.load(ui, name, None) | ||
FUJIWARA Katsunori
|
r17648 | if not mod.__doc__: | ||
r43364 | ui.notenoi18n( | |||
'skip checking %s extension: no help document\n' % name | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | continue | ||
Augie Fackler
|
r43346 | errorcnt += checkseclevel( | ||
ui, mod.__doc__, '%s extension' % name, initlevel_ext | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | |||
cmdtable = getattr(mod, 'cmdtable', None) | ||||
if cmdtable: | ||||
Augie Fackler
|
r43346 | errorcnt += checkcmdtable( | ||
ui, | ||||
cmdtable, | ||||
'%%s command of %s extension' % name, | ||||
initlevel_ext_cmd, | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | return errorcnt | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r26411 | def checkfile(ui, filename, initlevel): | ||
FUJIWARA Katsunori
|
r17648 | if filename == '-': | ||
filename = 'stdin' | ||||
doc = sys.stdin.read() | ||||
else: | ||||
Bryan O'Sullivan
|
r27770 | with open(filename) as fp: | ||
FUJIWARA Katsunori
|
r17648 | doc = fp.read() | ||
Augie Fackler
|
r43350 | ui.notenoi18n( | ||
Augie Fackler
|
r43346 | 'checking input from %s with initlevel %d\n' % (filename, initlevel) | ||
) | ||||
Yuya Nishihara
|
r26411 | return checkseclevel(ui, doc, 'input from %s' % filename, initlevel) | ||
FUJIWARA Katsunori
|
r17648 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r26398 | def main(): | ||
Augie Fackler
|
r43346 | optparser = optparse.OptionParser( | ||
"""%prog [options] | ||||
FUJIWARA Katsunori
|
r17648 | |||
This checks all help documents of Mercurial (topics, commands, | ||||
extensions and commands of them), if no file is specified by --file | ||||
option. | ||||
Augie Fackler
|
r43346 | """ | ||
) | ||||
optparser.add_option( | ||||
"-v", "--verbose", help="enable additional output", action="store_true" | ||||
) | ||||
optparser.add_option( | ||||
"-d", "--debug", help="debug mode", action="store_true" | ||||
) | ||||
optparser.add_option( | ||||
"-f", | ||||
"--file", | ||||
help="filename to read in (or '-' for stdin)", | ||||
action="store", | ||||
default="", | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | |||
Augie Fackler
|
r43346 | optparser.add_option( | ||
"-t", | ||||
"--topic", | ||||
help="parse file as help topic", | ||||
action="store_const", | ||||
dest="initlevel", | ||||
const=0, | ||||
) | ||||
optparser.add_option( | ||||
"-c", | ||||
"--command", | ||||
help="parse file as help of core command", | ||||
action="store_const", | ||||
dest="initlevel", | ||||
const=1, | ||||
) | ||||
optparser.add_option( | ||||
"-e", | ||||
"--extension", | ||||
help="parse file as help of extension", | ||||
action="store_const", | ||||
dest="initlevel", | ||||
const=1, | ||||
) | ||||
optparser.add_option( | ||||
"-C", | ||||
"--extension-command", | ||||
help="parse file as help of extension command", | ||||
action="store_const", | ||||
dest="initlevel", | ||||
const=3, | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | |||
Augie Fackler
|
r43346 | optparser.add_option( | ||
"-l", | ||||
"--initlevel", | ||||
help="set initial section level manually", | ||||
action="store", | ||||
type="int", | ||||
default=0, | ||||
) | ||||
FUJIWARA Katsunori
|
r17648 | |||
(options, args) = optparser.parse_args() | ||||
Yuya Nishihara
|
r30559 | ui = uimod.ui.load() | ||
Pulkit Goyal
|
r41391 | ui.setconfig(b'ui', b'verbose', options.verbose, b'--verbose') | ||
ui.setconfig(b'ui', b'debug', options.debug, b'--debug') | ||||
FUJIWARA Katsunori
|
r17648 | |||
if options.file: | ||||
Yuya Nishihara
|
r26411 | if checkfile(ui, options.file, options.initlevel): | ||
FUJIWARA Katsunori
|
r17648 | sys.exit(1) | ||
else: | ||||
Yuya Nishihara
|
r26411 | if checkhghelps(ui): | ||
FUJIWARA Katsunori
|
r17648 | sys.exit(1) | ||
Yuya Nishihara
|
r26398 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r26398 | if __name__ == "__main__": | ||
main() | ||||