mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2024-09-14 20:13:21 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			230 lines
		
	
	
	
		
			8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			230 lines
		
	
	
	
		
			8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """A plugin for pylint to identify imports and functions which should not be used."""
 | |
| from __future__ import (absolute_import, division, print_function)
 | |
| 
 | |
| __metaclass__ = type
 | |
| 
 | |
| import astroid
 | |
| 
 | |
| from pylint.checkers import BaseChecker
 | |
| from pylint.interfaces import IAstroidChecker
 | |
| 
 | |
| 
 | |
| class BlacklistEntry(object):
 | |
|     """Defines a import blacklist entry."""
 | |
|     def __init__(self, alternative, modules_only=False, names=None, ignore_paths=None):
 | |
|         """
 | |
|         :type alternative: str
 | |
|         :type modules_only: bool
 | |
|         :type names: tuple[str] | None
 | |
|         :type ignore_paths: tuple[str] | None
 | |
|         """
 | |
|         self.alternative = alternative
 | |
|         self.modules_only = modules_only
 | |
|         self.names = set(names) if names else None
 | |
|         self.ignore_paths = ignore_paths
 | |
| 
 | |
|     def applies_to(self, path, name=None):
 | |
|         """
 | |
|         :type path: str
 | |
|         :type name: str | None
 | |
|         :rtype: bool
 | |
|         """
 | |
|         if self.names:
 | |
|             if not name:
 | |
|                 return False
 | |
| 
 | |
|             if name not in self.names:
 | |
|                 return False
 | |
| 
 | |
|         if self.ignore_paths and any(path.endswith(ignore_path) for ignore_path in self.ignore_paths):
 | |
|             return False
 | |
| 
 | |
|         if self.modules_only:
 | |
|             return is_module_path(path)
 | |
| 
 | |
|         return True
 | |
| 
 | |
| 
 | |
| def is_module_path(path):
 | |
|     """
 | |
|     :type path: str
 | |
|     :rtype: bool
 | |
|     """
 | |
|     return '/lib/ansible/modules/' in path or '/lib/ansible/module_utils/' in path
 | |
| 
 | |
| 
 | |
| class AnsibleBlacklistChecker(BaseChecker):
 | |
|     """Checker for blacklisted imports and functions."""
 | |
|     __implements__ = (IAstroidChecker,)
 | |
| 
 | |
|     name = 'blacklist'
 | |
| 
 | |
|     BAD_IMPORT = 'ansible-bad-import'
 | |
|     BAD_IMPORT_FROM = 'ansible-bad-import-from'
 | |
|     BAD_FUNCTION = 'ansible-bad-function'
 | |
|     BAD_MODULE_IMPORT = 'ansible-bad-module-import'
 | |
| 
 | |
|     msgs = dict(
 | |
|         E5101=('Import %s instead of %s',
 | |
|                BAD_IMPORT,
 | |
|                'Identifies imports which should not be used.'),
 | |
|         E5102=('Import %s from %s instead of %s',
 | |
|                BAD_IMPORT_FROM,
 | |
|                'Identifies imports which should not be used.'),
 | |
|         E5103=('Call %s instead of %s',
 | |
|                BAD_FUNCTION,
 | |
|                'Identifies functions which should not be used.'),
 | |
|         E5104=('Import external package or ansible.module_utils not %s',
 | |
|                BAD_MODULE_IMPORT,
 | |
|                'Identifies imports which should not be used.'),
 | |
|     )
 | |
| 
 | |
|     blacklist_imports = dict(
 | |
|         # Additional imports that we may want to start checking:
 | |
|         # boto=BlacklistEntry('boto3', modules_only=True),
 | |
|         # requests=BlacklistEntry('ansible.module_utils.urls', modules_only=True),
 | |
|         # urllib=BlacklistEntry('ansible.module_utils.urls', modules_only=True),
 | |
| 
 | |
|         # see https://docs.python.org/2/library/urllib2.html
 | |
|         urllib2=BlacklistEntry('ansible.module_utils.urls',
 | |
|                                ignore_paths=(
 | |
|                                    '/lib/ansible/module_utils/urls.py',
 | |
|                                )),
 | |
| 
 | |
|         # see https://docs.python.org/3.7/library/collections.abc.html
 | |
|         collections=BlacklistEntry('ansible.module_utils.common._collections_compat',
 | |
|                                    ignore_paths=(
 | |
|                                        '/lib/ansible/module_utils/common/_collections_compat.py',
 | |
|                                    ),
 | |
|                                    names=(
 | |
|                                        'MappingView',
 | |
|                                        'ItemsView',
 | |
|                                        'KeysView',
 | |
|                                        'ValuesView',
 | |
|                                        'Mapping', 'MutableMapping',
 | |
|                                        'Sequence', 'MutableSequence',
 | |
|                                        'Set', 'MutableSet',
 | |
|                                        'Container',
 | |
|                                        'Hashable',
 | |
|                                        'Sized',
 | |
|                                        'Callable',
 | |
|                                        'Iterable',
 | |
|                                        'Iterator',
 | |
|                                    )),
 | |
|     )
 | |
