2016-01-15 09:58:38 -06:00
|
|
|
#
|
|
|
|
# Copyright (C) 2015 FreeIPA Contributors see COPYING for license
|
|
|
|
#
|
|
|
|
|
|
|
|
from __future__ import print_function
|
|
|
|
|
|
|
|
import copy
|
2017-02-14 02:58:44 -06:00
|
|
|
import os.path
|
2016-01-15 09:58:38 -06:00
|
|
|
import sys
|
2017-02-17 01:39:54 -06:00
|
|
|
import textwrap
|
2016-01-15 09:58:38 -06:00
|
|
|
|
2017-02-17 01:39:54 -06:00
|
|
|
from astroid import MANAGER, register_module_extender
|
2016-01-15 09:58:38 -06:00
|
|
|
from astroid import scoped_nodes
|
2017-02-14 02:58:44 -06:00
|
|
|
from pylint.checkers import BaseChecker
|
|
|
|
from pylint.checkers.utils import check_messages
|
|
|
|
from pylint.interfaces import IAstroidChecker
|
2017-02-17 01:39:54 -06:00
|
|
|
from astroid.builder import AstroidBuilder
|
2016-01-15 09:58:38 -06:00
|
|
|
|
|
|
|
|
|
|
|
def register(linter):
|
2017-02-14 02:58:44 -06:00
|
|
|
linter.register_checker(IPAChecker(linter))
|
2016-01-15 09:58:38 -06:00
|
|
|
|
|
|
|
|
|
|
|
def _warning_already_exists(cls, member):
|
|
|
|
print(
|
|
|
|
"WARNING: member '{member}' in '{cls}' already exists".format(
|
|
|
|
cls="{}.{}".format(cls.root().name, cls.name), member=member),
|
|
|
|
file=sys.stderr
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def fake_class(name_or_class_obj, members=()):
|
|
|
|
if isinstance(name_or_class_obj, scoped_nodes.Class):
|
|
|
|
cl = name_or_class_obj
|
|
|
|
else:
|
|
|
|
cl = scoped_nodes.Class(name_or_class_obj, None)
|
|
|
|
|
|
|
|
for m in members:
|
|
|
|
if isinstance(m, str):
|
|
|
|
if m in cl.locals:
|
|
|
|
_warning_already_exists(cl, m)
|
|
|
|
else:
|
|
|
|
cl.locals[m] = [scoped_nodes.Class(m, None)]
|
|
|
|
elif isinstance(m, dict):
|
|
|
|
for key, val in m.items():
|
|
|
|
assert isinstance(key, str), "key must be string"
|
|
|
|
if key in cl.locals:
|
|
|
|
_warning_already_exists(cl, key)
|
|
|
|
fake_class(cl.locals[key], val)
|
|
|
|
else:
|
|
|
|
cl.locals[key] = [fake_class(key, val)]
|
|
|
|
else:
|
|
|
|
# here can be used any astroid type
|
|
|
|
if m.name in cl.locals:
|
|
|
|
_warning_already_exists(cl, m.name)
|
|
|
|
else:
|
|
|
|
cl.locals[m.name] = [copy.copy(m)]
|
|
|
|
return cl
|
|
|
|
|
|
|
|
|
|
|
|
fake_backend = {'Backend': [
|
|
|
|
{'wsgi_dispatch': ['mount']},
|
|
|
|
]}
|
|
|
|
|
|
|
|
NAMESPACE_ATTRS = ['Command', 'Object', 'Method', fake_backend, 'Updater',
|
|
|
|
'Advice']
|
|
|
|
fake_api_env = {'env': [
|
|
|
|
'host',
|
|
|
|
'realm',
|
|
|
|
'session_auth_duration',
|
|
|
|
'session_duration_type',
|
2017-06-05 08:50:22 -05:00
|
|
|
'kinit_lifetime',
|
2016-01-15 09:58:38 -06:00
|
|
|
]}
|
|
|
|
|
|
|
|
# this is due ipaserver.rpcserver.KerberosSession where api is undefined
|
|
|
|
fake_api = {'api': [fake_api_env] + NAMESPACE_ATTRS}
|
|
|
|
|
|
|
|
_LOGGING_ATTRS = ['debug', 'info', 'warning', 'error', 'exception',
|
|
|
|
'critical']
|
|
|
|
LOGGING_ATTRS = [
|
|
|
|
{'log': _LOGGING_ATTRS},
|
|
|
|
] + _LOGGING_ATTRS
|
|
|
|
|
|
|
|
# 'class': ['generated', 'properties']
|
|
|
|
ipa_class_members = {
|
|
|
|
# Python standard library & 3rd party classes
|
|
|
|
'socket._socketobject': ['sendall'],
|
|
|
|
|
|
|
|
# IPA classes
|
|
|
|
'ipalib.base.NameSpace': [
|
|
|
|
'add',
|
|
|
|
'mod',
|
|
|
|
'del',
|
|
|
|
'show',
|
|
|
|
'find'
|
|
|
|
],
|
|
|
|
'ipalib.cli.Collector': ['__options'],
|
|
|
|
'ipalib.config.Env': [
|
|
|
|
{'__d': ['get']},
|
|
|
|
{'__done': ['add']},
|
|
|
|
'xmlrpc_uri',
|
|
|
|
'validate_api',
|
|
|
|
'startup_traceback',
|
2017-02-21 07:24:51 -06:00
|
|
|
'verbose',
|
|
|
|
'server',
|
|
|
|
{'domain': dir(str)},
|
2016-01-15 09:58:38 -06:00
|
|
|
] + LOGGING_ATTRS,
|
2016-02-24 11:48:29 -06:00
|
|
|
'ipalib.errors.ACIError': [
|
|
|
|
'info',
|
|
|
|
],
|
|
|
|
'ipalib.errors.ConversionError': [
|
|
|
|
'error',
|
|
|
|
],
|
|
|
|
'ipalib.errors.DatabaseError': [
|
|
|
|
'desc',
|
|
|
|
],
|
|
|
|
'ipalib.errors.NetworkError': [
|
|
|
|
'error',
|
|
|
|
],
|
|
|
|
'ipalib.errors.NotFound': [
|
|
|
|
'reason',
|
|
|
|
],
|
|
|
|
'ipalib.errors.PublicError': [
|
|
|
|
'msg',
|
|
|
|
'strerror',
|
2016-05-25 11:12:57 -05:00
|
|
|
'kw',
|
2016-02-24 11:48:29 -06:00
|
|
|
],
|
|
|
|
'ipalib.errors.SingleMatchExpected': [
|
|
|
|
'found',
|
|
|
|
],
|
|
|
|
'ipalib.errors.SkipPluginModule': [
|
|
|
|
'reason',
|
|
|
|
],
|
|
|
|
'ipalib.errors.ValidationError': [
|
|
|
|
'error',
|
|
|
|
],
|
2016-06-21 07:21:22 -05:00
|
|
|
'ipalib.errors.SchemaUpToDate': [
|
|
|
|
'fingerprint',
|
|
|
|
'ttl',
|
|
|
|
],
|
2016-02-24 11:48:29 -06:00
|
|
|
'ipalib.messages.PublicMessage': [
|
|
|
|
'msg',
|
|
|
|
'strerror',
|
|
|
|
'type',
|
2016-05-25 11:12:57 -05:00
|
|
|
'kw',
|
2016-02-24 11:48:29 -06:00
|
|
|
],
|
2016-01-15 09:58:38 -06:00
|
|
|
'ipalib.parameters.Param': [
|
|
|
|
'cli_name',
|
|
|
|
'cli_short_name',
|
|
|
|
'label',
|
|
|
|
'default',
|
|
|
|
'doc',
|
|
|
|
'required',
|
|
|
|
'multivalue',
|
|
|
|
'primary_key',
|
|
|
|
'normalizer',
|
|
|
|
'default_from',
|
|
|
|
'autofill',
|
|
|
|
'query',
|
|
|
|
'attribute',
|
|
|
|
'include',
|
|
|
|
'exclude',
|
|
|
|
'flags',
|
|
|
|
'hint',
|
|
|
|
'alwaysask',
|
|
|
|
'sortorder',
|
|
|
|
'option_group',
|
2016-06-02 00:31:12 -05:00
|
|
|
'no_convert',
|
2016-06-02 08:58:43 -05:00
|
|
|
'deprecated',
|
2016-01-15 09:58:38 -06:00
|
|
|
],
|
|
|
|
'ipalib.parameters.Bool': [
|
|
|
|
'truths',
|
|
|
|
'falsehoods'],
|
|
|
|
'ipalib.parameters.Data': [
|
|
|
|
'minlength',
|
|
|
|
'maxlength',
|
|
|
|
'length',
|
|
|
|
'pattern',
|
|
|
|
'pattern_errmsg',
|
|
|
|
],
|
|
|
|
'ipalib.parameters.Str': ['noextrawhitespace'],
|
|
|
|
'ipalib.parameters.Password': ['confirm'],
|
|
|
|
'ipalib.parameters.File': ['stdin_if_missing'],
|
|
|
|
'ipalib.parameters.Enum': ['values'],
|
|
|
|
'ipalib.parameters.Number': [
|
|
|
|
'minvalue',
|
|
|
|
'maxvalue',
|
|
|
|
],
|
|
|
|
'ipalib.parameters.Decimal': [
|
|
|
|
'precision',
|
|
|
|
'exponential',
|
|
|
|
'numberclass',
|
|
|
|
],
|
|
|
|
'ipalib.parameters.DNSNameParam': [
|
|
|
|
'only_absolute',
|
|
|
|
'only_relative',
|
|
|
|
],
|
2016-06-23 11:23:00 -05:00
|
|
|
'ipalib.parameters.Principal': [
|
|
|
|
'require_service',
|
|
|
|
],
|
2016-01-15 09:58:38 -06:00
|
|
|
'ipalib.plugable.API': [
|
|
|
|
fake_api_env,
|
|
|
|
] + NAMESPACE_ATTRS + LOGGING_ATTRS,
|
|
|
|
'ipalib.plugable.Plugin': [
|
|
|
|
'Object',
|
|
|
|
'Method',
|
|
|
|
'Updater',
|
|
|
|
'Advice',
|
2017-05-23 11:35:57 -05:00
|
|
|
],
|
2016-02-24 11:48:29 -06:00
|
|
|
'ipalib.util.ForwarderValidationError': [
|
|
|
|
'msg',
|
|
|
|
],
|
2016-04-28 03:30:05 -05:00
|
|
|
'ipaserver.plugins.dns.DNSRecord': [
|
|
|
|
'validatedns',
|
|
|
|
'normalizedns',
|
|
|
|
],
|
2016-01-15 09:58:38 -06:00
|
|
|
'ipaserver.rpcserver.KerberosSession': [
|
|
|
|
fake_api,
|
2017-05-23 11:35:57 -05:00
|
|
|
],
|
2016-01-15 09:58:38 -06:00
|
|
|
'ipatests.test_integration.base.IntegrationTest': [
|
|
|
|
'domain',
|
|
|
|
{'master': [
|
|
|
|
{'config': [
|
|
|
|
{'dirman_password': dir(str)},
|
|
|
|
{'admin_password': dir(str)},
|
|
|
|
{'admin_name': dir(str)},
|
|
|
|
{'dns_forwarder': dir(str)},
|
|
|
|
{'test_dir': dir(str)},
|
|
|
|
{'ad_admin_name': dir(str)},
|
|
|
|
{'ad_admin_password': dir(str)},
|
|
|
|
{'domain_level': dir(str)},
|
|
|
|
]},
|
|
|
|
{'domain': [
|
|
|
|
{'realm': dir(str)},
|
|
|
|
{'name': dir(str)},
|
|
|
|
]},
|
|
|
|
'hostname',
|
|
|
|
'ip',
|
|
|
|
'collect_log',
|
|
|
|
{'run_command': [
|
|
|
|
{'stdout_text': dir(str)},
|
|
|
|
'stderr_text',
|
|
|
|
'returncode',
|
|
|
|
]},
|
2016-10-14 03:06:27 -05:00
|
|
|
{'transport': ['put_file', 'file_exists']},
|
2016-01-15 09:58:38 -06:00
|
|
|
'put_file_contents',
|
|
|
|
'get_file_contents',
|
2016-04-14 09:39:10 -05:00
|
|
|
'ldap_connect',
|
2016-01-15 09:58:38 -06:00
|
|
|
]},
|
|
|
|
'replicas',
|
|
|
|
'clients',
|
|
|
|
'ad_domains',
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def fix_ipa_classes(cls):
|
|
|
|
class_name_with_module = "{}.{}".format(cls.root().name, cls.name)
|
|
|
|
if class_name_with_module in ipa_class_members:
|
|
|
|
fake_class(cls, ipa_class_members[class_name_with_module])
|
|
|
|
|
|
|
|
MANAGER.register_transform(scoped_nodes.Class, fix_ipa_classes)
|
2017-02-14 02:58:44 -06:00
|
|
|
|
|
|
|
|
2017-02-17 01:39:54 -06:00
|
|
|
def pytest_config_transform():
|
|
|
|
"""pylint.config attribute
|
|
|
|
"""
|
|
|
|
return AstroidBuilder(MANAGER).string_build(textwrap.dedent('''
|
|
|
|
from _pytest.config import get_config
|
|
|
|
config = get_config()
|
|
|
|
'''))
|
|
|
|
|
|
|
|
|
|
|
|
register_module_extender(MANAGER, 'pytest', pytest_config_transform)
|
|
|
|
|
|
|
|
|
2017-02-14 02:58:44 -06:00
|
|
|
class IPAChecker(BaseChecker):
|
|
|
|
__implements__ = IAstroidChecker
|
|
|
|
|
|
|
|
name = 'ipa'
|
|
|
|
msgs = {
|
|
|
|
'W9901': (
|
|
|
|
'Forbidden import %s (can\'t import from %s in %s)',
|
|
|
|
'ipa-forbidden-import',
|
|
|
|
'Used when an forbidden import is detected.',
|
|
|
|
),
|
|
|
|
}
|
|
|
|
options = (
|
|
|
|
(
|
|
|
|
'forbidden-imports',
|
|
|
|
{
|
|
|
|
'default': '',
|
|
|
|
'type': 'csv',
|
|
|
|
'metavar': '<path>[:<module>[:<module>...]][,<path>...]',
|
|
|
|
'help': 'Modules which are forbidden to be imported in the '
|
|
|
|
'given paths',
|
|
|
|
},
|
|
|
|
),
|
|
|
|
)
|
|
|
|
priority = -1
|
|
|
|
|
|
|
|
def open(self):
|
|
|
|
self._dir = os.path.abspath(os.path.dirname(__file__))
|
|
|
|
|
|
|
|
self._forbidden_imports = {self._dir: []}
|
|
|
|
for forbidden_import in self.config.forbidden_imports:
|
|
|
|
forbidden_import = forbidden_import.split(':')
|
|
|
|
path = os.path.join(self._dir, forbidden_import[0])
|
|
|
|
path = os.path.abspath(path)
|
|
|
|
modules = forbidden_import[1:]
|
|
|
|
self._forbidden_imports[path] = modules
|
|
|
|
|
|
|
|
self._forbidden_imports_stack = []
|
|
|
|
|
|
|
|
def _get_forbidden_import_rule(self, node):
|
2017-05-26 14:21:34 -05:00
|
|
|
path = node.path
|
2017-02-14 02:58:44 -06:00
|
|
|
if path:
|
|
|
|
path = os.path.abspath(path)
|
|
|
|
while path.startswith(self._dir):
|
|
|
|
if path in self._forbidden_imports:
|
|
|
|
return path
|
|
|
|
path = os.path.dirname(path)
|
|
|
|
return self._dir
|
|
|
|
|
|
|
|
def visit_module(self, node):
|
|
|
|
self._forbidden_imports_stack.append(
|
|
|
|
self._get_forbidden_import_rule(node))
|
|
|
|
|
|
|
|
def leave_module(self, node):
|
|
|
|
self._forbidden_imports_stack.pop()
|
|
|
|
|
|
|
|
def _check_forbidden_imports(self, node, names):
|
|
|
|
path = self._forbidden_imports_stack[-1]
|
|
|
|
relpath = os.path.relpath(path, self._dir)
|
|
|
|
modules = self._forbidden_imports[path]
|
|
|
|
for module in modules:
|
|
|
|
module_prefix = module + '.'
|
|
|
|
for name in names:
|
|
|
|
if name == module or name.startswith(module_prefix):
|
|
|
|
self.add_message('ipa-forbidden-import',
|
|
|
|
args=(name, module, relpath), node=node)
|
|
|
|
|
|
|
|
@check_messages('ipa-forbidden-import')
|
|
|
|
def visit_import(self, node):
|
|
|
|
names = [n[0] for n in node.names]
|
|
|
|
self._check_forbidden_imports(node, names)
|
|
|
|
|
|
|
|
@check_messages('ipa-forbidden-import')
|
|
|
|
def visit_importfrom(self, node):
|
|
|
|
names = ['{}.{}'.format(node.modname, n[0]) for n in node.names]
|
|
|
|
self._check_forbidden_imports(node, names)
|