2010-03-11 18:53:45 -04:00
|
|
|
#!/usr/bin/env python3
|
2009-01-03 17:15:20 -04:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
# Check for stylistic and formal issues in .rst and .py
|
|
|
|
# files included in the documentation.
|
|
|
|
#
|
|
|
|
# 01/2009, Georg Brandl
|
|
|
|
|
Merged revisions 68633,68648,68667,68706,68718,68720-68721,68724-68727,68739 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
r68633 | thomas.heller | 2009-01-16 12:53:44 -0600 (Fri, 16 Jan 2009) | 3 lines
Change an example in the docs to avoid a mistake when the code is copy
pasted and changed afterwards.
........
r68648 | benjamin.peterson | 2009-01-16 22:28:57 -0600 (Fri, 16 Jan 2009) | 1 line
use enumerate
........
r68667 | amaury.forgeotdarc | 2009-01-17 14:18:59 -0600 (Sat, 17 Jan 2009) | 3 lines
#4077: No need to append \n when calling Py_FatalError
+ fix a declaration to make it match the one in pythonrun.h
........
r68706 | benjamin.peterson | 2009-01-17 19:28:46 -0600 (Sat, 17 Jan 2009) | 1 line
fix grammar
........
r68718 | georg.brandl | 2009-01-18 04:42:35 -0600 (Sun, 18 Jan 2009) | 1 line
#4976: union() and intersection() take multiple args, but talk about "the other".
........
r68720 | georg.brandl | 2009-01-18 04:45:22 -0600 (Sun, 18 Jan 2009) | 1 line
#4974: fix redundant mention of lists and tuples.
........
r68721 | georg.brandl | 2009-01-18 04:48:16 -0600 (Sun, 18 Jan 2009) | 1 line
#4914: trunc is in math.
........
r68724 | georg.brandl | 2009-01-18 07:24:10 -0600 (Sun, 18 Jan 2009) | 1 line
#4979: correct result range for some random functions.
........
r68725 | georg.brandl | 2009-01-18 07:47:26 -0600 (Sun, 18 Jan 2009) | 1 line
#4857: fix augmented assignment target spec.
........
r68726 | georg.brandl | 2009-01-18 08:41:52 -0600 (Sun, 18 Jan 2009) | 1 line
#4923: clarify what was added.
........
r68727 | georg.brandl | 2009-01-18 12:25:30 -0600 (Sun, 18 Jan 2009) | 1 line
#4986: augassigns are not expressions.
........
r68739 | benjamin.peterson | 2009-01-18 15:11:38 -0600 (Sun, 18 Jan 2009) | 1 line
fix test that wasn't working as expected #4990
........
2009-01-18 18:27:04 -04:00
|
|
|
# TODO: - wrong versions in versionadded/changed
|
|
|
|
# - wrong markup after versionchanged directive
|
|
|
|
|
2009-01-03 17:15:20 -04:00
|
|
|
import os
|
|
|
|
import re
|
|
|
|
import sys
|
|
|
|
import getopt
|
2020-12-18 05:48:08 -04:00
|
|
|
from string import ascii_letters
|
2009-01-03 17:15:20 -04:00
|
|
|
from os.path import join, splitext, abspath, exists
|
|
|
|
from collections import defaultdict
|
|
|
|
|
|
|
|
directives = [
|
|
|
|
# standard docutils ones
|
|
|
|
'admonition', 'attention', 'caution', 'class', 'compound', 'container',
|
|
|
|
'contents', 'csv-table', 'danger', 'date', 'default-role', 'epigraph',
|
|
|
|
'error', 'figure', 'footer', 'header', 'highlights', 'hint', 'image',
|
|
|
|
'important', 'include', 'line-block', 'list-table', 'meta', 'note',
|
|
|
|
'parsed-literal', 'pull-quote', 'raw', 'replace',
|
|
|
|
'restructuredtext-test-directive', 'role', 'rubric', 'sectnum', 'sidebar',
|
|
|
|
'table', 'target-notes', 'tip', 'title', 'topic', 'unicode', 'warning',
|
2014-10-30 18:35:55 -03:00
|
|
|
# Sphinx and Python docs custom ones
|
2009-01-03 17:15:20 -04:00
|
|
|
'acks', 'attribute', 'autoattribute', 'autoclass', 'autodata',
|
2018-10-12 11:55:20 -03:00
|
|
|
'autoexception', 'autofunction', 'automethod', 'automodule',
|
|
|
|
'availability', 'centered', 'cfunction', 'class', 'classmethod', 'cmacro',
|
|
|
|
'cmdoption', 'cmember', 'code-block', 'confval', 'cssclass', 'ctype',
|
|
|
|
'currentmodule', 'cvar', 'data', 'decorator', 'decoratormethod',
|
|
|
|
'deprecated-removed', 'deprecated(?!-removed)', 'describe', 'directive',
|
|
|
|
'doctest', 'envvar', 'event', 'exception', 'function', 'glossary',
|
|
|
|
'highlight', 'highlightlang', 'impl-detail', 'index', 'literalinclude',
|
|
|
|
'method', 'miscnews', 'module', 'moduleauthor', 'opcode', 'pdbcommand',
|
|
|
|
'productionlist', 'program', 'role', 'sectionauthor', 'seealso',
|
|
|
|
'sourcecode', 'staticmethod', 'tabularcolumns', 'testcode', 'testoutput',
|
|
|
|
'testsetup', 'toctree', 'todo', 'todolist', 'versionadded',
|
|
|
|
'versionchanged'
|
2009-01-03 17:15:20 -04:00
|
|
|
]
|
|
|
|
|
|
|
|
all_directives = '(' + '|'.join(directives) + ')'
|
2016-02-25 15:14:10 -04:00
|
|
|
seems_directive_re = re.compile(r'(?<!\.)\.\. %s([^a-z:]|:(?!:))' % all_directives)
|
2009-01-03 17:15:20 -04:00
|
|
|
default_role_re = re.compile(r'(^| )`\w([^`]*?\w)?`($| )')
|
2014-10-30 18:49:54 -03:00
|
|
|
leaked_markup_re = re.compile(r'[a-z]::\s|`|\.\.\s*\w+:')
|
2009-01-03 17:15:20 -04:00
|
|
|
|
|
|
|
|
|
|
|
checkers = {}
|
|
|
|
|
|
|
|
checker_props = {'severity': 1, 'falsepositives': False}
|
|
|
|
|
2014-10-30 18:30:01 -03:00
|
|
|
|
2009-01-03 17:15:20 -04:00
|
|
|
def checker(*suffixes, **kwds):
|
|
|
|
"""Decorator to register a function as a checker."""
|
|
|
|
def deco(func):
|
|
|
|
for suffix in suffixes:
|
|
|
|
checkers.setdefault(suffix, []).append(func)
|
|
|
|
for prop in checker_props:
|
|
|
|
setattr(func, prop, kwds.get(prop, checker_props[prop]))
|
|
|
|
return func
|
|
|
|
return deco
|
|
|
|
|
|
|
|
|
|
|
|
@checker('.py', severity=4)
|
|
|
|
def check_syntax(fn, lines):
|
|
|
|
"""Check Python examples for valid syntax."""
|
Merged revisions 68288-68291,68325-68326,68338,68388,68393,68423 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
................
r68288 | benjamin.peterson | 2009-01-03 18:39:07 -0600 (Sat, 03 Jan 2009) | 1 line
only check the actual compile() call for a SyntaxError
................
r68289 | georg.brandl | 2009-01-04 02:26:10 -0600 (Sun, 04 Jan 2009) | 2 lines
Test commit.
................
r68290 | georg.brandl | 2009-01-04 04:23:49 -0600 (Sun, 04 Jan 2009) | 4 lines
Add "suspicious" builder which finds leftover markup in the HTML files.
Patch by Gabriel Genellina.
................
r68291 | georg.brandl | 2009-01-04 04:24:09 -0600 (Sun, 04 Jan 2009) | 2 lines
Fix two issues found by the suspicious builder.
................
r68325 | benjamin.peterson | 2009-01-04 16:00:18 -0600 (Sun, 04 Jan 2009) | 1 line
use Jinja 2.1.1
................
r68326 | georg.brandl | 2009-01-04 16:03:10 -0600 (Sun, 04 Jan 2009) | 2 lines
Update make.bat.
................
r68338 | neal.norwitz | 2009-01-04 21:57:25 -0600 (Sun, 04 Jan 2009) | 1 line
Make sure to checkout any new packages
................
r68388 | benjamin.peterson | 2009-01-07 21:39:46 -0600 (Wed, 07 Jan 2009) | 1 line
string exceptions are gone
................
r68393 | benjamin.peterson | 2009-01-07 22:01:00 -0600 (Wed, 07 Jan 2009) | 1 line
use new sphinx modules
................
r68423 | benjamin.peterson | 2009-01-08 20:13:34 -0600 (Thu, 08 Jan 2009) | 29 lines
Merged revisions 68306-68308,68340,68368,68422 via svnmerge from
svn+ssh://pythondev@svn.python.org/sandbox/trunk/2to3/lib2to3
........
r68306 | benjamin.peterson | 2009-01-04 12:27:19 -0600 (Sun, 04 Jan 2009) | 1 line
fix_urllib: add mappings for the url parsing functions
........
r68307 | benjamin.peterson | 2009-01-04 12:30:01 -0600 (Sun, 04 Jan 2009) | 1 line
remove duplicated function
........
r68308 | benjamin.peterson | 2009-01-04 12:50:34 -0600 (Sun, 04 Jan 2009) | 1 line
turtle is no longer renamed
........
r68340 | georg.brandl | 2009-01-05 02:11:39 -0600 (Mon, 05 Jan 2009) | 2 lines
Fix undefined locals in parse_tokens().
........
r68368 | benjamin.peterson | 2009-01-06 17:56:10 -0600 (Tue, 06 Jan 2009) | 1 line
fix typo (thanks to Robert Lehmann)
........
r68422 | benjamin.peterson | 2009-01-08 20:01:03 -0600 (Thu, 08 Jan 2009) | 1 line
run the imports fixers after fix_import, so fix_import doesn't try to make stdlib renames into relative imports #4876
........
................
2009-01-08 23:03:23 -04:00
|
|
|
code = ''.join(lines)
|
|
|
|
if '\r' in code:
|
|
|
|
if os.name != 'nt':
|
|
|
|
yield 0, '\\r in code file'
|
|
|
|
code = code.replace('\r', '')
|
2009-01-03 17:15:20 -04:00
|
|
|
try:
|
|
|
|
compile(code, fn, 'exec')
|
|
|
|
except SyntaxError as err:
|
|
|
|
yield err.lineno, 'not compilable: %s' % err
|
|
|
|
|
|
|
|
|
|
|
|
@checker('.rst', severity=2)
|
|
|
|
def check_suspicious_constructs(fn, lines):
|
|
|
|
"""Check for suspicious reST constructs."""
|
|
|
|
inprod = False
|
|
|
|
for lno, line in enumerate(lines):
|
2016-02-25 15:14:10 -04:00
|
|
|
if seems_directive_re.search(line):
|
2009-01-03 17:15:20 -04:00
|
|
|
yield lno+1, 'comment seems to be intended as a directive'
|
|
|
|
if '.. productionlist::' in line:
|
|
|
|
inprod = True
|
|
|
|
elif not inprod and default_role_re.search(line):
|
|
|
|
yield lno+1, 'default role used'
|
|
|
|
elif inprod and not line.strip():
|
|
|
|
inprod = False
|
|
|
|
|
|
|
|
|
|
|
|
@checker('.py', '.rst')
|
|
|
|
def check_whitespace(fn, lines):
|
|
|
|
"""Check for whitespace and line length issues."""
|
|
|
|
for lno, line in enumerate(lines):
|
|
|
|
if '\r' in line:
|
|
|
|
yield lno+1, '\\r in line'
|
|
|
|
if '\t' in line:
|
|
|
|
yield lno+1, 'OMG TABS!!!1'
|
|
|
|
if line[:-1].rstrip(' \t') != line[:-1]:
|
|
|
|
yield lno+1, 'trailing whitespace'
|
2009-01-03 17:30:40 -04:00
|
|
|
|
|
|
|
|
|
|
|
@checker('.rst', severity=0)
|
|
|
|
def check_line_length(fn, lines):
|
|
|
|
"""Check for line length; this checker is not run by default."""
|
|
|
|
for lno, line in enumerate(lines):
|
|
|
|
if len(line) > 81:
|
2009-01-03 17:15:20 -04:00
|
|
|
# don't complain about tables, links and function signatures
|
|
|
|
if line.lstrip()[0] not in '+|' and \
|
|
|
|
'http://' not in line and \
|
|
|
|
not line.lstrip().startswith(('.. function',
|
|
|
|
'.. method',
|
|
|
|
'.. cfunction')):
|
|
|
|
yield lno+1, "line too long"
|
|
|
|
|
|
|
|
|
|
|
|
@checker('.html', severity=2, falsepositives=True)
|
|
|
|
def check_leaked_markup(fn, lines):
|
|
|
|
"""Check HTML files for leaked reST markup; this only works if
|
|
|
|
the HTML files have been built.
|
|
|
|
"""
|
|
|
|
for lno, line in enumerate(lines):
|
|
|
|
if leaked_markup_re.search(line):
|
|
|
|
yield lno+1, 'possibly leaked markup: %r' % line
|
|
|
|
|
|
|
|
|
2020-12-18 05:48:08 -04:00
|
|
|
def hide_literal_blocks(lines):
|
|
|
|
"""Tool to remove literal blocks from given lines.
|
|
|
|
|
|
|
|
It yields empty lines in place of blocks, so line numbers are
|
|
|
|
still meaningful.
|
|
|
|
"""
|
|
|
|
in_block = False
|
|
|
|
for line in lines:
|
|
|
|
if line.endswith("::\n"):
|
|
|
|
in_block = True
|
|
|
|
elif in_block:
|
|
|
|
if line == "\n" or line.startswith(" "):
|
|
|
|
line = "\n"
|
|
|
|
else:
|
|
|
|
in_block = False
|
|
|
|
yield line
|
|
|
|
|
|
|
|
|
|
|
|
def type_of_explicit_markup(line):
|
|
|
|
if re.match(fr'\.\. {all_directives}::', line):
|
|
|
|
return 'directive'
|
|
|
|
if re.match(r'\.\. \[[0-9]+\] ', line):
|
|
|
|
return 'footnote'
|
|
|
|
if re.match(r'\.\. \[[^\]]+\] ', line):
|
|
|
|
return 'citation'
|
|
|
|
if re.match(r'\.\. _.*[^_]: ', line):
|
|
|
|
return 'target'
|
|
|
|
if re.match(r'\.\. \|[^\|]*\| ', line):
|
|
|
|
return 'substitution_definition'
|
|
|
|
return 'comment'
|
|
|
|
|
|
|
|
|
|
|
|
def hide_comments(lines):
|
|
|
|
"""Tool to remove comments from given lines.
|
|
|
|
|
|
|
|
It yields empty lines in place of comments, so line numbers are
|
|
|
|
still meaningfull.
|
|
|
|
"""
|
|
|
|
in_multiline_comment = False
|
|
|
|
for line in lines:
|
|
|
|
if line == "..\n":
|
|
|
|
in_multiline_comment = True
|
|
|
|
elif in_multiline_comment:
|
|
|
|
if line == "\n" or line.startswith(" "):
|
|
|
|
line = "\n"
|
|
|
|
else:
|
|
|
|
in_multiline_comment = False
|
|
|
|
if line.startswith(".. ") and type_of_explicit_markup(line) == 'comment':
|
|
|
|
line = "\n"
|
|
|
|
yield line
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@checker(".rst", severity=2)
|
|
|
|
def check_missing_surrogate_space_on_plural(fn, lines):
|
|
|
|
r"""Check for missing 'backslash-space' between a code sample a letter.
|
|
|
|
|
|
|
|
Good: ``Point``\ s
|
|
|
|
Bad: ``Point``s
|
|
|
|
"""
|
|
|
|
in_code_sample = False
|
|
|
|
check_next_one = False
|
|
|
|
for lno, line in enumerate(hide_comments(hide_literal_blocks(lines))):
|
|
|
|
tokens = line.split("``")
|
|
|
|
for token_no, token in enumerate(tokens):
|
|
|
|
if check_next_one:
|
|
|
|
if token[0] in ascii_letters:
|
|
|
|
yield lno + 1, f"Missing backslash-space between code sample and {token!r}."
|
|
|
|
check_next_one = False
|
|
|
|
if token_no == len(tokens) - 1:
|
|
|
|
continue
|
|
|
|
if in_code_sample:
|
|
|
|
check_next_one = True
|
|
|
|
in_code_sample = not in_code_sample
|
|
|
|
|
2009-01-03 17:15:20 -04:00
|
|
|
def main(argv):
|
|
|
|
usage = '''\
|
|
|
|
Usage: %s [-v] [-f] [-s sev] [-i path]* [path]
|
|
|
|
|
|
|
|
Options: -v verbose (print all checked file names)
|
|
|
|
-f enable checkers that yield many false positives
|
|
|
|
-s sev only show problems with severity >= sev
|
|
|
|
-i path ignore subdir or file path
|
|
|
|
''' % argv[0]
|
|
|
|
try:
|
|
|
|
gopts, args = getopt.getopt(argv[1:], 'vfs:i:')
|
|
|
|
except getopt.GetoptError:
|
|
|
|
print(usage)
|
|
|
|
return 2
|
|
|
|
|
|
|
|
verbose = False
|
|
|
|
severity = 1
|
|
|
|
ignore = []
|
|
|
|
falsepos = False
|
|
|
|
for opt, val in gopts:
|
|
|
|
if opt == '-v':
|
|
|
|
verbose = True
|
|
|
|
elif opt == '-f':
|
|
|
|
falsepos = True
|
|
|
|
elif opt == '-s':
|
|
|
|
severity = int(val)
|
|
|
|
elif opt == '-i':
|
|
|
|
ignore.append(abspath(val))
|
|
|
|
|
|
|
|
if len(args) == 0:
|
|
|
|
path = '.'
|
|
|
|
elif len(args) == 1:
|
|
|
|
path = args[0]
|
|
|
|
else:
|
|
|
|
print(usage)
|
|
|
|
return 2
|
|
|
|
|
|
|
|
if not exists(path):
|
|
|
|
print('Error: path %s does not exist' % path)
|
|
|
|
return 2
|
|
|
|
|
|
|
|
count = defaultdict(int)
|
|
|
|
|
|
|
|
for root, dirs, files in os.walk(path):
|
|
|
|
# ignore subdirs in ignore list
|
|
|
|
if abspath(root) in ignore:
|
|
|
|
del dirs[:]
|
|
|
|
continue
|
|
|
|
|
|
|
|
for fn in files:
|
|
|
|
fn = join(root, fn)
|
|
|
|
if fn[:2] == './':
|
|
|
|
fn = fn[2:]
|
|
|
|
|
|
|
|
# ignore files in ignore list
|
|
|
|
if abspath(fn) in ignore:
|
|
|
|
continue
|
|
|
|
|
|
|
|
ext = splitext(fn)[1]
|
|
|
|
checkerlist = checkers.get(ext, None)
|
|
|
|
if not checkerlist:
|
|
|
|
continue
|
|
|
|
|
|
|
|
if verbose:
|
|
|
|
print('Checking %s...' % fn)
|
|
|
|
|
|
|
|
try:
|
2015-07-22 00:50:29 -03:00
|
|
|
with open(fn, 'r', encoding='utf-8') as f:
|
2009-01-03 17:15:20 -04:00
|
|
|
lines = list(f)
|
|
|
|
except (IOError, OSError) as err:
|
|
|
|
print('%s: cannot open: %s' % (fn, err))
|
|
|
|
count[4] += 1
|
|
|
|
continue
|
|
|
|
|
|
|
|
for checker in checkerlist:
|
|
|
|
if checker.falsepositives and not falsepos:
|
|
|
|
continue
|
|
|
|
csev = checker.severity
|
|
|
|
if csev >= severity:
|
|
|
|
for lno, msg in checker(fn, lines):
|
2010-03-12 06:04:37 -04:00
|
|
|
print('[%d] %s:%d: %s' % (csev, fn, lno, msg))
|
2009-01-03 17:15:20 -04:00
|
|
|
count[csev] += 1
|
|
|
|
if verbose:
|
|
|
|
print()
|
|
|
|
if not count:
|
|
|
|
if severity > 1:
|
|
|
|
print('No problems with severity >= %d found.' % severity)
|
|
|
|
else:
|
|
|
|
print('No problems found.')
|
|
|
|
else:
|
|
|
|
for severity in sorted(count):
|
|
|
|
number = count[severity]
|
|
|
|
print('%d problem%s with severity %d found.' %
|
|
|
|
(number, number > 1 and 's' or '', severity))
|
|
|
|
return int(bool(count))
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
sys.exit(main(sys.argv))
|