| 
 | |
|     blacklist_functions = {
 | |
|         # see https://docs.python.org/2/library/tempfile.html#tempfile.mktemp
 | |
|         'tempfile.mktemp': BlacklistEntry('tempfile.mkstemp'),
 | |
| 
 | |
|         'sys.exit': BlacklistEntry('exit_json or fail_json',
 | |
|                                    ignore_paths=(
 | |
|                                        '/lib/ansible/module_utils/basic.py',
 | |
|                                        '/lib/ansible/modules/utilities/logic/async_wrapper.py',
 | |
|                                        '/lib/ansible/module_utils/common/removed.py',
 | |
|                                    ),
 | |
|                                    modules_only=True),
 | |
|     }
 | |
| 
 | |
|     def visit_import(self, node):
 | |
|         """
 | |
|         :type node: astroid.node_classes.Import
 | |
|         """
 | |
|         for name in node.names:
 | |
|             self._check_import(node, name[0])
 | |
| 
 | |
|     def visit_importfrom(self, node):
 | |
|         """
 | |
|         :type node: astroid.node_classes.ImportFrom
 | |
|         """
 | |
|         self._check_importfrom(node, node.modname, node.names)
 | |
| 
 | |
|     def visit_attribute(self, node):
 | |
|         """
 | |
|         :type node: astroid.node_classes.Attribute
 | |
|         """
 | |
|         last_child = node.last_child()
 | |
| 
 | |
|         # this is faster than using type inference and will catch the most common cases
 | |
|         if not isinstance(last_child, astroid.node_classes.Name):
 | |
|             return
 | |
| 
 | |
|         module = last_child.name
 | |
| 
 | |
|         entry = self.blacklist_imports.get(module)
 | |
| 
 | |
|         if entry and entry.names:
 | |
|             if entry.applies_to(self.linter.current_file, node.attrname):
 | |
|                 self.add_message(self.BAD_IMPORT_FROM, args=(node.attrname, entry.alternative, module), node=node)
 | |
| 
 | |
|     def visit_call(self, node):
 | |
|         """
 | |
|         :type node: astroid.node_classes.Call
 | |
|         """
 | |
|         try:
 | |
|             for i in node.func.inferred():
 | |
|                 func = None
 | |
| 
 | |
|                 if isinstance(i, astroid.scoped_nodes.FunctionDef) and isinstance(i.parent, astroid.scoped_nodes.Module):
 | |
|                     func = '%s.%s' % (i.parent.name, i.name)
 | |
| 
 | |
|                 if not func:
 | |
|                     continue
 | |
| 
 | |
|                 entry = self.blacklist_functions.get(func)
 | |
| 
 | |
|                 if entry and entry.applies_to(self.linter.current_file):
 | |
|                     self.add_message(self.BAD_FUNCTION, args=(entry.alternative, func), node=node)
 | |
|         except astroid.exceptions.InferenceError:
 | |
|             pass
 | |
| 
 | |
|     def _check_import(self, node, modname):
 | |
|         """
 | |
|         :type node: astroid.node_classes.Import
 | |
|         :type modname: str
 | |
|         """
 | |
|         self._check_module_import(node, modname)
 | |
| 
 | |
|         entry = self.blacklist_imports.get(modname)
 | |
| 
 | |
|         if not entry:
 | |
|             return
 | |
| 
 | |
|         if entry.applies_to(self.linter.current_file):
 | |
|             self.add_message(self.BAD_IMPORT, args=(entry.alternative, modname), node=node)
 | |
| 
 | |
|     def _check_importfrom(self, node, modname, names):
 | |
|         """
 | |
|         :type node: astroid.node_classes.ImportFrom
 | |
|         :type modname: str
 | |
|         :type names:  list[str[
 | |
|         """
 | |
|         self._check_module_import(node, modname)
 | |
| 
 | |
|         entry = self.blacklist_imports.get(modname)
 | |
| 
 | |
|         if not entry:
 | |
|             return
 | |
| 
 | |
|         for name in names:
 | |
|             if entry.applies_to(self.linter.current_file, name[0]):
 | |
|                 self.add_message(self.BAD_IMPORT_FROM, args=(name[0], entry.alternative, modname), node=node)
 | |
| 
 | |
|     def _check_module_import(self, node, modname):
 | |
|         """
 | |
|         :type node: astroid.node_classes.Import | astroid.node_classes.ImportFrom
 | |
|         :type modname: str
 | |
|         """
 | |
|         if not is_module_path(self.linter.current_file):
 | |
|             return
 | |
| 
 | |
|         if modname == 'ansible.module_utils' or modname.startswith('ansible.module_utils.'):
 | |
|             return
 | |
| 
 | |
|         if modname == 'ansible' or modname.startswith('ansible.'):
 | |
|             self.add_message(self.BAD_MODULE_IMPORT, args=(modname,), node=node)
 | |
| 
 | |
| 
 | |
| def register(linter):
 | |
|     """required method to auto register this checker """
 | |
|     linter.register_checker(AnsibleBlacklistChecker(linter))
 |