Show More
@@ -0,0 +1,95 b'' | |||||
|
1 | #!/usr/bin/env python | |||
|
2 | # | |||
|
3 | # check-config - a config flag documentation checker for Mercurial | |||
|
4 | # | |||
|
5 | # Copyright 2015 Matt Mackall <mpm@selenic.com> | |||
|
6 | # | |||
|
7 | # This software may be used and distributed according to the terms of the | |||
|
8 | # GNU General Public License version 2 or any later version. | |||
|
9 | ||||
|
10 | import re | |||
|
11 | import sys | |||
|
12 | ||||
|
13 | foundopts = {} | |||
|
14 | documented = {} | |||
|
15 | ||||
|
16 | configre = (r"""ui\.config(|int|bool|list)\(['"](\S+)['"], ?""" | |||
|
17 | r"""['"](\S+)['"](,\s(?:default=)?(\S+?))?\)""") | |||
|
18 | ||||
|
19 | def main(args): | |||
|
20 | for f in args: | |||
|
21 | sect = '' | |||
|
22 | prevname = '' | |||
|
23 | confsect = '' | |||
|
24 | for l in open(f): | |||
|
25 | ||||
|
26 | # check topic-like bits | |||
|
27 | m = re.match('\s*``(\S+)``', l) | |||
|
28 | if m: | |||
|
29 | prevname = m.group(1) | |||
|
30 | continue | |||
|
31 | if re.match('^\s*-+$', l): | |||
|
32 | sect = prevname | |||
|
33 | prevname = '' | |||
|
34 | continue | |||
|
35 | ||||
|
36 | if sect and prevname: | |||
|
37 | name = sect + '.' + prevname | |||
|
38 | documented[name] = 1 | |||
|
39 | ||||
|
40 | # check docstring bits | |||
|
41 | m = re.match(r'^\s+\[(\S+)\]', l) | |||
|
42 | if m: | |||
|
43 | confsect = m.group(1) | |||
|
44 | continue | |||
|
45 | m = re.match(r'^\s+(?:#\s*)?([a-z._]+) = ', l) | |||
|
46 | if m: | |||
|
47 | name = confsect + '.' + m.group(1) | |||
|
48 | documented[name] = 1 | |||
|
49 | ||||
|
50 | # like the bugzilla extension | |||
|
51 | m = re.match(r'^\s*([a-z]+\.[a-z]+)$', l) | |||
|
52 | if m: | |||
|
53 | documented[m.group(1)] = 1 | |||
|
54 | ||||
|
55 | # quoted in help or docstrings | |||
|
56 | m = re.match(r'.*?``([-a-z_]+\.[-a-z_]+)``', l) | |||
|
57 | if m: | |||
|
58 | documented[m.group(1)] = 1 | |||
|
59 | ||||
|
60 | # look for ignore markers | |||
|
61 | m = re.search(r'# (?:internal|experimental|deprecated|developer)' | |||
|
62 | ' config: (\S+.\S+)$', l) | |||
|
63 | if m: | |||
|
64 | documented[m.group(1)] = 1 | |||
|
65 | ||||
|
66 | # look for code-like bits | |||
|
67 | m = re.search(configre, l) | |||
|
68 | if m: | |||
|
69 | ctype = m.group(1) | |||
|
70 | if not ctype: | |||
|
71 | ctype = 'str' | |||
|
72 | name = m.group(2) + "." + m.group(3) | |||
|
73 | default = m.group(5) | |||
|
74 | if default in (None, 'False', 'None', '0', '[]', '""', "''"): | |||
|
75 | default = '' | |||
|
76 | if re.match('[a-z.]+$', default): | |||
|
77 | default = '<variable>' | |||
|
78 | if name in foundopts and (ctype, default) != foundopts[name]: | |||
|
79 | print l | |||
|
80 | print "conflict on %s: %r != %r" % (name, (ctype, default), | |||
|
81 | foundopts[name]) | |||
|
82 | foundopts[name] = (ctype, default) | |||
|
83 | ||||
|
84 | for name in sorted(foundopts): | |||
|
85 | if name not in documented: | |||
|
86 | if not (name.startswith("devel.") or | |||
|
87 | name.startswith("experimental.") or | |||
|
88 | name.startswith("debug.")): | |||
|
89 | ctype, default = foundopts[name] | |||
|
90 | if default: | |||
|
91 | default = ' [%s]' % default | |||
|
92 | print "undocumented: %s (%s)%s" % (name, ctype, default) | |||
|
93 | ||||
|
94 | if __name__ == "__main__": | |||
|
95 | sys.exit(main(sys.argv[1:])) |
General Comments 0
You need to be logged in to leave comments.
Login now