관리-도구
편집 파일: __init__.py
from typing import List from pylint.exceptions import UnknownMessageError from pylint.lint import PyLinter def get_class(module_name, kls): parts = kls.split(".") m = __import__(module_name) for mp in module_name.split(".")[1:]: m = getattr(m, mp) klass = getattr(m, parts[0]) return klass class NoSuchChecker(Exception): def __init__(self, checker_class): self.message = "Checker class %s was not found" % checker_class def __repr__(self): return self.message def get_checker(linter: PyLinter, checker_class): for checker in linter.get_checkers(): if isinstance(checker, checker_class): return checker raise NoSuchChecker(checker_class) def augment_visit(linter: PyLinter, checker_method, augmentation): """ Augmenting a visit enables additional errors to be raised (although that case is better served using a new checker) or to suppress all warnings in certain circumstances. Augmenting functions should accept a 'chain' function, which runs the checker method and possibly any other augmentations, and secondly an Astroid node. "chain()" can be called at any point to trigger the continuation of other checks, or not at all to prevent any further checking. """ try: checker = get_checker(linter, checker_method.__self__.__class__) except AttributeError: checker = get_checker( linter, get_class(checker_method.__module__, checker_method.__qualname__) ) old_method = getattr(checker, checker_method.__name__) setattr(checker, checker_method.__name__, AugmentFunc(old_method, augmentation)) class AugmentFunc: def __init__(self, old_method, augmentation_func): self.old_method = old_method self.augmentation_func = augmentation_func def __call__(self, node): self.augmentation_func(Chain(self.old_method, node), node) class Chain: def __init__(self, old_method, node): self.old_method = old_method self.node = node def __call__(self): self.old_method(self.node) class Suppress: def __init__(self, linter): self._linter = linter self._suppress = [] self._messages_to_append = [] def __enter__(self): self._orig_add_message = self._linter.add_message self._linter.add_message = self.add_message return self def add_message(self, *args, **kwargs): self._messages_to_append.append((args, kwargs)) def suppress(self, *symbols): for symbol in symbols: self._suppress.append(symbol) def __exit__(self, exc_type, exc_val, exc_tb): self._linter.add_message = self._orig_add_message for to_append_args, to_append_kwargs in self._messages_to_append: if to_append_args[0] in self._suppress: continue self._linter.add_message(*to_append_args, **to_append_kwargs) def suppress_message(linter: PyLinter, checker_method, message_id_or_symbol, test_func): """ This wrapper allows the suppression of a message if the supplied test function returns True. It is useful to prevent one particular message from being raised in one particular case, while leaving the rest of the messages intact. """ augment_visit( linter, checker_method, DoSuppress(linter, message_id_or_symbol, test_func) ) class DoSuppress: def __init__(self, linter: PyLinter, message_id_or_symbol, test_func): self.linter = linter self.message_id_or_symbol = message_id_or_symbol self.test_func = test_func def __call__(self, chain, node): with Suppress(self.linter) as s: if self.test_func(node): s.suppress(*self.symbols) chain() @property def symbols(self) -> List: # At some point, pylint started preferring message symbols to message IDs. # However, this is not done consistently or uniformly # - occasionally there are some message IDs with no matching symbols. # We try to work around this here by suppressing both the ID and the symbol. # This also gives us compatability with a broader range of pylint versions. # Similarly, between version 1.2 and 1.3 changed where the messages are stored # - see: # https://bitbucket.org/logilab/pylint/commits/0b67f42799bed08aebb47babdc9fb0e761efc4ff#chg-reporters/__init__.py # Therefore here, we try the new attribute name, and fall back to the old # version for compatability with <=1.2 and >=1.3 try: pylint_messages = self.get_message_definitions(self.message_id_or_symbol) the_symbols = [ symbol for pylint_message in pylint_messages for symbol in (pylint_message.msgid, pylint_message.symbol) if symbol is not None ] except UnknownMessageError: # This can happen due to mismatches of pylint versions and plugin # expectations of available messages the_symbols = [self.message_id_or_symbol] return the_symbols def get_message_definitions(self, message_id_or_symbol): msgs_store = getattr(self.linter, "msgs_store", self.linter) if hasattr(msgs_store, "check_message_id"): return [msgs_store.check_message_id(message_id_or_symbol)] # pylint 2.0 renamed check_message_id to get_message_definition in: # https://github.com/PyCQA/pylint/commit/5ccbf9eaa54c0c302c9180bdfb745566c16e416d elif hasattr(msgs_store, "get_message_definition"): return [msgs_store.get_message_definition(message_id_or_symbol)] # pylint 2.3.0 renamed get_message_definition to get_message_definitions in: # https://github.com/PyCQA/pylint/commit/da67a9da682e51844fbc674229ff6619eb9c816a elif hasattr(msgs_store, "get_message_definitions"): return msgs_store.get_message_definitions(message_id_or_symbol) else: msg = ( "pylint.utils.MessagesStore does not have a " "get_message_definition(s) method" ) raise ValueError(msg)