ansible/ansible_testing/modules.py

285 lines
9.3 KiB
Python
Raw Normal View History

2015-05-18 16:00:34 +00:00
#!/usr/bin/env python
from __future__ import print_function
import os
import abc
2015-05-18 16:00:34 +00:00
import ast
import argparse
from fnmatch import fnmatch
2015-05-18 16:00:34 +00:00
from ansible.utils.module_docs import get_docstring, BLACKLIST_MODULES
BLACKLIST_DIRS = frozenset(('.git',))
2015-05-18 19:11:15 +00:00
class Validator(object):
"""Validator instances are intended to be run on a single object. if you
are scanning multiple objects for problems, you'll want to have a separate
Validator for each one."""
__metaclass__ = abc.ABCMeta
def __init__(self):
self.reset()
def reset(self):
"""Reset the test results"""
self.errors = []
self.warnings = []
@abc.abstractproperty
def object_name(self):
"""Name of the object we validated"""
pass
@abc.abstractmethod
def validate(self, reset=True):
"""Run this method to generate the test results"""
if reset:
self.reset()
def report(self, warnings=False):
"""Print out the test results"""
if self.errors or (warnings and self.warnings):
print('=' * 76)
print(self.object_name)
print('=' * 76)
for error in self.errors:
print('ERROR: %s' % error)
if warnings:
for warning in self.warnings:
print('WARNING: %s' % warning)
if self.errors or (warnings and self.warnings):
print()
class ModuleValidator(Validator):
BLACKLIST_PATTERNS = ('.git*', '*.pyc', '*.pyo', '.*')
2015-05-18 19:11:15 +00:00
BLACKLIST_FILES = frozenset(('.git', '.gitignore', '.travis.yml',
'.gitattributes', '.gitmodules', 'COPYING',
'CONTRIBUTING.md', 'README.md',
'__init__.py'))
BLACKLIST = BLACKLIST_FILES.union(BLACKLIST_MODULES)
2015-05-18 16:00:34 +00:00
def __init__(self, path):
super(ModuleValidator, self).__init__()
2015-05-18 16:00:34 +00:00
self.path = path
self.basename = os.path.basename(self.path)
self.name, _ = os.path.splitext(self.basename)
with open(path) as f:
2015-05-18 20:30:58 +00:00
self.text = f.read()
self.length = len(self.text.splitlines())
try:
2015-05-18 20:30:58 +00:00
self.ast = ast.parse(self.text)
except:
self.ast = None
@property
def object_name(self):
return self.basename
def _python_module(self):
if self.path.endswith('.py'):
return True
return False
def _powershell_module(self):
if self.path.endswith('.ps1'):
return True
return False
2015-05-18 16:00:34 +00:00
def _just_docs(self):
for child in self.ast.body:
if not isinstance(child, ast.Assign):
return False
return True
2015-05-18 20:30:58 +00:00
def _check_interpreter(self):
if not self.text.startswith('#!/usr/bin/python'):
self.errors.append('Interpreter line is not "#!/usr/bin/python"')
2015-05-18 20:41:16 +00:00
def _find_json_import(self):
for child in self.ast.body:
if isinstance(child, ast.Import):
for name in child.names:
if name.name == 'json':
self.warnings.append('JSON import found, '
'already provided by '
'ansible.module_utils.basic')
2015-05-18 16:00:34 +00:00
def _find_module_utils(self):
linenos = []
for child in self.ast.body:
found_module_utils_import = False
if isinstance(child, ast.ImportFrom):
if child.module.startswith('ansible.module_utils.'):
found_module_utils_import = True
linenos.append(child.lineno)
if not child.names:
self.errors.append('%s: not a "from" import"' %
child.module)
found_alias = False
for name in child.names:
if isinstance(name, ast.alias):
found_alias = True
if name.asname or name.name != '*':
self.errors.append('%s: did not import "*"' %
child.module)
if found_module_utils_import and not found_alias:
self.errors.append('%s: did not import "*"' % child.module)
if not linenos:
self.errors.append('Did not find a module_utils import')
return linenos
def _find_main_call(self):
lineno = False
if_bodies = []
for child in self.ast.body:
if isinstance(child, ast.If):
try:
if child.test.left.id == '__name__':
if_bodies.extend(child.body)
except AttributeError:
pass
bodies = self.ast.body
bodies.extend(if_bodies)
for child in bodies:
if isinstance(child, ast.Expr):
if isinstance(child.value, ast.Call):
if (isinstance(child.value.func, ast.Name) and
child.value.func.id == 'main'):
lineno = child.lineno
if lineno < self.length - 1:
self.errors.append('Call to main() not the last '
2015-05-18 16:00:34 +00:00
'line')
if not lineno:
self.errors.append('Did not find a call to main')
return lineno or 0
def _find_has_import(self):
for child in self.ast.body:
found_try_except_import = False
found_has = False
if isinstance(child, ast.TryExcept):
bodies = child.body
2015-05-18 16:59:53 +00:00
for handler in child.handlers:
bodies.extend(handler.body)
2015-05-18 16:00:34 +00:00
for grandchild in bodies:
if isinstance(grandchild, ast.Import):
found_try_except_import = True
2015-05-18 16:59:53 +00:00
if isinstance(grandchild, ast.Assign):
2015-05-18 16:00:34 +00:00
for target in grandchild.targets:
if target.id.lower().startswith('has_'):
found_has = True
if found_try_except_import and not found_has:
self.warnings.append('Found Try/Except block without HAS_ '
'assginment')
def validate(self):
super(ModuleValidator, self).validate()
# Blacklists -- these files are not checked
2015-05-18 19:11:15 +00:00
if not frozenset((self.basename,
self.name)).isdisjoint(self.BLACKLIST):
return
for pat in self.BLACKLIST_PATTERNS:
if fnmatch(self.basename, pat):
return
if self._powershell_module():
2015-05-18 19:11:15 +00:00
self.warnings.append('Cannot check powershell modules at this '
'time. Skipping')
return
if not self._python_module():
2015-05-18 19:11:15 +00:00
self.errors.append('Official Ansible modules must have a .py '
'extension')
return
if self.ast is None:
self.errors.append('Python SyntaxError while parsing module')
2015-05-18 16:00:34 +00:00
return
doc, examples, ret = get_docstring(self.path)
if not bool(doc):
self.errors.append('Invalid or no DOCUMENTATION provided')
if not bool(examples):
self.errors.append('No EXAMPLES provided')
if not bool(ret):
self.warnings.append('No RETURN provided')
if not self._just_docs():
2015-05-18 20:30:58 +00:00
self._check_interpreter()
2015-05-18 20:41:16 +00:00
self._find_json_import()
2015-05-18 16:00:34 +00:00
module_utils = self._find_module_utils()
main = self._find_main_call()
for mu in module_utils:
if mu < main - 10:
self.warnings.append('module_utils import not near main()')
2015-05-18 16:00:34 +00:00
self._find_has_import()
class PythonPackageValidator(Validator):
def __init__(self, path):
super(PythonPackageValidator, self).__init__()
2015-05-18 16:00:34 +00:00
self.path = path
self.basename = os.path.basename(path)
@property
def object_name(self):
return self.basename
def validate(self):
super(PythonPackageValidator, self).validate()
init_file = os.path.join(self.path, '__init__.py')
if not os.path.exists(init_file):
2015-05-18 19:11:15 +00:00
self.errors.append('Ansible module subdirectories must contain an '
'__init__.py')
2015-05-18 16:00:34 +00:00
def main():
parser = argparse.ArgumentParser()
parser.add_argument('modules', help='Path to modules')
parser.add_argument('-w', '--warnings', help='Show warnings',
action='store_true')
args = parser.parse_args()
args.modules = args.modules.rstrip('/')
2015-05-18 16:00:34 +00:00
for root, dirs, files in os.walk(args.modules):
basedir = root[len(args.modules)+1:].split('/', 1)[0]
if basedir in BLACKLIST_DIRS:
continue
for dirname in dirs:
if root == args.modules and dirname in BLACKLIST_DIRS:
continue
path = os.path.join(root, dirname)
pv = PythonPackageValidator(os.path.abspath(path))
pv.validate()
pv.report(args.warnings)
2015-05-18 16:00:34 +00:00
for filename in files:
path = os.path.join(root, filename)
mv = ModuleValidator(os.path.abspath(path))
mv.validate()
mv.report(args.warnings)
if __name__ == '__main__':
main()