import-checker.py
247 lines
| 7.9 KiB
| text/x-python
|
PythonLexer
/ contrib / import-checker.py
Augie Fackler
|
r20036 | import ast | ||
import os | ||||
import sys | ||||
Chris Jerdonek
|
r20198 | # Import a minimal set of stdlib modules needed for list_stdlib_modules() | ||
# to work when run from a virtualenv. The modules were chosen empirically | ||||
# so that the return value matches the return value without virtualenv. | ||||
import BaseHTTPServer | ||||
import zlib | ||||
Augie Fackler
|
r20036 | def dotted_name_of_path(path): | ||
"""Given a relative path to a source file, return its dotted module name. | ||||
>>> dotted_name_of_path('mercurial/error.py') | ||||
'mercurial.error' | ||||
Mads Kiilerich
|
r20383 | >>> dotted_name_of_path('zlibmodule.so') | ||
'zlib' | ||||
Augie Fackler
|
r20036 | """ | ||
parts = path.split('/') | ||||
parts[-1] = parts[-1][:-3] # remove .py | ||||
Mads Kiilerich
|
r20383 | if parts[-1].endswith('module'): | ||
parts[-1] = parts[-1][:-6] | ||||
Augie Fackler
|
r20036 | return '.'.join(parts) | ||
def list_stdlib_modules(): | ||||
"""List the modules present in the stdlib. | ||||
>>> mods = set(list_stdlib_modules()) | ||||
>>> 'BaseHTTPServer' in mods | ||||
True | ||||
os.path isn't really a module, so it's missing: | ||||
>>> 'os.path' in mods | ||||
False | ||||
sys requires special treatment, because it's baked into the | ||||
interpreter, but it should still appear: | ||||
>>> 'sys' in mods | ||||
True | ||||
>>> 'collections' in mods | ||||
True | ||||
>>> 'cStringIO' in mods | ||||
True | ||||
""" | ||||
for m in sys.builtin_module_names: | ||||
yield m | ||||
# These modules only exist on windows, but we should always | ||||
# consider them stdlib. | ||||
for m in ['msvcrt', '_winreg']: | ||||
yield m | ||||
# These get missed too | ||||
for m in 'ctypes', 'email': | ||||
yield m | ||||
yield 'builtins' # python3 only | ||||
Chris Jerdonek
|
r20197 | stdlib_prefixes = set([sys.prefix, sys.exec_prefix]) | ||
Chris Jerdonek
|
r20198 | # We need to supplement the list of prefixes for the search to work | ||
# when run from within a virtualenv. | ||||
for mod in (BaseHTTPServer, zlib): | ||||
try: | ||||
# Not all module objects have a __file__ attribute. | ||||
filename = mod.__file__ | ||||
except AttributeError: | ||||
continue | ||||
dirname = os.path.dirname(filename) | ||||
for prefix in stdlib_prefixes: | ||||
if dirname.startswith(prefix): | ||||
# Then this directory is redundant. | ||||
break | ||||
else: | ||||
stdlib_prefixes.add(dirname) | ||||
Augie Fackler
|
r20036 | for libpath in sys.path: | ||
Augie Fackler
|
r20201 | # We want to walk everything in sys.path that starts with | ||
# something in stdlib_prefixes. check-code suppressed because | ||||
# the ast module used by this script implies the availability | ||||
# of any(). | ||||
Simon Heimberg
|
r20238 | if not any(libpath.startswith(p) for p in stdlib_prefixes): # no-py24 | ||
Augie Fackler
|
r20036 | continue | ||
if 'site-packages' in libpath: | ||||
continue | ||||
for top, dirs, files in os.walk(libpath): | ||||
for name in files: | ||||
if name == '__init__.py': | ||||
continue | ||||
if not (name.endswith('.py') or name.endswith('.so')): | ||||
continue | ||||
full_path = os.path.join(top, name) | ||||
if 'site-packages' in full_path: | ||||
continue | ||||
rel_path = full_path[len(libpath) + 1:] | ||||
mod = dotted_name_of_path(rel_path) | ||||
yield mod | ||||
stdlib_modules = set(list_stdlib_modules()) | ||||
Augie Fackler
|
r20037 | def imported_modules(source, ignore_nested=False): | ||
Augie Fackler
|
r20036 | """Given the source of a file as a string, yield the names | ||
imported by that file. | ||||
Augie Fackler
|
r20037 | Args: | ||
source: The python source to examine as a string. | ||||
ignore_nested: If true, import statements that do not start in | ||||
column zero will be ignored. | ||||
Returns: | ||||
A list of module names imported by the given source. | ||||
>>> sorted(imported_modules( | ||||
Augie Fackler
|
r20036 | ... 'import foo ; from baz import bar; import foo.qux')) | ||
Augie Fackler
|
r20037 | ['baz.bar', 'foo', 'foo.qux'] | ||
>>> sorted(imported_modules( | ||||
... '''import foo | ||||
... def wat(): | ||||
... import bar | ||||
... ''', ignore_nested=True)) | ||||
['foo'] | ||||
Augie Fackler
|
r20036 | """ | ||
for node in ast.walk(ast.parse(source)): | ||||
Augie Fackler
|
r20037 | if ignore_nested and getattr(node, 'col_offset', 0) > 0: | ||
continue | ||||
Augie Fackler
|
r20036 | if isinstance(node, ast.Import): | ||
for n in node.names: | ||||
yield n.name | ||||
elif isinstance(node, ast.ImportFrom): | ||||
prefix = node.module + '.' | ||||
for n in node.names: | ||||
yield prefix + n.name | ||||
def verify_stdlib_on_own_line(source): | ||||
"""Given some python source, verify that stdlib imports are done | ||||
in separate statements from relative local module imports. | ||||
Observing this limitation is important as it works around an | ||||
annoying lib2to3 bug in relative import rewrites: | ||||
http://bugs.python.org/issue19510. | ||||
>>> list(verify_stdlib_on_own_line('import sys, foo')) | ||||
Mads Kiilerich
|
r20386 | ['mixed imports\\n stdlib: sys\\n relative: foo'] | ||
Augie Fackler
|
r20036 | >>> list(verify_stdlib_on_own_line('import sys, os')) | ||
[] | ||||
>>> list(verify_stdlib_on_own_line('import foo, bar')) | ||||
[] | ||||
""" | ||||
for node in ast.walk(ast.parse(source)): | ||||
if isinstance(node, ast.Import): | ||||
Mads Kiilerich
|
r20386 | from_stdlib = {False: [], True: []} | ||
Augie Fackler
|
r20036 | for n in node.names: | ||
Mads Kiilerich
|
r20386 | from_stdlib[n.name in stdlib_modules].append(n.name) | ||
if from_stdlib[True] and from_stdlib[False]: | ||||
yield ('mixed imports\n stdlib: %s\n relative: %s' % | ||||
(', '.join(sorted(from_stdlib[True])), | ||||
', '.join(sorted(from_stdlib[False])))) | ||||
Augie Fackler
|
r20036 | |||
class CircularImport(Exception): | ||||
pass | ||||
def cyclekey(names): | ||||
return tuple(sorted(set(names))) | ||||
def check_one_mod(mod, imports, path=None, ignore=None): | ||||
if path is None: | ||||
path = [] | ||||
if ignore is None: | ||||
ignore = [] | ||||
path = path + [mod] | ||||
for i in sorted(imports.get(mod, [])): | ||||
if i not in stdlib_modules: | ||||
i = mod.rsplit('.', 1)[0] + '.' + i | ||||
if i in path: | ||||
firstspot = path.index(i) | ||||
cycle = path[firstspot:] + [i] | ||||
if cyclekey(cycle) not in ignore: | ||||
raise CircularImport(cycle) | ||||
continue | ||||
check_one_mod(i, imports, path=path, ignore=ignore) | ||||
Augie Fackler
|
r20038 | def rotatecycle(cycle): | ||
"""arrange a cycle so that the lexicographically first module listed first | ||||
>>> rotatecycle(['foo', 'bar', 'foo']) | ||||
['bar', 'foo', 'bar'] | ||||
""" | ||||
lowest = min(cycle) | ||||
idx = cycle.index(lowest) | ||||
return cycle[idx:] + cycle[1:idx] + [lowest] | ||||
Augie Fackler
|
r20036 | |||
def find_cycles(imports): | ||||
"""Find cycles in an already-loaded import graph. | ||||
>>> imports = {'top.foo': ['bar', 'os.path', 'qux'], | ||||
... 'top.bar': ['baz', 'sys'], | ||||
... 'top.baz': ['foo'], | ||||
... 'top.qux': ['foo']} | ||||
>>> print '\\n'.join(sorted(find_cycles(imports))) | ||||
Augie Fackler
|
r20038 | top.bar -> top.baz -> top.foo -> top.bar -> top.bar | ||
top.foo -> top.qux -> top.foo -> top.foo | ||||
Augie Fackler
|
r20036 | """ | ||
cycles = {} | ||||
for mod in sorted(imports.iterkeys()): | ||||
try: | ||||
check_one_mod(mod, imports, ignore=cycles) | ||||
except CircularImport, e: | ||||
cycle = e.args[0] | ||||
cycles[cyclekey(cycle)] = ' -> '.join(rotatecycle(cycle)) | ||||
return cycles.values() | ||||
def _cycle_sortkey(c): | ||||
return len(c), c | ||||
def main(argv): | ||||
if len(argv) < 2: | ||||
print 'Usage: %s file [file] [file] ...' | ||||
return 1 | ||||
used_imports = {} | ||||
any_errors = False | ||||
for source_path in argv[1:]: | ||||
f = open(source_path) | ||||
modname = dotted_name_of_path(source_path) | ||||
src = f.read() | ||||
Augie Fackler
|
r20037 | used_imports[modname] = sorted( | ||
imported_modules(src, ignore_nested=True)) | ||||
Augie Fackler
|
r20036 | for error in verify_stdlib_on_own_line(src): | ||
any_errors = True | ||||
print source_path, error | ||||
f.close() | ||||
cycles = find_cycles(used_imports) | ||||
if cycles: | ||||
firstmods = set() | ||||
for c in sorted(cycles, key=_cycle_sortkey): | ||||
first = c.split()[0] | ||||
# As a rough cut, ignore any cycle that starts with the | ||||
# same module as some other cycle. Otherwise we see lots | ||||
# of cycles that are effectively duplicates. | ||||
if first in firstmods: | ||||
continue | ||||
print 'Import cycle:', c | ||||
firstmods.add(first) | ||||
any_errors = True | ||||
return not any_errors | ||||
if __name__ == '__main__': | ||||
sys.exit(int(main(sys.argv))) | ||||